package monero import ( "fmt" "slices" "strings" "xmr-remote-nodes/internal/database" "github.com/google/uuid" ) const ProberAPIKey = "X-Prober-Api-Key" // HTTP header key type ProberRepository interface { Add(name string) (Prober, error) Edit(id int, name string) error Probers(QueryProbers) ([]Prober, error) CheckApi(key string) (Prober, error) Delete(id int) error } type ProberRepo struct { db *database.DB } type Prober struct { ID int64 `json:"id" db:"id"` Name string `json:"name" db:"name"` ApiKey uuid.UUID `json:"api_key" db:"api_key"` LastSubmitTs int64 `json:"last_submit_ts" db:"last_submit_ts"` } // Initializes a new ProberRepository // // NOTE: This "prober" is different with "probe" which is used to fetch a new job func NewProber() ProberRepository { return &ProberRepo{db: database.GetDB()} } // Add a new prober machine func (repo *ProberRepo) Add(name string) (Prober, error) { apiKey := uuid.New() query := ` INSERT INTO tbl_prober ( name, api_key, last_submit_ts ) VALUES ( ?, ?, ? )` _, err := repo.db.Exec(query, name, apiKey, 0) if err != nil { return Prober{}, err } return Prober{Name: name, ApiKey: apiKey}, nil } // Edit an existing prober func (repo *ProberRepo) Edit(id int, name string) error { query := `UPDATE tbl_prober SET name = ? WHERE id = ?` res, err := repo.db.Exec(query, name, id) if err != nil { return err } row, err := res.RowsAffected() if err != nil { return err } if row == 0 { return fmt.Errorf("no rows affected") } return err } // Delete an existing prober func (repo *ProberRepo) Delete(id int) error { res, err := repo.db.Exec(`DELETE FROM tbl_prober WHERE id = ?`, id) if err != nil { return err } row, err := res.RowsAffected() if err != nil { return err } if row == 0 { return fmt.Errorf("no rows affected") } return err } type QueryProbers struct { Search string SortBy string SortDirection string } func (repo *ProberRepo) Probers(q QueryProbers) ([]Prober, error) { queryParams := []interface{}{} whereQueries := []string{} where := "" if q.Search != "" { whereQueries = append(whereQueries, "(name LIKE ? OR api_key LIKE ?)") queryParams = append(queryParams, "%"+q.Search+"%") queryParams = append(queryParams, "%"+q.Search+"%") } if len(whereQueries) > 0 { where = "WHERE " + strings.Join(whereQueries, " AND ") } var probers []Prober allowedSort := []string{"id", "last_submit_ts"} sortBy := "last_submit_ts" if slices.Contains(allowedSort, q.SortBy) { sortBy = q.SortBy } sortDirection := "DESC" if q.SortDirection == "asc" { sortDirection = "ASC" } query := fmt.Sprintf(` SELECT id, name, api_key, last_submit_ts FROM tbl_prober %s -- where clause if any ORDER BY %s %s`, where, sortBy, sortDirection) row, err := repo.db.Query(query, queryParams...) if err != nil { return probers, err } defer row.Close() for row.Next() { var p Prober err = row.Scan(&p.ID, &p.Name, &p.ApiKey, &p.LastSubmitTs) if err != nil { return probers, err } probers = append(probers, p) } return probers, nil } func (repo *ProberRepo) CheckApi(key string) (Prober, error) { var p Prober query := ` SELECT id, name, api_key, last_submit_ts FROM tbl_prober WHERE api_key = ? LIMIT 1` err := repo.db.QueryRow(query, key).Scan(&p.ID, &p.Name, &p.ApiKey, &p.LastSubmitTs) return p, err }