From eafdd752212917c7cc23f3354c603b278e86d3aa Mon Sep 17 00:00:00 2001 From: zibthedog Date: Sun, 28 Jul 2024 12:41:48 -0400 Subject: [PATCH 1/3] add FF tables --- db/db.go | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/db/db.go b/db/db.go index 885c3b7..09c9bc8 100644 --- a/db/db.go +++ b/db/db.go @@ -103,6 +103,11 @@ func setupDb(tx *sql.Tx) error { `ALTER TABLE accounts ADD COLUMN IF NOT EXISTS discordId VARCHAR(32) UNIQUE DEFAULT NULL`, `ALTER TABLE accounts ADD COLUMN IF NOT EXISTS googleId VARCHAR(32) UNIQUE DEFAULT NULL`, + + // ---------------------------------- + // MIGRATION 004 + `CREATE TABLE IF NOT EXISTS featureFlags (id INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, name VARCHAR(64) UNIQUE NOT NULL, enabled TINYINT(1) NOT NULL, percentage TINYINT(3) NOT NULL DEFAULT 0)` + `CREATE TABLE IF NOT EXISTS accountFeatureFlagOverrides (accountId BINARY(16) NOT_NULL, flagId INT(11) NOT NULL, PRIMARY KEY (accountId, flagId), FOREIGN KEY (accountId) REFERENCES accounts (uuid) ON DELETE CASCADE ON UPDATE CASCADE, FOREIGN KEY (flagId) REFERENCES featureFlags (id) ON DELETE CASCADE ON UPDATE CASCADE)` } for _, q := range queries { From 5c8b07138eef0dae58d09aee76821a4d9a379a41 Mon Sep 17 00:00:00 2001 From: zibthedog Date: Sun, 28 Jul 2024 17:17:59 -0400 Subject: [PATCH 2/3] partitioning, db changes + fetchers --- api/featureflags.go | 44 ++++++++++++++++++++++++++++ db/db.go | 4 +-- db/featureflags.go | 70 ++++++++++++++++++++++++++++++++++++++++++++ defs/featureflags.go | 28 ++++++++++++++++++ 4 files changed, 144 insertions(+), 2 deletions(-) create mode 100644 api/featureflags.go create mode 100644 db/featureflags.go create mode 100644 defs/featureflags.go diff --git a/api/featureflags.go b/api/featureflags.go new file mode 100644 index 0000000..3cd6a02 --- /dev/null +++ b/api/featureflags.go @@ -0,0 +1,44 @@ +/* + Copyright (C) 2024 Pagefault Games + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU Affero General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU Affero General Public License for more details. + + You should have received a copy of the GNU Affero General Public License + along with this program. If not, see . +*/ + +package api + +import ( + "hash/crc32" + "math" +) + +func CheckFeatureFlagEnabled(accountId []byte, flag string, percent int) bool { + if percent <= 0 { + return false + } else if percent >= 100 { + return true + } + + pct := math.Max(0, math.Min(100, float64(percent))) + flagBytes := []byte(flag) + + // hash key is [...accountId, ...flagBytes] + hashKey := make([]byte, len(accountId)+len(flagBytes)) + copy(hashKey, accountId) + copy(hashKey[len(accountId):], flagBytes) + + boundary := float64(math.MaxUint32) * (pct / float64(100)) + hash := float64(crc32.ChecksumIEEE(hashKey)) + + return hash >= boundary +} diff --git a/db/db.go b/db/db.go index 09c9bc8..664e64c 100644 --- a/db/db.go +++ b/db/db.go @@ -106,8 +106,8 @@ func setupDb(tx *sql.Tx) error { // ---------------------------------- // MIGRATION 004 - `CREATE TABLE IF NOT EXISTS featureFlags (id INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, name VARCHAR(64) UNIQUE NOT NULL, enabled TINYINT(1) NOT NULL, percentage TINYINT(3) NOT NULL DEFAULT 0)` - `CREATE TABLE IF NOT EXISTS accountFeatureFlagOverrides (accountId BINARY(16) NOT_NULL, flagId INT(11) NOT NULL, PRIMARY KEY (accountId, flagId), FOREIGN KEY (accountId) REFERENCES accounts (uuid) ON DELETE CASCADE ON UPDATE CASCADE, FOREIGN KEY (flagId) REFERENCES featureFlags (id) ON DELETE CASCADE ON UPDATE CASCADE)` + `CREATE TABLE IF NOT EXISTS featureFlags (id INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, name VARCHAR(64) UNIQUE NOT NULL, enabled TINYINT(1) NOT NULL, percentage TINYINT(3) NOT NULL DEFAULT 0, CONSTRAINT chk_featureFlagsPercentage_withinRange CHECK(percentage >= 0 AND percentage <= 100))`, + `CREATE TABLE IF NOT EXISTS accountFeatureFlagOverrides (accountId BINARY(16) NOT NULL, flagId INT(11) NOT NULL, enabled TINYINT(1) NOT NULL, PRIMARY KEY (accountId, flagId), FOREIGN KEY (accountId) REFERENCES accounts (uuid) ON DELETE CASCADE ON UPDATE CASCADE, FOREIGN KEY (flagId) REFERENCES featureFlags (id) ON DELETE CASCADE ON UPDATE CASCADE)`, } for _, q := range queries { diff --git a/db/featureflags.go b/db/featureflags.go new file mode 100644 index 0000000..c89fab4 --- /dev/null +++ b/db/featureflags.go @@ -0,0 +1,70 @@ +/* + Copyright (C) 2024 Pagefault Games + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU Affero General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU Affero General Public License for more details. + + You should have received a copy of the GNU Affero General Public License + along with this program. If not, see . +*/ + +package db + +import "github.com/pagefaultgames/rogueserver/defs" + +func GetEnabledFeatureFlags() ([]defs.FeatureFlag, error) { + var activeFlags []defs.FeatureFlag + + results, err := handle.Query("SELECT name, percentage FROM featureFlags WHERE enabled = 1") + + if err != nil { + return activeFlags, err + } + + defer results.Close() + + for results.Next() { + var flag defs.FeatureFlag + + err = results.Scan(&flag.Name, &flag.Percentage) + if err != nil { + return activeFlags, err + } + + activeFlags = append(activeFlags, flag) + } + + return activeFlags, nil +} + +func GetFeatureFlagOverrides(accountId []byte) ([]defs.FeatureFlagOverride, error) { + var overrides []defs.FeatureFlagOverride + + results, err := handle.Query("SELECT ff.name, o.enabled FROM accoutFeatureFlagOverrides o JOIN featureFlags ff ON o.flagId = ff.id WHERE accountId = ?", accountId) + + if err != nil { + return overrides, err + } + + defer results.Close() + + for results.Next() { + var override defs.FeatureFlagOverride + + err = results.Scan(&override.FlagName, &override.Enabled) + if err != nil { + return overrides, err + } + + overrides = append(overrides, override) + } + + return overrides, nil +} diff --git a/defs/featureflags.go b/defs/featureflags.go new file mode 100644 index 0000000..19a2fce --- /dev/null +++ b/defs/featureflags.go @@ -0,0 +1,28 @@ +/* + Copyright (C) 2024 Pagefault Games + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU Affero General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU Affero General Public License for more details. + + You should have received a copy of the GNU Affero General Public License + along with this program. If not, see . +*/ + +package defs + +type FeatureFlag struct { + Name string + Percentage int +} + +type FeatureFlagOverride struct { + FlagName string + Enabled bool +} From 4c3a42418818faf537be4897d919c564857b75f0 Mon Sep 17 00:00:00 2001 From: zibthedog Date: Tue, 30 Jul 2024 02:11:20 -0400 Subject: [PATCH 3/3] refactor: Simplify, hardcode (empty) access groups --- api/account/common.go | 5 +++++ api/account/discord.go | 16 +++++++++++++++ api/account/info.go | 42 ++++++++++++++++++++++++++++++++++++---- api/featureflags.go | 44 ------------------------------------------ db/db.go | 3 +-- db/featureflags.go | 34 ++++++-------------------------- defs/featureflags.go | 9 ++------- 7 files changed, 68 insertions(+), 85 deletions(-) delete mode 100644 api/featureflags.go diff --git a/api/account/common.go b/api/account/common.go index 9c9102f..9fc79d4 100644 --- a/api/account/common.go +++ b/api/account/common.go @@ -37,6 +37,11 @@ const ( UUIDSize = 16 TokenSize = 32 + + // feature access groups + DEV_STAFF = "DEV_STAFF" + CONTRIBUTOR = "CONTRIBUTOR" + EVERYONE = "EVERYONE" ) var ( diff --git a/api/account/discord.go b/api/account/discord.go index 792afe6..271868f 100644 --- a/api/account/discord.go +++ b/api/account/discord.go @@ -106,3 +106,19 @@ func RetrieveDiscordId(code string) (string, error) { return user.Id, nil } + +// TODO: fetch these instead of hardcoding them +var devsAndStaff = map[string]bool{} +var contributors = map[string]bool{} + +func GetAccessGroupByDiscordRole(discordId string) (group string) { + if devsAndStaff[discordId] { + return DEV_STAFF + } + + if contributors[discordId] { + return CONTRIBUTOR + } + + return "" +} diff --git a/api/account/info.go b/api/account/info.go index 6802238..bb905dc 100644 --- a/api/account/info.go +++ b/api/account/info.go @@ -22,20 +22,54 @@ import ( ) type InfoResponse struct { - Username string `json:"username"` - DiscordId string `json:"discordId"` - GoogleId string `json:"googleId"` - LastSessionSlot int `json:"lastSessionSlot"` + Username string `json:"username"` + DiscordId string `json:"discordId"` + GoogleId string `json:"googleId"` + LastSessionSlot int `json:"lastSessionSlot"` + FeatureFlags []string `json:"featureFlags"` } // /account/info - get account info func Info(username string, discordId string, googleId string, uuid []byte) (InfoResponse, error) { slot, _ := db.GetLatestSessionSaveDataSlot(uuid) + featureFlags := getFeatureFlags(discordId) response := InfoResponse{ Username: username, LastSessionSlot: slot, DiscordId: discordId, GoogleId: googleId, + FeatureFlags: featureFlags, } return response, nil } + +func getFeatureFlags(discordId string) []string { + var flags []string + + enabledFlags, err := db.GetEnabledFeatureFlags() + if err != nil { + return flags + } + + for _, flag := range enabledFlags { + var hasAccess = false + + if flag.AccessLevel == EVERYONE { + hasAccess = true + } else { + accessGroup := GetAccessGroupByDiscordRole(discordId) + + if flag.AccessLevel == DEV_STAFF { + hasAccess = accessGroup == DEV_STAFF + } else if flag.AccessLevel == CONTRIBUTOR { + hasAccess = accessGroup == CONTRIBUTOR || accessGroup == DEV_STAFF + } + } + + if hasAccess { + flags = append(flags, flag.Name) + } + } + + return flags +} diff --git a/api/featureflags.go b/api/featureflags.go deleted file mode 100644 index 3cd6a02..0000000 --- a/api/featureflags.go +++ /dev/null @@ -1,44 +0,0 @@ -/* - Copyright (C) 2024 Pagefault Games - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU Affero General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU Affero General Public License for more details. - - You should have received a copy of the GNU Affero General Public License - along with this program. If not, see . -*/ - -package api - -import ( - "hash/crc32" - "math" -) - -func CheckFeatureFlagEnabled(accountId []byte, flag string, percent int) bool { - if percent <= 0 { - return false - } else if percent >= 100 { - return true - } - - pct := math.Max(0, math.Min(100, float64(percent))) - flagBytes := []byte(flag) - - // hash key is [...accountId, ...flagBytes] - hashKey := make([]byte, len(accountId)+len(flagBytes)) - copy(hashKey, accountId) - copy(hashKey[len(accountId):], flagBytes) - - boundary := float64(math.MaxUint32) * (pct / float64(100)) - hash := float64(crc32.ChecksumIEEE(hashKey)) - - return hash >= boundary -} diff --git a/db/db.go b/db/db.go index 664e64c..1eb0b21 100644 --- a/db/db.go +++ b/db/db.go @@ -106,8 +106,7 @@ func setupDb(tx *sql.Tx) error { // ---------------------------------- // MIGRATION 004 - `CREATE TABLE IF NOT EXISTS featureFlags (id INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, name VARCHAR(64) UNIQUE NOT NULL, enabled TINYINT(1) NOT NULL, percentage TINYINT(3) NOT NULL DEFAULT 0, CONSTRAINT chk_featureFlagsPercentage_withinRange CHECK(percentage >= 0 AND percentage <= 100))`, - `CREATE TABLE IF NOT EXISTS accountFeatureFlagOverrides (accountId BINARY(16) NOT NULL, flagId INT(11) NOT NULL, enabled TINYINT(1) NOT NULL, PRIMARY KEY (accountId, flagId), FOREIGN KEY (accountId) REFERENCES accounts (uuid) ON DELETE CASCADE ON UPDATE CASCADE, FOREIGN KEY (flagId) REFERENCES featureFlags (id) ON DELETE CASCADE ON UPDATE CASCADE)`, + `CREATE TABLE IF NOT EXISTS featureFlags (id INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, name VARCHAR(64) UNIQUE NOT NULL, accessLevel VARCHAR(16), CONSTRAINT chk_featureFlags_accessLevel CHECK(accessLevel IN ('DEV_STAFF', 'CONTRIBUTOR', 'EVERYONE')))`, } for _, q := range queries { diff --git a/db/featureflags.go b/db/featureflags.go index c89fab4..cb38605 100644 --- a/db/featureflags.go +++ b/db/featureflags.go @@ -17,12 +17,14 @@ package db -import "github.com/pagefaultgames/rogueserver/defs" +import ( + "github.com/pagefaultgames/rogueserver/defs" +) func GetEnabledFeatureFlags() ([]defs.FeatureFlag, error) { var activeFlags []defs.FeatureFlag - results, err := handle.Query("SELECT name, percentage FROM featureFlags WHERE enabled = 1") + results, err := handle.Query("SELECT name, accessLevel FROM featureFlags WHERE enabled = 1") if err != nil { return activeFlags, err @@ -33,7 +35,8 @@ func GetEnabledFeatureFlags() ([]defs.FeatureFlag, error) { for results.Next() { var flag defs.FeatureFlag - err = results.Scan(&flag.Name, &flag.Percentage) + err = results.Scan(&flag.Name, &flag.AccessLevel) + if err != nil { return activeFlags, err } @@ -43,28 +46,3 @@ func GetEnabledFeatureFlags() ([]defs.FeatureFlag, error) { return activeFlags, nil } - -func GetFeatureFlagOverrides(accountId []byte) ([]defs.FeatureFlagOverride, error) { - var overrides []defs.FeatureFlagOverride - - results, err := handle.Query("SELECT ff.name, o.enabled FROM accoutFeatureFlagOverrides o JOIN featureFlags ff ON o.flagId = ff.id WHERE accountId = ?", accountId) - - if err != nil { - return overrides, err - } - - defer results.Close() - - for results.Next() { - var override defs.FeatureFlagOverride - - err = results.Scan(&override.FlagName, &override.Enabled) - if err != nil { - return overrides, err - } - - overrides = append(overrides, override) - } - - return overrides, nil -} diff --git a/defs/featureflags.go b/defs/featureflags.go index 19a2fce..5b595da 100644 --- a/defs/featureflags.go +++ b/defs/featureflags.go @@ -18,11 +18,6 @@ package defs type FeatureFlag struct { - Name string - Percentage int -} - -type FeatureFlagOverride struct { - FlagName string - Enabled bool + Name string + AccessLevel string // DEV_STAFF | CONTRIBUTOR | EVERYONE }