Skip to content

Commit

Permalink
Add Counters
Browse files Browse the repository at this point in the history
This adds a simple way to have namespaced counters per user and get the
global count per namespace. In essence this can be used for a variety of
use-cases, e.g. leaderboards, group challenges, etc. Introducing only
the barest of functionality it will be extended as use-cases arise and
applications have been identified. A variety of dimensions might be
added for future extension to have contextual leadboards, for example
slicing by location, user core data or free form tags.

Counter payload:

```
{
  "value": 123
}
```

In the first iteration we support two endpoints:

* `PUT /me/counters/<counterName>`: Expects the counter payload and sets
the named counter for the current user to the given value.
* `GET /counters/<counterName>`: Returns the sum of all values for all
users for the named counter.
  • Loading branch information
Alexander Simmerl committed Jul 18, 2017
1 parent 5fce8cb commit f9c100a
Show file tree
Hide file tree
Showing 7 changed files with 445 additions and 0 deletions.
18 changes: 18 additions & 0 deletions cmd/gateway-http/gateway-http.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import (
"github.com/tapglue/snaas/platform/redis"
"github.com/tapglue/snaas/service/app"
"github.com/tapglue/snaas/service/connection"
"github.com/tapglue/snaas/service/counter"
"github.com/tapglue/snaas/service/device"
"github.com/tapglue/snaas/service/event"
"github.com/tapglue/snaas/service/invite"
Expand Down Expand Up @@ -342,6 +343,9 @@ func main() {
// Combine connection service and source.
connections = connection.SourcingServiceMiddleware(conSource)(connections)

var counters counter.Service
counters = counter.PostgresService(pgClient)

var devices device.Service
devices = device.PostgresService(pgClient)
devices = device.InstrumentServiceMiddleware(
Expand Down Expand Up @@ -505,6 +509,20 @@ func main() {
),
)

current.Methods("PUT").Path(`/me/counters/{counterName:[a-z_\-]+}`).Name("counterGetMe").HandlerFunc(
handler.Wrap(
withUser,
handler.CounterSet(core.CounterSet(counters)),
),
)

current.Methods("GET").Path(`/counters/{counterName:[a-z_\-]+}`).Name("counterGetAll").HandlerFunc(
handler.Wrap(
withUser,
handler.CounterGetAll(core.CounterGetAll(counters)),
),
)

current.Methods("GET").Path(`/me/followers`).Name("connectionFollowersMe").HandlerFunc(
handler.Wrap(
withUser,
Expand Down
38 changes: 38 additions & 0 deletions core/counter.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
package core

import (
"github.com/tapglue/snaas/service/app"
"github.com/tapglue/snaas/service/counter"
)

// CounterGetAllFunc returns the sum of all counter for a coutner name.
type CounterGetAllFunc func(currentApp *app.App, name string) (uint64, error)

// CounterGetAll returns the sum of all counter for a coutner name.
func CounterGetAll(counters counter.Service) CounterGetAllFunc {
return func(currentApp *app.App, name string) (uint64, error) {
return counters.CountAll(currentApp.Namespace(), name)
}
}

// CounterSetFunc sets the counter for the current user and the given counter
// name to the new value.
type CounterSetFunc func(
currentApp *app.App,
origin uint64,
name string,
value uint64,
) error

// CounterSet sets the counter for the current user and the given counter name
// to the new value.
func CounterSet(counters counter.Service) CounterSetFunc {
return func(
currentApp *app.App,
origin uint64,
name string,
value uint64,
) error {
return counters.Set(currentApp.Namespace(), name, origin, value)
}
}
69 changes: 69 additions & 0 deletions handler/http/counter.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
package http

import (
"encoding/json"
"net/http"

"golang.org/x/net/context"

"github.com/tapglue/snaas/core"
)

// CounterGetAll returns the sum of all counter for a coutner name.
func CounterGetAll(fn core.CounterGetAllFunc) Handler {
return func(ctx context.Context, w http.ResponseWriter, r *http.Request) {
var (
currentApp = appFromContext(ctx)
)

name, err := extractCounterName(r)
if err != nil {
respondError(w, 0, wrapError(ErrBadRequest, err.Error()))
return
}

v, err := fn(currentApp, name)
if err != nil {
respondError(w, 0, err)
return
}

respondJSON(w, http.StatusOK, &payloadCounter{Value: v})
}
}

// CounterSet sets the counter for the current user and the given counter name
// to the new value.
func CounterSet(fn core.CounterSetFunc) Handler {
return func(ctx context.Context, w http.ResponseWriter, r *http.Request) {
var (
currentApp = appFromContext(ctx)
currentUser = userFromContext(ctx)
p = payloadCounter{}
)

name, err := extractCounterName(r)
if err != nil {
respondError(w, 0, wrapError(ErrBadRequest, err.Error()))
return
}

err = json.NewDecoder(r.Body).Decode(&p)
if err != nil {
respondError(w, 0, wrapError(ErrBadRequest, err.Error()))
return
}

err = fn(currentApp, currentUser.ID, name, p.Value)
if err != nil {
respondError(w, 0, err)
return
}

respondJSON(w, http.StatusNoContent, nil)
}
}

type payloadCounter struct {
Value uint64 `json:"value"`
}
5 changes: 5 additions & 0 deletions handler/http/query.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ const (

keyAppID = "appID"
keyCommentID = "commentID"
keyCounterName = "counterName"
keyCursorAfter = "after"
keyCursorBefore = "before"
keyInviteConnections = "invite-connections"
Expand Down Expand Up @@ -151,6 +152,10 @@ func extractConnectionOpts(r *http.Request) (connection.QueryOptions, error) {
return connection.QueryOptions{}, nil
}

func extractCounterName(r *http.Request) (string, error) {
return mux.Vars(r)[keyCounterName], nil
}

type condition struct {
EQ string `json:"eq"`
IN []string `json:"in"`
Expand Down
17 changes: 17 additions & 0 deletions service/counter/counter.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
package counter

import (
"github.com/tapglue/snaas/platform/service"
)

// Service for counter interactions.
type Service interface {
service.Lifecycle

Count(namespace, name string, userID uint64) (uint64, error)
CountAll(namespace, name string) (uint64, error)
Set(namespace, name string, userID, value uint64) error
}

// ServiceMiddleware is a chainable behaviour modifier for Service.
type ServiceMiddleware func(Service) Service
195 changes: 195 additions & 0 deletions service/counter/postgres.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,195 @@
package counter

import (
"fmt"

"github.com/jmoiron/sqlx"

"github.com/tapglue/snaas/platform/pg"
)

const (
pgGetCounter = `
SELECT
value
FROM
%s.counters
WHERE
deleted = false
AND name = $1
ANd user_id = $2
LIMIT
1`
pgGetCounterAll = `
SELECT
sum(value)
FROM
%s.counters
WHERE
deleted = false
AND name = $1`
pgSetCounter = `
INSERT INTO %s.counters(name, user_id, value)
VALUES($1, $2, $3)
ON CONFLICT (name, user_id) DO
UPDATE SET
value = $3`

pgCreateSchema = `CREATE SCHEMA IF NOT EXISTS %s`
pgCreateTable = `
CREATE TABLE IF NOT EXISTS %s.counters(
name TEXT NOT NULL,
user_id BIGINT NOT NULL,
value BIGINT NOT NULL,
deleted BOOL DEFAULT false,
created_at TIMESTAMP WITHOUT TIME ZONE DEFAULT (now() AT TIME ZONE 'utc'),
updated_at TIMESTAMP WITHOUT TIME ZONE DEFAULT (now() AT TIME ZONE 'utc'),
CONSTRAINT counter_id UNIQUE (name, user_id),
PRIMARY KEY (name, user_id)
)`
pgDropTable = `DROP TABLE IF EXISTS %s.counters CASCADE`

pgIndexCounterID = `
CREATE UNIQUE INDEX
%s
ON
%s.counters
USING
btree(name, user_id)`
pgIndexCounterName = `
CREATE INDEX
%s
ON
%s.counters
USING
btree(name)`

// Extensions.
pgCreateExtensionModdatetime = `CREATE EXTENSION IF NOT EXISTS moddatetime`

// Trigger to autoamtically set the latest time on updated_at, depends on
// the moddatetime extension:
// * https://www.postgresql.org/docs/current/static/contrib-spi.html
// * https://github.com/postgres/postgres/blob/master/contrib/spi/moddatetime.example
// * https://dba.stackexchange.com/a/158750
pgAlterTriggerUpdatedAt = `
ALTER TRIGGER %s ON %s.counters DEPENDS ON EXTENSION moddatetime`
pgCreateTriggerUpdatedAt = `
CREATE TRIGGER
%s
BEFORE UPDATE ON
%s.counters
FOR EACH ROW EXECUTE PROCEDURE
moddatetime(updated_at)`
pgDropTriggerUpdatedAt = `
DROP TRIGGER IF EXISTS %s ON %s.counters`
)

type pgService struct {
db *sqlx.DB
}

func PostgresService(db *sqlx.DB) Service {
return &pgService{db: db}
}

func (s *pgService) Count(ns, name string, userID uint64) (uint64, error) {
var (
args = []interface{}{name, userID}
query = fmt.Sprintf(pgGetCounter, ns)

value uint64
)

err := s.db.Get(&value, query, args...)
if err != nil && pg.IsRelationNotFound(pg.WrapError(err)) {
if err := s.Setup(ns); err != nil {
return 0, err
}

err = s.db.Get(&value, query, args...)
}

return value, err
}

func (s *pgService) CountAll(ns, name string) (uint64, error) {
var (
args = []interface{}{name}
query = fmt.Sprintf(pgGetCounterAll, ns)

value uint64
)

err := s.db.Get(&value, query, args...)
if err != nil && pg.IsRelationNotFound(pg.WrapError(err)) {
if err := s.Setup(ns); err != nil {
return 0, err
}

err = s.db.Get(&value, query, args...)
}

return value, err
}

func (s *pgService) Set(ns, name string, userID, value uint64) error {
var (
args = []interface{}{
name,
userID,
value,
}
query = fmt.Sprintf(pgSetCounter, ns)
)

_, err := s.db.Exec(query, args...)
if err != nil && pg.IsRelationNotFound(pg.WrapError(err)) {
if err := s.Setup(ns); err != nil {
return err
}

_, err = s.db.Exec(query, args...)
}

return err
}

func (s *pgService) Setup(ns string) error {
for _, q := range []string{
fmt.Sprintf(pgCreateSchema, ns),
fmt.Sprintf(pgCreateTable, ns),

// Indexes.
pg.GuardIndex(ns, "counter_counter_id", pgIndexCounterID),
pg.GuardIndex(ns, "counter_counter_name", pgIndexCounterName),

// FIXME: Re-enable when migrated to Postgres 9.6
// Setup idempotent updated_at trigger.
// pgCreateExtensionModdatetime,
// fmt.Sprintf(pgDropTriggerUpdatedAt, "counter_updated_at", ns),
// fmt.Sprintf(pgCreateTriggerUpdatedAt, "counter_updated_at", ns),
// fmt.Sprintf(pgAlterTriggerUpdatedAt, "counter_updated_at", ns),
} {
_, err := s.db.Exec(q)
if err != nil {
return fmt.Errorf("setup '%s': %s", q, err)
}
}

return nil
}

func (s *pgService) Teardown(ns string) error {
for _, q := range []string{
fmt.Sprintf(pgDropTable, ns),
} {
_, err := s.db.Exec(q)
if err != nil {
return fmt.Errorf("teardown '%s': %s", q, err)
}
}

return nil
}
Loading

0 comments on commit f9c100a

Please sign in to comment.