2021-08-10 20:12:08 -07:00
|
|
|
// Package p contains an HTTP Cloud Function.
|
|
|
|
package p
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2021-08-23 23:01:00 -07:00
|
|
|
"html"
|
2021-08-10 20:12:08 -07:00
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
2021-11-16 10:45:15 -08:00
|
|
|
|
|
|
|
"cloud.google.com/go/bigtable"
|
2021-08-10 20:12:08 -07:00
|
|
|
)
|
|
|
|
|
2021-09-21 09:08:07 -07:00
|
|
|
// fetch a single row by the row key
|
2021-08-10 20:12:08 -07:00
|
|
|
func ReadRow(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// Set CORS headers for the preflight request
|
|
|
|
if r.Method == http.MethodOptions {
|
|
|
|
w.Header().Set("Access-Control-Allow-Origin", "*")
|
|
|
|
w.Header().Set("Access-Control-Allow-Methods", "POST")
|
|
|
|
w.Header().Set("Access-Control-Allow-Headers", "Content-Type")
|
|
|
|
w.Header().Set("Access-Control-Max-Age", "3600")
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Set CORS headers for the main request.
|
|
|
|
w.Header().Set("Access-Control-Allow-Origin", "*")
|
|
|
|
|
2021-09-21 09:08:07 -07:00
|
|
|
var emitterChain, emitterAddress, sequence, rowKey string
|
2021-08-10 20:12:08 -07:00
|
|
|
|
|
|
|
// allow GET requests with querystring params, or POST requests with json body.
|
|
|
|
switch r.Method {
|
|
|
|
case http.MethodGet:
|
|
|
|
queryParams := r.URL.Query()
|
2021-09-21 09:08:07 -07:00
|
|
|
emitterChain = queryParams.Get("emitterChain")
|
|
|
|
emitterAddress = queryParams.Get("emitterAddress")
|
|
|
|
sequence = queryParams.Get("sequence")
|
2021-08-10 20:12:08 -07:00
|
|
|
|
2021-08-23 23:01:00 -07:00
|
|
|
readyCheck := queryParams.Get("readyCheck")
|
|
|
|
if readyCheck != "" {
|
|
|
|
// for running in devnet
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
fmt.Fprint(w, html.EscapeString("ready"))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-08-10 20:12:08 -07:00
|
|
|
// check for empty values
|
|
|
|
if emitterChain == "" || emitterAddress == "" || sequence == "" {
|
2021-09-21 09:08:07 -07:00
|
|
|
fmt.Fprint(w, "query params ['emitterChain', 'emitterAddress', 'sequence'] cannot be empty")
|
2021-08-10 20:12:08 -07:00
|
|
|
http.Error(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
case http.MethodPost:
|
|
|
|
// declare request body properties
|
|
|
|
var d struct {
|
|
|
|
EmitterChain string `json:"emitterChain"`
|
|
|
|
EmitterAddress string `json:"emitterAddress"`
|
|
|
|
Sequence string `json:"sequence"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// deserialize request body
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&d); err != nil {
|
|
|
|
switch err {
|
|
|
|
case io.EOF:
|
|
|
|
fmt.Fprint(w, "request body required")
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
log.Printf("json.NewDecoder: %v", err)
|
|
|
|
http.Error(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for empty values
|
|
|
|
if d.EmitterChain == "" || d.EmitterAddress == "" || d.Sequence == "" {
|
2021-09-21 09:08:07 -07:00
|
|
|
fmt.Fprint(w, "body values ['emitterChain', 'emitterAddress', 'sequence'] cannot be empty")
|
2021-08-10 20:12:08 -07:00
|
|
|
http.Error(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
2021-09-21 09:08:07 -07:00
|
|
|
emitterChain = d.EmitterChain
|
|
|
|
emitterAddress = d.EmitterAddress
|
|
|
|
sequence = d.Sequence
|
2021-08-10 20:12:08 -07:00
|
|
|
default:
|
|
|
|
http.Error(w, "405 - Method Not Allowed", http.StatusMethodNotAllowed)
|
|
|
|
log.Println("Method Not Allowed")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-09-21 09:08:07 -07:00
|
|
|
// pad sequence to 16 characters
|
|
|
|
if len(sequence) <= 15 {
|
|
|
|
sequence = fmt.Sprintf("%016s", sequence)
|
|
|
|
}
|
|
|
|
// convert chain name to chainID
|
|
|
|
if len(emitterChain) > 1 {
|
|
|
|
chainNameMap := map[string]string{
|
|
|
|
"solana": "1",
|
|
|
|
"ethereum": "2",
|
|
|
|
"terra": "3",
|
|
|
|
"bsc": "4",
|
2021-10-18 08:53:13 -07:00
|
|
|
"polygon": "5",
|
2021-09-21 09:08:07 -07:00
|
|
|
}
|
|
|
|
lowercaseChain := strings.ToLower(emitterChain)
|
|
|
|
if _, ok := chainNameMap[lowercaseChain]; ok {
|
|
|
|
emitterChain = chainNameMap[lowercaseChain]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rowKey = emitterChain + ":" + emitterAddress + ":" + sequence
|
|
|
|
|
2021-11-16 10:45:15 -08:00
|
|
|
row, err := tbl.ReadRow(r.Context(), rowKey, bigtable.RowFilter(bigtable.LatestNFilter(1)))
|
2021-08-10 20:12:08 -07:00
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "Error reading rows", http.StatusInternalServerError)
|
|
|
|
log.Printf("tbl.ReadRows(): %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if row == nil {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
log.Printf("did not find row for key %v", rowKey)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-10-29 01:08:36 -07:00
|
|
|
details := makeDetails(row)
|
|
|
|
jsonBytes, err := json.Marshal(details)
|
2021-08-10 20:12:08 -07:00
|
|
|
if err != nil {
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
w.Write([]byte(err.Error()))
|
|
|
|
log.Println(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
w.Write(jsonBytes)
|
|
|
|
}
|