-
Notifications
You must be signed in to change notification settings - Fork 19
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* migration files * go files for blocks dir * write queries * implement observed-block controller and route * implement unprocessed block apis * integrate go apis to listener api.ts * implement get observed block api and integrate * change insertUnprocessedBlock api to batch insert array of blocks * revamp listener.ts * revamp state.ts * rearrange execution sequence in state.add() * generalize block payload validation * define separate logPrefix var in latestJob * refactor latestJob creating unprocessed blocks logic * update upsert observed_block query to disallow decreasing observedBlock value * return void in delete api * fix logPrefix on exception in latestJob * fallthrough clear option in state.add() * throw error if processing event fails * move checking observedBlock existence in db to controller * replace template literal with simple string
- Loading branch information
Showing
11 changed files
with
419 additions
and
89 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,117 @@ | ||
package blocks | ||
|
||
import ( | ||
"bisonai.com/orakl/api/utils" | ||
"github.com/go-playground/validator/v10" | ||
"github.com/gofiber/fiber/v2" | ||
) | ||
|
||
type BlockModel struct { | ||
Service string `db:"service" json:"service" validate:"required"` | ||
BlockNumber int64 `db:"block_number" json:"blockNumber" validate:"isZeroOrPositive"` | ||
} | ||
|
||
type BlocksModel struct { | ||
Service string `db:"service" json:"service" validate:"required"` | ||
Blocks []int64 `db:"blocks" json:"blocks" validate:"dive,isZeroOrPositive"` | ||
} | ||
|
||
var validate *validator.Validate | ||
|
||
func init() { | ||
validate = validator.New() | ||
validate.RegisterValidation("isZeroOrPositive", func(fl validator.FieldLevel) bool { | ||
return fl.Field().Int() >= 0 | ||
}) | ||
} | ||
|
||
func validateBlockPayload(payload interface{}) error { | ||
return validate.Struct(payload) | ||
} | ||
|
||
func getObservedBlock(c *fiber.Ctx) error { | ||
service := c.Query("service") | ||
if service == "" { | ||
return fiber.NewError(fiber.StatusBadRequest, "service is required") | ||
} | ||
result, err := utils.QueryRow[BlockModel](c, GetObservedBlock, map[string]any{ | ||
"service": service, | ||
}) | ||
if err != nil { | ||
return err | ||
} | ||
if result.Service == "" { | ||
return c.JSON(nil) | ||
} | ||
|
||
return c.JSON(result) | ||
} | ||
|
||
func upsertObservedBlock(c *fiber.Ctx) error { | ||
payload := new(BlockModel) | ||
if err := c.BodyParser(payload); err != nil { | ||
return err | ||
} | ||
|
||
if err := validateBlockPayload(payload); err != nil { | ||
return err | ||
} | ||
|
||
result, err := utils.QueryRow[BlockModel](c, UpsertObservedBlock, map[string]any{ | ||
"service": payload.Service, | ||
"block_number": payload.BlockNumber, | ||
}) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
return c.JSON(result) | ||
} | ||
|
||
func getUnprocessedBlocks(c *fiber.Ctx) error { | ||
service := c.Query("service") | ||
if service == "" { | ||
return fiber.NewError(fiber.StatusBadRequest, "service is required") | ||
} | ||
result, err := utils.QueryRows[BlockModel](c, GetUnprocessedBlocks, map[string]any{ | ||
"service": service, | ||
}) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
return c.JSON(result) | ||
} | ||
|
||
func insertUnprocessedBlocks(c *fiber.Ctx) error { | ||
payload := new(BlocksModel) | ||
if err := c.BodyParser(payload); err != nil { | ||
return err | ||
} | ||
|
||
if err := validateBlockPayload(payload); err != nil { | ||
return err | ||
} | ||
|
||
result, err := utils.QueryRows[BlocksModel](c, GenerateInsertBlocksQuery(payload.Blocks, payload.Service), map[string]any{}) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
return c.JSON(result) | ||
} | ||
|
||
func deleteUnprocessedBlock(c *fiber.Ctx) error { | ||
service := c.Params("service") | ||
blockNumber := c.Params("blockNumber") | ||
|
||
result, err := utils.QueryRow[BlockModel](c, DeleteUnprocessedBlock, map[string]any{ | ||
"service": service, | ||
"block_number": blockNumber, | ||
}) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
return c.JSON(result) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
package blocks | ||
|
||
import ( | ||
"fmt" | ||
"strings" | ||
) | ||
|
||
const ( | ||
// get observedBlock given service | ||
GetObservedBlock = ` | ||
SELECT * FROM observed_blocks | ||
WHERE service = @service; | ||
` | ||
|
||
// upsert to observed_blocks given service and block_number | ||
UpsertObservedBlock = ` | ||
INSERT INTO observed_blocks (service, block_number) | ||
VALUES (@service, @block_number) | ||
ON CONFLICT (service) DO UPDATE SET block_number = GREATEST(observed_blocks.block_number, EXCLUDED.block_number) | ||
RETURNING *; | ||
` | ||
|
||
// get all unprocessed blocks given service | ||
GetUnprocessedBlocks = ` | ||
SELECT * FROM unprocessed_blocks | ||
WHERE service = @service; | ||
` | ||
|
||
// delete unprocessed block given service and block_number | ||
DeleteUnprocessedBlock = ` | ||
DELETE FROM unprocessed_blocks | ||
WHERE service = @service AND block_number = @block_number | ||
RETURNING *; | ||
` | ||
) | ||
|
||
func GenerateInsertBlocksQuery(blocks []int64, service string) string { | ||
baseQuery := ` | ||
INSERT INTO unprocessed_blocks (service, block_number) | ||
VALUES | ||
` | ||
onConflict := ` | ||
ON CONFLICT (service, block_number) DO NOTHING; | ||
` | ||
values := make([]string, 0, len(blocks)) | ||
for _, block := range blocks { | ||
values = append(values, fmt.Sprintf("('%s', %d)", service, block)) | ||
} | ||
|
||
return baseQuery + strings.Join(values, ",") + onConflict | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
package blocks | ||
|
||
import ( | ||
"github.com/gofiber/fiber/v2" | ||
) | ||
|
||
func Routes(router fiber.Router) { | ||
blocks := router.Group("/blocks") | ||
|
||
blocks.Get("/observed", getObservedBlock) | ||
blocks.Post("/observed", upsertObservedBlock) | ||
blocks.Post("/unprocessed", insertUnprocessedBlocks) | ||
blocks.Get("/unprocessed", getUnprocessedBlocks) | ||
blocks.Delete("/unprocessed/:service/:blockNumber", deleteUnprocessedBlock) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
DROP TABLE IF EXISTS "observed_blocks"; | ||
DROP TABLE IF EXISTS "unprocessed_blocks"; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
CREATE TABLE IF NOT EXISTS "observed_blocks" ( | ||
service TEXT NOT NULL UNIQUE, | ||
block_number BIGINT NOT NULL | ||
); | ||
|
||
CREATE TABLE IF NOT EXISTS "unprocessed_blocks" ( | ||
service TEXT NOT NULL, | ||
block_number BIGINT NOT NULL, | ||
UNIQUE (service, block_number) | ||
); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.