diff --git a/api/v1/inference_tasks/create_task_test.go b/api/v1/inference_tasks/create_task_test.go index 2a47a00..62f22a2 100644 --- a/api/v1/inference_tasks/create_task_test.go +++ b/api/v1/inference_tasks/create_task_test.go @@ -16,9 +16,9 @@ import ( func TestCreateTaskBeforeBlockchainConfirmation(t *testing.T) { - task := v1.PrepareRandomTask() + task := tests.PrepareRandomTask() - _, privateKeys, err := v1.PrepareAccounts() + _, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare account error") timestamp, signature, err := v1.SignData(task, privateKeys[0]) @@ -31,10 +31,10 @@ func TestCreateTaskBeforeBlockchainConfirmation(t *testing.T) { func TestCreateTaskAfterBlockchainConfirmation(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare account error") - taskInput, task, err := v1.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) + taskInput, task, err := tests.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "prepare task error") timestamp, signature, err := v1.SignData(taskInput, privateKeys[0]) @@ -52,10 +52,10 @@ func TestCreateTaskAfterBlockchainConfirmation(t *testing.T) { } func TestCreateTaskUsingUnauthorizedAccount(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare account error") - taskInput, _, err := v1.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) + taskInput, _, err := tests.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "prepare task error") timestamp, signature, err := v1.SignData(taskInput, privateKeys[1]) @@ -68,10 +68,10 @@ func TestCreateTaskUsingUnauthorizedAccount(t *testing.T) { func TestCreateDuplicateTask(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare account error") - taskInput, task, err := v1.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) + taskInput, task, err := tests.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "prepare task error") timestamp, signature, err := v1.SignData(taskInput, privateKeys[0]) @@ -94,10 +94,10 @@ func TestCreateDuplicateTask(t *testing.T) { } func TestCreateTaskWithMismatchedParamHash(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare account error") - taskInput, _, err := v1.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) + taskInput, _, err := tests.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "prepare task error") oldPrompt := taskInput.Prompt diff --git a/api/v1/inference_tasks/get_result.go b/api/v1/inference_tasks/get_result.go index fc5eee7..8c478d6 100644 --- a/api/v1/inference_tasks/get_result.go +++ b/api/v1/inference_tasks/get_result.go @@ -13,9 +13,8 @@ import ( ) type GetResultInput struct { - ImageNum string `path:"image_num" json:"image_num" description:"Image number" validate:"required"` - SelectedNode string `path:"selected_node" json:"selected_node" description:"Selected nodes" validate:"required"` - TaskId uint64 `path:"task_id" json:"task_id" description:"Task id" validate:"required"` + ImageNum string `path:"image_num" json:"image_num" description:"Image number" validate:"required"` + TaskId uint64 `path:"task_id" json:"task_id" description:"Task id" validate:"required"` } type GetResultInputWithSignature struct { @@ -55,7 +54,7 @@ func GetResult(ctx *gin.Context, in *GetResultInputWithSignature) error { imageFile := filepath.Join( appConfig.DataDir.InferenceTasks, task.GetTaskIdAsString(), - in.SelectedNode, + "results", in.ImageNum+".png", ) diff --git a/api/v1/inference_tasks/get_result_test.go b/api/v1/inference_tasks/get_result_test.go index 0daa6e3..182b966 100644 --- a/api/v1/inference_tasks/get_result_test.go +++ b/api/v1/inference_tasks/get_result_test.go @@ -6,7 +6,6 @@ import ( "h_relay/config" "h_relay/tests" v1 "h_relay/tests/api/v1" - "image/png" "io" "net/http" "net/http/httptest" @@ -17,19 +16,15 @@ import ( ) func TestUnauthorizedGetImage(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare accounts error") - _, task, err := v1.PrepareParamsUploadedTask(addresses, config.GetDB()) + _, task, err := tests.PrepareResultUploadedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "prepare task error") - err = prepareImagesForNode(task.GetTaskIdAsString(), addresses[1]) - assert.Equal(t, nil, err, "create image error") - getResultInput := &inference_tasks.GetResultInput{ - TaskId: task.TaskId, - SelectedNode: addresses[1], - ImageNum: "0", + TaskId: task.TaskId, + ImageNum: "0", } timestamp, signature, err := v1.SignData(getResultInput, privateKeys[1]) @@ -37,7 +32,6 @@ func TestUnauthorizedGetImage(t *testing.T) { r := callGetImageApi( task.GetTaskIdAsString(), - addresses[1], "0", timestamp, signature) @@ -54,19 +48,15 @@ func TestUnauthorizedGetImage(t *testing.T) { func TestGetImage(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare accounts error") - _, task, err := v1.PrepareParamsUploadedTask(addresses, config.GetDB()) + _, task, err := tests.PrepareResultUploadedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "prepare task error") - err = prepareImagesForNode(task.GetTaskIdAsString(), addresses[1]) - assert.Equal(t, nil, err, "create image error") - getResultInput := &inference_tasks.GetResultInput{ - TaskId: task.TaskId, - SelectedNode: addresses[1], - ImageNum: "2", + TaskId: task.TaskId, + ImageNum: "2", } timestamp, signature, err := v1.SignData(getResultInput, privateKeys[0]) @@ -74,7 +64,6 @@ func TestGetImage(t *testing.T) { r := callGetImageApi( task.GetTaskIdAsString(), - addresses[1], "2", timestamp, signature) @@ -85,7 +74,7 @@ func TestGetImage(t *testing.T) { imageFolder := filepath.Join( appConfig.DataDir.InferenceTasks, task.GetTaskIdAsString(), - addresses[1], + "results", ) out, err := os.Create(filepath.Join(imageFolder, "downloaded.png")) @@ -115,12 +104,11 @@ func TestGetImage(t *testing.T) { func callGetImageApi( taskIdStr string, - nodeAddress string, imageNum string, timestamp int64, signature string) *httptest.ResponseRecorder { - endpoint := "/v1/inference_tasks/" + taskIdStr + "/results/" + nodeAddress + "/" + imageNum + endpoint := "/v1/inference_tasks/" + taskIdStr + "/results/" + imageNum query := "?timestamp=" + strconv.FormatInt(timestamp, 10) + "&signature=" + signature req, _ := http.NewRequest("GET", endpoint+query, nil) @@ -129,35 +117,3 @@ func callGetImageApi( return w } - -func prepareImagesForNode(taskIdStr, nodeAddress string) error { - appConfig := config.GetConfig() - - imageFolder := filepath.Join( - appConfig.DataDir.InferenceTasks, - taskIdStr, - nodeAddress, - ) - - if err := os.MkdirAll(imageFolder, os.ModeDir); err != nil { - return err - } - - for i := 0; i < 5; i++ { - filename := filepath.Join(imageFolder, strconv.Itoa(i)+".png") - img := tests.CreateImage() - f, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, 0777) - if err != nil { - return err - } - - if err := png.Encode(f, img); err != nil { - return err - } - - if err := f.Close(); err != nil { - return err - } - } - return nil -} diff --git a/api/v1/inference_tasks/get_task_by_id_test.go b/api/v1/inference_tasks/get_task_by_id_test.go index 573fa4f..734b04a 100644 --- a/api/v1/inference_tasks/get_task_by_id_test.go +++ b/api/v1/inference_tasks/get_task_by_id_test.go @@ -14,10 +14,10 @@ import ( ) func TestGetBlockchainConfirmedTask(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "error preparing accounts") - taskInput, _, err := v1.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) + taskInput, _, err := tests.PrepareBlockchainConfirmedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "error preparing task") getResultInput := inference_tasks.GetTaskInput{TaskId: taskInput.TaskId} @@ -31,10 +31,10 @@ func TestGetBlockchainConfirmedTask(t *testing.T) { } func TestGetParamsUploadedTask(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "error preparing accounts") - taskInput, task, err := v1.PrepareParamsUploadedTask(addresses, config.GetDB()) + taskInput, task, err := tests.PrepareParamsUploadedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "error preparing task") getResultInput := inference_tasks.GetTaskInput{TaskId: taskInput.TaskId} @@ -48,10 +48,10 @@ func TestGetParamsUploadedTask(t *testing.T) { } func TestGetUnauthorizedTask(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "error preparing accounts") - taskInput, _, err := v1.PrepareParamsUploadedTask(addresses, config.GetDB()) + taskInput, _, err := tests.PrepareParamsUploadedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "error preparing task") getResultInput := inference_tasks.GetTaskInput{TaskId: taskInput.TaskId} diff --git a/api/v1/inference_tasks/upload_result.go b/api/v1/inference_tasks/upload_result.go index 788a399..827427d 100644 --- a/api/v1/inference_tasks/upload_result.go +++ b/api/v1/inference_tasks/upload_result.go @@ -2,9 +2,12 @@ package inference_tasks import ( "errors" + "github.com/ethereum/go-ethereum/common/hexutil" "github.com/gin-gonic/gin" + log "github.com/sirupsen/logrus" "gorm.io/gorm" "h_relay/api/v1/response" + "h_relay/blockchain" "h_relay/config" "h_relay/models" "os" @@ -37,7 +40,7 @@ func UploadResult(ctx *gin.Context, in *ResultInputWithSignature) (*response.Res var task models.InferenceTask - if result := config.GetDB().Where(&models.InferenceTask{TaskId: in.TaskId}).Preload("SelectedNodes").First(&task); result.Error != nil { + if result := config.GetDB().Where(&models.InferenceTask{TaskId: in.TaskId}).First(&task); result.Error != nil { if errors.Is(result.Error, gorm.ErrRecordNotFound) { validationErr := response.NewValidationErrorResponse("task_id", "Task not found") return nil, validationErr @@ -46,16 +49,21 @@ func UploadResult(ctx *gin.Context, in *ResultInputWithSignature) (*response.Res } } - var selectedNodeAddress string + resultNode := &models.SelectedNode{ + InferenceTaskID: task.ID, + IsResultSelected: true, + } + + if err := config.GetDB().Where(resultNode).First(resultNode).Error; err != nil { - for _, selectedNode := range task.SelectedNodes { - if selectedNode.NodeAddress == address { - selectedNodeAddress = address - break + if errors.Is(err, gorm.ErrRecordNotFound) { + return nil, response.NewValidationErrorResponse("task_id", "Task not ready") + } else { + return nil, response.NewExceptionResponse(err) } } - if selectedNodeAddress == "" { + if resultNode.NodeAddress != address { validationErr := response.NewValidationErrorResponse("signature", "Signer not allowed") return nil, validationErr } @@ -63,12 +71,47 @@ func UploadResult(ctx *gin.Context, in *ResultInputWithSignature) (*response.Res form, _ := ctx.MultipartForm() files := form.File["images"] + // Check whether the images are correct + var pHashBytes []byte + + for _, file := range files { + + imageFile, err := file.Open() + + if err != nil { + return nil, response.NewExceptionResponse(err) + } + + pHash, err := blockchain.GetPHashForImage(imageFile) + + if err != nil { + return nil, response.NewExceptionResponse(err) + } + + pHashBytes = append(pHashBytes, pHash...) + + err = imageFile.Close() + if err != nil { + return nil, response.NewExceptionResponse(err) + } + } + + uploadedResult := hexutil.Encode(pHashBytes) + + log.Debugln("image compare: result from the blockchain: " + resultNode.Result) + log.Debugln("image compare: result from the uploaded file: " + uploadedResult) + + if resultNode.Result != uploadedResult { + validationErr := response.NewValidationErrorResponse("images", "Wrong images uploaded") + return nil, validationErr + } + appConfig := config.GetConfig() taskWorkspace := appConfig.DataDir.InferenceTasks taskIdStr := task.GetTaskIdAsString() - taskDir := filepath.Join(taskWorkspace, taskIdStr, selectedNodeAddress) + taskDir := filepath.Join(taskWorkspace, taskIdStr, "results") if err = os.MkdirAll(taskDir, os.ModeDir); err != nil { return nil, response.NewExceptionResponse(err) } @@ -76,7 +119,6 @@ func UploadResult(ctx *gin.Context, in *ResultInputWithSignature) (*response.Res fileNum := 0 for _, file := range files { - filename := filepath.Join(taskDir, strconv.Itoa(fileNum)+".png") if err := ctx.SaveUploadedFile(file, filename); err != nil { return nil, response.NewExceptionResponse(err) diff --git a/api/v1/inference_tasks/upload_result_test.go b/api/v1/inference_tasks/upload_result_test.go index 09011a7..9cfb63e 100644 --- a/api/v1/inference_tasks/upload_result_test.go +++ b/api/v1/inference_tasks/upload_result_test.go @@ -19,10 +19,10 @@ import ( ) func TestWrongTaskId(t *testing.T) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare accounts error") - _, _, err = v1.PrepareParamsUploadedTask(addresses, config.GetDB()) + taskInput, _, err := tests.PrepareParamsUploadedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "prepare task error") uploadResultInput := &inference_tasks.ResultInput{ @@ -35,7 +35,7 @@ func TestWrongTaskId(t *testing.T) { timestamp, signature, err := v1.SignData(uploadResultInput, privateKeys[1]) assert.Equal(t, nil, err, "sign data error") - prepareFileForm(t, writer, timestamp, signature) + prepareFileForm(t, taskInput, writer, timestamp, signature) r := callUploadResultApi(666, writer, pr) @@ -79,7 +79,7 @@ func TestSuccessfulUpload(t *testing.T) { v1.AssertEmptySuccessResponse(t, r) for i := 0; i < 5; i++ { - assertFileExists(t, task.TaskId, addresses[2], i) + assertFileExists(t, task.TaskId, i) } }) } @@ -93,10 +93,10 @@ func testUsingAddressNum( task *models.InferenceTask, addresses []string)) { - addresses, privateKeys, err := v1.PrepareAccounts() + addresses, privateKeys, err := tests.PrepareAccounts() assert.Equal(t, nil, err, "prepare accounts error") - _, task, err := v1.PrepareParamsUploadedTask(addresses, config.GetDB()) + taskInput, task, err := tests.PrepareResultUploadedTask(addresses, config.GetDB()) assert.Equal(t, nil, err, "prepare task error") uploadResultInput := &inference_tasks.ResultInput{ @@ -109,7 +109,7 @@ func testUsingAddressNum( timestamp, signature, err := v1.SignData(uploadResultInput, privateKeys[num]) assert.Equal(t, nil, err, "sign data error") - prepareFileForm(t, writer, timestamp, signature) + prepareFileForm(t, taskInput, writer, timestamp, signature) r := callUploadResultApi(task.TaskId, writer, pr) @@ -123,7 +123,7 @@ func testUsingAddressNum( }) } -func prepareFileForm(t *testing.T, writer *multipart.Writer, timestamp int64, signature string) { +func prepareFileForm(t *testing.T, taskInput *inference_tasks.TaskInput, writer *multipart.Writer, timestamp int64, signature string) { go func() { defer func(writer *multipart.Writer) { err := writer.Close() @@ -140,7 +140,7 @@ func prepareFileForm(t *testing.T, writer *multipart.Writer, timestamp int64, si err = writer.WriteField("signature", signature) assert.Equal(t, nil, err, "write signature failed") - for i := 0; i < 5; i++ { + for i := 0; i < taskInput.TaskConfig.NumImages; i++ { part, err := writer.CreateFormFile("images", "test_image_"+strconv.Itoa(i)+".png") if err != nil { t.Error(err) @@ -160,12 +160,12 @@ func prepareFileForm(t *testing.T, writer *multipart.Writer, timestamp int64, si }() } -func assertFileExists(t *testing.T, taskId uint64, selectedNode string, imageNum int) { +func assertFileExists(t *testing.T, taskId uint64, imageNum int) { taskIdStr := strconv.FormatUint(taskId, 10) imageFilename := strconv.Itoa(imageNum) + ".png" appConfig := config.GetConfig() - imageFilePath := filepath.Join(appConfig.DataDir.InferenceTasks, taskIdStr, selectedNode, imageFilename) + imageFilePath := filepath.Join(appConfig.DataDir.InferenceTasks, taskIdStr, "results", imageFilename) _, err := os.Stat(imageFilePath) diff --git a/api/v1/routes.go b/api/v1/routes.go index 92726aa..c874375 100644 --- a/api/v1/routes.go +++ b/api/v1/routes.go @@ -30,7 +30,7 @@ func InitRoutes(r *fizz.Fizz) { fizz.Response("500", "exception", response.ExceptionResponse{}, nil, nil), }, tonic.Handler(inference_tasks.UploadResult, 200)) - tasksGroup.GET("/:task_id/results/:selected_node/:image_num", []fizz.OperationOption{ + tasksGroup.GET("/:task_id/results/:image_num", []fizz.OperationOption{ fizz.Summary("Get the result of the inference task by node address"), fizz.Response("400", "validation errors", response.ValidationErrorResponse{}, nil, nil), }, tonic.Handler(inference_tasks.GetResult, 200)) diff --git a/blockchain/task.go b/blockchain/task.go index 96cd7d8..e0c33f9 100644 --- a/blockchain/task.go +++ b/blockchain/task.go @@ -1,8 +1,11 @@ package blockchain import ( + "bufio" + "bytes" "context" "errors" + "github.com/corona10/goimagehash" "github.com/ethereum/go-ethereum" "github.com/ethereum/go-ethereum/accounts/abi/bind" "github.com/ethereum/go-ethereum/common" @@ -11,6 +14,8 @@ import ( "h_relay/blockchain/bindings" "h_relay/config" "h_relay/models" + "image/png" + "io" "math/big" "math/rand" "strconv" @@ -151,6 +156,28 @@ func GetTaskById(taskId uint64) (*bindings.TaskTaskInfo, error) { return &taskInfo, nil } -func GetPHashForImage() string { - return "" +func GetPHashForImage(reader io.Reader) ([]byte, error) { + image, err := png.Decode(reader) + if err != nil { + return nil, err + } + pHash, err := goimagehash.PerceptionHash(image) + if err != nil { + return nil, err + } + + var b bytes.Buffer + foo := bufio.NewWriter(&b) + + err = pHash.Dump(foo) + if err != nil { + return nil, err + } + + err = foo.Flush() + if err != nil { + return nil, err + } + + return b.Bytes(), nil } diff --git a/tasks/sync_block_test.go b/tasks/sync_block_test.go index 1f854dc..16545e8 100644 --- a/tasks/sync_block_test.go +++ b/tasks/sync_block_test.go @@ -8,7 +8,6 @@ import ( "h_relay/models" "h_relay/tasks" "h_relay/tests" - v1 "h_relay/tests/api/v1" "math/big" "testing" "time" @@ -35,7 +34,7 @@ func TestTaskCreatedAndSuccessOnChain(t *testing.T) { appConfig.Blockchain.Account.Address = addresses[0] appConfig.Blockchain.Account.PrivateKey = privateKeys[0] - taskInput := v1.PrepareRandomTask() + taskInput := tests.PrepareRandomTask() task := &models.InferenceTask{ Prompt: taskInput.Prompt, diff --git a/tests/create_image.go b/tests/image.go similarity index 97% rename from tests/create_image.go rename to tests/image.go index fddda55..1d6acc9 100644 --- a/tests/create_image.go +++ b/tests/image.go @@ -1,8 +1,13 @@ package tests import ( + "h_relay/config" "image" "image/color" + "image/png" + "os" + "path/filepath" + "strconv" ) func CreateImage() *image.RGBA { @@ -47,3 +52,35 @@ func CreateImage() *image.RGBA { func CreateImageDataURL() string { return "" } + +func prepareResultImagesForTask(taskIdStr string, numImages int) error { + appConfig := config.GetConfig() + + imageFolder := filepath.Join( + appConfig.DataDir.InferenceTasks, + taskIdStr, + "results", + ) + + if err := os.MkdirAll(imageFolder, os.ModeDir); err != nil { + return err + } + + for i := 0; i < numImages; i++ { + filename := filepath.Join(imageFolder, strconv.Itoa(i)+".png") + img := CreateImage() + f, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, 0777) + if err != nil { + return err + } + + if err := png.Encode(f, img); err != nil { + return err + } + + if err := f.Close(); err != nil { + return err + } + } + return nil +} diff --git a/tests/api/v1/task.go b/tests/task.go similarity index 66% rename from tests/api/v1/task.go rename to tests/task.go index 91700b7..d1187a4 100644 --- a/tests/api/v1/task.go +++ b/tests/task.go @@ -1,16 +1,23 @@ -package v1 +package tests import ( + "github.com/ethereum/go-ethereum/common/hexutil" log "github.com/sirupsen/logrus" "gorm.io/gorm" "h_relay/api/v1/inference_tasks" + "h_relay/blockchain" + "h_relay/config" "h_relay/models" + "h_relay/tests/api/v1" + "os" + "path" + "strconv" ) func PrepareAccounts() (addresses []string, privateKeys []string, err error) { for i := 0; i < 5; i++ { - address, pk, err := CreateAccount() + address, pk, err := v1.CreateAccount() if err != nil { return nil, nil, err } @@ -121,3 +128,51 @@ func PrepareParamsUploadedTask(addresses []string, db *gorm.DB) (*inference_task return taskInput, task, nil } + +func PrepareResultUploadedTask(addresses []string, db *gorm.DB) (*inference_tasks.TaskInput, *models.InferenceTask, error) { + taskInput, task, err := PrepareParamsUploadedTask(addresses, db) + if err != nil { + return nil, nil, err + } + + // Prepare result images + err = prepareResultImagesForTask(task.GetTaskIdAsString(), taskInput.TaskConfig.NumImages) + if err != nil { + return nil, nil, err + } + + // Calculate the pHash for the images + var result []byte + + appConfig := config.GetConfig() + for i := 0; i < taskInput.TaskConfig.NumImages; i++ { + imageFilename := path.Join( + appConfig.DataDir.InferenceTasks, + task.GetTaskIdAsString(), + "results", + strconv.Itoa(i)+".png", + ) + + imageFile, err := os.Open(imageFilename) + if err != nil { + return nil, nil, err + } + + pHash, err := blockchain.GetPHashForImage(imageFile) + if err != nil { + return nil, nil, err + } + + result = append(result, pHash...) + } + + resultNode := task.SelectedNodes[1] + resultNode.IsResultSelected = true + resultNode.Result = hexutil.Encode(result) + + if err := db.Model(resultNode).Select("IsResultSelected", "Result").Updates(resultNode).Error; err != nil { + return nil, nil, err + } + + return taskInput, task, nil +}