diff --git a/cluster/idgen.go b/cluster/idgen.go deleted file mode 100644 index 89050c44aeabeb0125edc3987201a17064b20175..0000000000000000000000000000000000000000 --- a/cluster/idgen.go +++ /dev/null @@ -1,41 +0,0 @@ -package cluster - -import ( - "errors" - "fmt" - "io" - "net/http" - "time" - - "github.com/turt2live/matrix-media-repo/common/config" - "github.com/turt2live/matrix-media-repo/util" - "github.com/turt2live/matrix-media-repo/util/stream_util" -) - -func GetId() (string, error) { - req, err := http.NewRequest("GET", util.MakeUrl(config.Get().Cluster.IDGenerator.Location, "/v1/id"), nil) - if err != nil { - return "", err - } - - req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", config.Get().Cluster.IDGenerator.Secret)) - - client := &http.Client{ - Timeout: 1 * time.Second, // the server should be fast (much faster than this) - } - res, err := client.Do(req) - if err != nil { - return "", err - } - defer stream_util.DumpAndCloseStream(res.Body) - - contents, err := io.ReadAll(res.Body) - if err != nil { - return "", err - } - if res.StatusCode != http.StatusOK { - return "", errors.New(fmt.Sprintf("unexpected status code from ID generator: %d", res.StatusCode)) - } - - return string(contents), nil -} diff --git a/cmd/export_synapse_for_import/main.go b/cmd/export_synapse_for_import/main.go index c71b071bd087b4e412f023e8cee0ad436f7111b6..7c6c7c2baf02eaabb74b711e528a714ae29432cd 100644 --- a/cmd/export_synapse_for_import/main.go +++ b/cmd/export_synapse_for_import/main.go @@ -16,6 +16,7 @@ import ( "github.com/turt2live/matrix-media-repo/common/config" "github.com/turt2live/matrix-media-repo/common/logging" "github.com/turt2live/matrix-media-repo/common/rcontext" + "github.com/turt2live/matrix-media-repo/common/runtime" "github.com/turt2live/matrix-media-repo/synapse" "github.com/turt2live/matrix-media-repo/util" "github.com/turt2live/matrix-media-repo/util/stream_util" @@ -69,6 +70,7 @@ func main() { } logrus.Info("Setting up for importing...") + runtime.CheckIdGenerator() connectionString := "postgres://" + *postgresUsername + ":" + realPsqlPassword + "@" + *postgresHost + ":" + strconv.Itoa(*postgresPort) + "/" + *postgresDatabase + "?sslmode=disable" diff --git a/cmd/gdpr_export/main.go b/cmd/gdpr_export/main.go index 0b5887e06dec24d228750797c8d30c79028880d5..722c52e977f70d8756098537371fe02d214d27c7 100644 --- a/cmd/gdpr_export/main.go +++ b/cmd/gdpr_export/main.go @@ -41,6 +41,7 @@ func main() { } config.Path = *configPath + runtime.CheckIdGenerator() assets.SetupMigrations(*migrationsPath) assets.SetupTemplates(*templatesPath) diff --git a/cmd/gdpr_import/main.go b/cmd/gdpr_import/main.go index 95cf08bda8a129a47831a15458a85693030bcb7e..9b14db3fbcb744918a66e580301773e14ec3ca2a 100644 --- a/cmd/gdpr_import/main.go +++ b/cmd/gdpr_import/main.go @@ -31,6 +31,7 @@ func main() { } config.Path = *configPath + runtime.CheckIdGenerator() assets.SetupMigrations(*migrationsPath) var err error diff --git a/cmd/import_synapse/main.go b/cmd/import_synapse/main.go index 4c8b246a6f40dcc8ac96b5b44dce3f53f115e6c4..5de58b74b285fedb13217e6b861d30c399af12ef 100644 --- a/cmd/import_synapse/main.go +++ b/cmd/import_synapse/main.go @@ -52,6 +52,7 @@ func main() { } config.Path = *configPath + runtime.CheckIdGenerator() assets.SetupMigrations(*migrationsPath) var realPsqlPassword string diff --git a/cmd/s3_consistency_check/main.go b/cmd/s3_consistency_check/main.go index e9162a975a2a8e60bf6017cdb8622bc3f43e8742..61e145a3d180de3722e151abd23cffdc355088f4 100644 --- a/cmd/s3_consistency_check/main.go +++ b/cmd/s3_consistency_check/main.go @@ -3,6 +3,8 @@ package main import ( "flag" "fmt" + "os" + "github.com/sirupsen/logrus" "github.com/turt2live/matrix-media-repo/common/assets" "github.com/turt2live/matrix-media-repo/common/config" @@ -11,7 +13,6 @@ import ( "github.com/turt2live/matrix-media-repo/common/runtime" "github.com/turt2live/matrix-media-repo/storage" "github.com/turt2live/matrix-media-repo/storage/datastore" - "os" ) func main() { @@ -29,6 +30,7 @@ func main() { } config.Path = *configPath + runtime.CheckIdGenerator() assets.SetupMigrations(*migrationsPath) assets.SetupTemplates(*templatesPath) diff --git a/cmd/service_idgen/main.go b/cmd/service_idgen/main.go deleted file mode 100644 index c7b2b1dfe71492beb45df1feebe789e722b52412..0000000000000000000000000000000000000000 --- a/cmd/service_idgen/main.go +++ /dev/null @@ -1,73 +0,0 @@ -package main - -import ( - "flag" - "fmt" - "github.com/bwmarrin/snowflake" - "github.com/sirupsen/logrus" - "github.com/turt2live/matrix-media-repo/util" - "net/http" - "os" - "strconv" -) - -func main() { - machineId := flag.Int("machine", getIdFromEnv(), "The machine ID. 0-1023 (inclusive)") - secret := flag.String("secret", getValFromEnv("API_SECRET", ""), "The API secret to require on requests") - bind := flag.String("bind", getValFromEnv("API_BIND", ":8090"), "Where to bind the API to") - flag.Parse() - - node, err := snowflake.NewNode(int64(*machineId)) - if err != nil { - panic(err) - } - - fmt.Printf("Running as machine %d\n", *machineId) - - expectedSecret := fmt.Sprintf("Bearer %s", *secret) - - http.HandleFunc("/v1/id", func(w http.ResponseWriter, req *http.Request) { - if req.Header.Get("Authorization") != expectedSecret { - w.WriteHeader(http.StatusForbidden) - return - } - - // Generate a random string to pad out the returned ID - r, err := util.GenerateRandomString(32) - if err != nil { - logrus.Error(err) - w.WriteHeader(500) - return - } - s := r + node.Generate().String() - - w.Header().Set("Content-Type", "text/plain") - w.WriteHeader(http.StatusOK) - _, err = w.Write([]byte(s)) - if err != nil { - fmt.Println(err) - return - } - }) - - err = http.ListenAndServe(*bind, nil) - if err != nil { - panic(err) - } -} - -func getIdFromEnv() int { - if val, ok := os.LookupEnv("MACHINE_ID"); ok { - if i, err := strconv.Atoi(val); err == nil { - return i - } - } - return 0 -} - -func getValFromEnv(key string, def string) string { - if val, ok := os.LookupEnv(key); ok { - return val - } - return def -} diff --git a/common/config/conf_main.go b/common/config/conf_main.go index 0353e781c08b21e7a2e2c592094973be27b22941..ecc508d093b6ba79d36f8e1a95011ae18e541d90 100644 --- a/common/config/conf_main.go +++ b/common/config/conf_main.go @@ -16,7 +16,6 @@ type MainRepoConfig struct { Plugins []PluginConfig `yaml:"plugins,flow"` Sentry SentryConfig `yaml:"sentry"` Redis RedisConfig `yaml:"redis"` - Cluster ClusterConfig `yaml:"cluster"` } func NewDefaultMainConfig() MainRepoConfig { @@ -135,11 +134,5 @@ func NewDefaultMainConfig() MainRepoConfig { Enabled: false, Shards: []RedisShardConfig{}, }, - Cluster: ClusterConfig{ - IDGenerator: IDGeneratorConfig{ - Location: "", - Secret: "", - }, - }, } } diff --git a/common/config/models_main.go b/common/config/models_main.go index d8f5948aaa41de78d8c99f755b3b8243aedb268d..fcd043a6fc43256156a44f040e7a18a6453142ca 100644 --- a/common/config/models_main.go +++ b/common/config/models_main.go @@ -90,12 +90,3 @@ type RedisShardConfig struct { Name string `yaml:"name"` Address string `yaml:"addr"` } - -type ClusterConfig struct { - IDGenerator IDGeneratorConfig `yaml:"idGenerator"` -} - -type IDGeneratorConfig struct { - Location string `yaml:"location"` - Secret string `yaml:"secret"` -} diff --git a/common/runtime/init.go b/common/runtime/init.go index 4d27803068c645dc0f21a71cdda864fce64cac58..4aaf5065063217f6995015e87f040ab5f2994eae 100644 --- a/common/runtime/init.go +++ b/common/runtime/init.go @@ -4,6 +4,7 @@ import ( "fmt" "github.com/getsentry/sentry-go" + "github.com/turt2live/matrix-media-repo/util/ids" "github.com/sirupsen/logrus" "github.com/turt2live/matrix-media-repo/common/config" @@ -17,6 +18,7 @@ import ( func RunStartupSequence() { version.Print(true) + CheckIdGenerator() config.PrintDomainInfo() config.CheckDeprecations() LoadDatabase() @@ -80,3 +82,14 @@ func LoadDatastores() { } } } + +func CheckIdGenerator() { + // Create a throwaway ID to ensure no errors + _, err := ids.NewUniqueId() + if err != nil { + panic(err) + } + + id := ids.GetMachineId() + logrus.Infof("Running as machine %d for ID generation. This ID must be unique within your cluster.", id) +} diff --git a/config.sample.yaml b/config.sample.yaml index e3570bb466fda47ce249774f9748f04a6ce54fbc..a3900d94d6e36ef8d7442b1374eeb46d0bab8bf0 100644 --- a/config.sample.yaml +++ b/config.sample.yaml @@ -578,17 +578,3 @@ sentry: # Whether or not to turn on sentry's built in debugging. This will increase log output. debug: false - -# Options for controlling clustering behaviour of the media repo. This requires an ID generator -# service in your infrastructure. -# -# For more information see https://docs.t2bot.io/matrix-media-repo/installing/environments/clustered.html -cluster: - # Options for accessing the ID generator service. - idGenerator: - # The secret being used by the ID generator service. Clustering is disabled unless this is set - # to a non-empty string. - secret: "" - - # The URL for where the ID generator service can be reached. - location: "http://localhost:8090" diff --git a/util/ids/snowflake.go b/util/ids/snowflake.go new file mode 100644 index 0000000000000000000000000000000000000000..73f531acfb8a33a671e4b7f8dfde588ceef0c5e0 --- /dev/null +++ b/util/ids/snowflake.go @@ -0,0 +1,32 @@ +package ids + +import ( + "os" + "strconv" + + "github.com/bwmarrin/snowflake" +) + +func GetMachineId() int64 { + if val, ok := os.LookupEnv("MACHINE_ID"); ok { + if i, err := strconv.ParseInt(val, 10, 64); err == nil { + return i + } + } + return 0 +} + +var sfnode *snowflake.Node + +func makeSnowflake() (*snowflake.Node, error) { + if sfnode != nil { + return sfnode, nil + } + machineId := GetMachineId() + node, err := snowflake.NewNode(machineId) + if err != nil { + return nil, err + } + sfnode = node + return sfnode, nil +} diff --git a/util/ids/unique.go b/util/ids/unique.go index 2baa7fec730375d0fb2c5de7966c716b09479f22..a7ff7e43a16d7462a4a86d5c8c32269c5fc6be7c 100644 --- a/util/ids/unique.go +++ b/util/ids/unique.go @@ -1,20 +1,17 @@ package ids import ( - "github.com/turt2live/matrix-media-repo/cluster" - "github.com/turt2live/matrix-media-repo/common/config" "github.com/turt2live/matrix-media-repo/util" - "strconv" ) func NewUniqueId() (string, error) { - if config.Get().Cluster.IDGenerator.Secret != "" { - return cluster.GetId() + r, err := util.GenerateRandomString(32) // pad out the snowflake + if err != nil { + return "", err } - - b, err := util.GenerateRandomBytes(64) + sf, err := makeSnowflake() if err != nil { return "", err } - return util.GetSha1OfString(string(b) + strconv.FormatInt(util.NowMillis(), 10)) + return r + sf.Generate().String(), nil }