From 91841ab3d3c3f19e228320eff2eda9d851309ccd Mon Sep 17 00:00:00 2001 From: phga Date: Wed, 13 Oct 2021 11:06:56 +0200 Subject: [PATCH] Init --- .gitignore | 4 ++ build/app/Dockerfile | 7 ++ build/docker-compose.yml | 32 +++++++++ cmd/bs1in/bs1in.go | 67 ++++++++++++++++++ cmd/bs1in/config.go | 33 +++++++++ cmd/bs1in/crypto.go | 116 ++++++++++++++++++++++++++++++++ cmd/bs1in/database.go | 34 ++++++++++ cmd/bs1in/helpers.go | 13 ++++ cmd/bs1in/mail.go | 21 ++++++ cmd/bs1in/mail_test.go | 12 ++++ go.mod | 19 ++++++ go.sum | 116 ++++++++++++++++++++++++++++++++ web/static/css/bs1in.css | 76 +++++++++++++++++++++ web/static/js/helper.js | 21 ++++++ web/templates/header.html | 4 ++ web/templates/layouts/base.html | 21 ++++++ web/templates/main.html | 8 +++ web/templates/soon.html | 3 + 18 files changed, 607 insertions(+) create mode 100644 .gitignore create mode 100644 build/app/Dockerfile create mode 100644 build/docker-compose.yml create mode 100644 cmd/bs1in/bs1in.go create mode 100644 cmd/bs1in/config.go create mode 100644 cmd/bs1in/crypto.go create mode 100644 cmd/bs1in/database.go create mode 100644 cmd/bs1in/helpers.go create mode 100644 cmd/bs1in/mail.go create mode 100644 cmd/bs1in/mail_test.go create mode 100644 go.mod create mode 100644 go.sum create mode 100644 web/static/css/bs1in.css create mode 100644 web/static/js/helper.js create mode 100644 web/templates/header.html create mode 100644 web/templates/layouts/base.html create mode 100644 web/templates/main.html create mode 100644 web/templates/soon.html diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..53ef257 --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +web/static/fonts/* +web/data/ +configs/* +cmd/bs1in/bs1in \ No newline at end of file diff --git a/build/app/Dockerfile b/build/app/Dockerfile new file mode 100644 index 0000000..66f42a1 --- /dev/null +++ b/build/app/Dockerfile @@ -0,0 +1,7 @@ +FROM golang:1.16.2 + +ENV GO111MODULE=off +RUN go get -u -d g.phga.de/phga/bs1in/... +WORKDIR /go/src/g.phga.de/phga/bs1in/cmd/bs1in +RUN go build +CMD ["./bs1in"] \ No newline at end of file diff --git a/build/docker-compose.yml b/build/docker-compose.yml new file mode 100644 index 0000000..edd046a --- /dev/null +++ b/build/docker-compose.yml @@ -0,0 +1,32 @@ +version: '3' + +services: + bs1in: + build: + context: ./app + dockerfile: Dockerfile + restart: 'no' + ports: + - 8099:6969 + volumes: + - /srv/bs1in/configs:CONFIGFOLDER + - /srv/bs1in/web/data:DATAFOLDER + - /srv/bs1in/web/static/fonts:FONTSFOLDER + + mongo: + image: mongo + restart: 'no' + env_file: + - /srv/gott/.env + ports: + - 27017:27017 + volumes: + - /srv/bs1in/db:/data/db + + mongo-express: + image: mongo-express + restart: 'no' + ports: + - 9099:8081 + env_file: + - /srv/bs1in/.env \ No newline at end of file diff --git a/cmd/bs1in/bs1in.go b/cmd/bs1in/bs1in.go new file mode 100644 index 0000000..efa0758 --- /dev/null +++ b/cmd/bs1in/bs1in.go @@ -0,0 +1,67 @@ +// author: phga +// backend for the portfolio +// docker-compose up -d +package main + +import ( + "html/template" + "path" + + "log" + "net/http" +) + +func main() { + // cheap routing + http.HandleFunc("/", handleIndex) + http.HandleFunc("/db_user", handleDbUser) + + // provide the inc directory to the useragent + http.HandleFunc("/static/", handleStatic) + // listen on port 8080 (I use nginx to proxy this local server) + log.Fatalln(http.ListenAndServe(":"+config.WebPort, nil)) +} + +// ---------------------------- HELPER TYPES ----------------------------------- + +type TmplData struct { + HeaderData interface{} + BodyData interface{} +} + +// ------------------------- REQUEST HANDLING ---------------------------------- + +func handleStatic(w http.ResponseWriter, r *http.Request) { + http.StripPrefix("/static/", http.FileServer(http.Dir("../../web/static"))).ServeHTTP(w, r) +} + +func handleIndex(w http.ResponseWriter, r *http.Request) { + // Parses all required html files to provide the actual html that is shipped. + t, _ := getTemplate("layouts/base.html", "main.html", "header.html") + t.Execute(w, t) +} + +func handleDbUser(w http.ResponseWriter, r *http.Request) { + // Parses all required html files to provide the actual html that is shipped. + t, _ := getTemplate("layouts/base.html", "db_user.html", "header.html") + t.Execute(w, t) +} + +// ------------------------- TEMPLATE HELPERS ---------------------------------- + +func getTemplate(files ...string) (*template.Template, error) { + tmplFolder := "../../web/templates/" + for i, f := range files { + files[i] = tmplFolder + f + } + // Name has to be the basename of at least one of the template files + t := template.New(path.Base(files[0])) + // Add custom helper funcs to template + t.Funcs(template.FuncMap{ + "UnwrapOID": unwrapObjectID, + }) + + t, err := t.ParseFiles(files...) + + return t, err +} diff --git a/cmd/bs1in/config.go b/cmd/bs1in/config.go new file mode 100644 index 0000000..c3f43eb --- /dev/null +++ b/cmd/bs1in/config.go @@ -0,0 +1,33 @@ +package main + +import ( + "encoding/json" + "log" + "os" +) + +type Config struct { + WebPort string `json:"web_port"` + DbUser string `json:"db_user"` + DbPass string `json:"db_pass"` + DbHost string `json:"db_host"` + DbPort string `json:"db_port"` + MailUser string `json:"mail_user"` + MailPass string `json:"mail_pass"` + MailSmtpServer string `json:"mail_smtp_server"` + MailSmtpPort string `json:"mail_smtp_port"` +} + +var config Config + +func init() { + f, err := os.Open("../../configs/config.json") + if err != nil { + log.Fatalln(err, "Could not read config file") + } + defer f.Close() + err = json.NewDecoder(f).Decode(&config) + if err != nil { + log.Fatalln(err, "Could not decode config file") + } +} diff --git a/cmd/bs1in/crypto.go b/cmd/bs1in/crypto.go new file mode 100644 index 0000000..6cad133 --- /dev/null +++ b/cmd/bs1in/crypto.go @@ -0,0 +1,116 @@ +package main + +import ( + "crypto/aes" + "crypto/cipher" + cryptorand "crypto/rand" + "encoding/hex" + "errors" + "io" + "math/rand" + "time" +) + +// ----------------------------- CYPHER STUFF ---------------------------------- +var encKey *[32]byte + +func init() { + // Seed Random Fung + rand.Seed(time.Now().UTC().UnixNano()) + // Generate Encryption Key + // MAYB: Change key every X hours, keep last key till all sessions are timed out + encKey = newEncryptionKey() + // Mongo DB setup +} + +// Code copied from +// https://github.com/gtank/cryptopasta/blob/master/encrypt.go + +// NewEncryptionKey generates a random 256-bit key for Encrypt() and +// Decrypt(). It panics if the source of randomness fails. +func newEncryptionKey() *[32]byte { + key := [32]byte{} + _, err := io.ReadFull(cryptorand.Reader, key[:]) + if err != nil { + panic(err) + } + return &key +} + +// Encrypt encrypts data using 256-bit AES-GCM. This both hides the content of +// the data and provides a check that it hasn't been altered. Output takes the +// form nonce|ciphertext|tag where '|' indicates concatenation. +func encrypt(plaintext []byte, key *[32]byte) (ciphertext []byte, err error) { + block, err := aes.NewCipher(key[:]) + if err != nil { + return nil, err + } + + gcm, err := cipher.NewGCM(block) + if err != nil { + return nil, err + } + + nonce := make([]byte, gcm.NonceSize()) + _, err = io.ReadFull(cryptorand.Reader, nonce) + if err != nil { + return nil, err + } + + // Encode to hex so we avoid err: + // net/http: invalid byte 'ó' in Cookie.Value; dropping invalid bytes + ciphertext = gcm.Seal(nonce, nonce, plaintext, nil) + hexCiphertext := make([]byte, hex.EncodedLen(len(ciphertext))) + hex.Encode(hexCiphertext, ciphertext) + + return hexCiphertext, nil +} + +// Decrypt decrypts data using 256-bit AES-GCM. This both hides the content of +// the data and provides a check that it hasn't been altered. Expects input +// form nonce|ciphertext|tag where '|' indicates concatenation. +func decrypt(hexCiphertext []byte, key *[32]byte) (plaintext []byte, err error) { + block, err := aes.NewCipher(key[:]) + if err != nil { + return nil, err + } + + ciphertext := make([]byte, hex.DecodedLen(len(hexCiphertext))) + hex.Decode(ciphertext, hexCiphertext) + + gcm, err := cipher.NewGCM(block) + if err != nil { + return nil, err + } + + if len(ciphertext) < gcm.NonceSize() { + return nil, errors.New("malformed ciphertext") + } + + return gcm.Open(nil, + ciphertext[:gcm.NonceSize()], + ciphertext[gcm.NonceSize():], + nil, + ) +} + +// ----------------------------- CRYPTO HELPER FUNCS --------------------------- + +// generateRandomString generates a pseudo random string from a given set of +// chars s +func generateRandomString(n int, s string) string { + str := make([]byte, n) + l := len(s) + // Create random Participant ID + for i := range str { + str[i] = s[rand.Intn(l)] + } + return string(str) +} + +// generatePassword generates a pseudo random password +func generatePassword() (pw string) { + charSet := "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890!?_-&#@&()" + pw = generateRandomString(10, charSet) + return +} diff --git a/cmd/bs1in/database.go b/cmd/bs1in/database.go new file mode 100644 index 0000000..760b23a --- /dev/null +++ b/cmd/bs1in/database.go @@ -0,0 +1,34 @@ +package main + +import ( + "context" + "log" + "time" + + "go.mongodb.org/mongo-driver/mongo" + "go.mongodb.org/mongo-driver/mongo/options" +) + +var client mongo.Client +var colDbUsers *mongo.Collection +var mongoCtx context.Context + +func init() { + // Mongo DB setup + clientOptions := options.Client().ApplyURI("mongodb://" + config.DbUser + ":" + config.DbPass + "@" + config.DbHost + ":" + config.DbPort) + client, err := mongo.NewClient(clientOptions) + if err != nil { + log.Fatal(err) + } + // Cancel if Timeout + ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second) + defer cancel() + + // Connect to DB + err = client.Connect(ctx) + if err != nil { + log.Fatal(err) + } + + colDbUsers = client.Database("bs1in").Collection("db_users") +} diff --git a/cmd/bs1in/helpers.go b/cmd/bs1in/helpers.go new file mode 100644 index 0000000..e5244b8 --- /dev/null +++ b/cmd/bs1in/helpers.go @@ -0,0 +1,13 @@ +package main + +import ( + "go.mongodb.org/mongo-driver/bson/primitive" +) + +func unwrapObjectID(oid primitive.ObjectID) string { + return oid.String()[10:34] +} + +func wrapObjectID(oid string) (primitive.ObjectID, error) { + return primitive.ObjectIDFromHex(oid) +} diff --git a/cmd/bs1in/mail.go b/cmd/bs1in/mail.go new file mode 100644 index 0000000..06faa8a --- /dev/null +++ b/cmd/bs1in/mail.go @@ -0,0 +1,21 @@ +package main + +import ( + "fmt" + "net/smtp" +) + +func sendMail(to string, subject string, mailTxt string) error { + from := config.MailUser + header := `From: Philip - BS1IN <%s> +To: %s +Subject: %s + + +` + header = fmt.Sprintf(header, from, to, subject) + mailTxt = header + mailTxt + auth := smtp.PlainAuth("", from, config.MailPass, config.MailSmtpServer) + err := smtp.SendMail(config.MailSmtpServer+":"+config.MailSmtpPort, auth, from, []string{to}, []byte(mailTxt)) + return err +} diff --git a/cmd/bs1in/mail_test.go b/cmd/bs1in/mail_test.go new file mode 100644 index 0000000..d3e42a9 --- /dev/null +++ b/cmd/bs1in/mail_test.go @@ -0,0 +1,12 @@ +package main + +import ( + "testing" +) + +func TestSendMail(t *testing.T) { + err := sendMail("mrc1rz+8ca4wocgn0mz8@sharklasers.com", "Test Test 123", "Hello, this is just a test") + if err != nil { + t.Error(err) + } +} diff --git a/go.mod b/go.mod new file mode 100644 index 0000000..fb2a774 --- /dev/null +++ b/go.mod @@ -0,0 +1,19 @@ +module g.phga.de/phga/bs1in + +go 1.17 + +require go.mongodb.org/mongo-driver v1.7.3 + +require ( + github.com/go-stack/stack v1.8.0 // indirect + github.com/golang/snappy v0.0.1 // indirect + github.com/klauspost/compress v1.13.6 // indirect + github.com/pkg/errors v0.9.1 // indirect + github.com/xdg-go/pbkdf2 v1.0.0 // indirect + github.com/xdg-go/scram v1.0.2 // indirect + github.com/xdg-go/stringprep v1.0.2 // indirect + github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d // indirect + golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073 // indirect + golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e // indirect + golang.org/x/text v0.3.5 // indirect +) diff --git a/go.sum b/go.sum new file mode 100644 index 0000000..ce9b3e7 --- /dev/null +++ b/go.sum @@ -0,0 +1,116 @@ +github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/go-stack/stack v1.8.0 h1:5SgMzNM5HxrEjV0ww2lTmX6E2Izsfxas4+YHWRs3Lsk= +github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= +github.com/gobuffalo/attrs v0.0.0-20190224210810-a9411de4debd/go.mod h1:4duuawTqi2wkkpB4ePgWMaai6/Kc6WEz83bhFwpHzj0= +github.com/gobuffalo/depgen v0.0.0-20190329151759-d478694a28d3/go.mod h1:3STtPUQYuzV0gBVOY3vy6CfMm/ljR4pABfrTeHNLHUY= +github.com/gobuffalo/depgen v0.1.0/go.mod h1:+ifsuy7fhi15RWncXQQKjWS9JPkdah5sZvtHc2RXGlg= +github.com/gobuffalo/envy v1.6.15/go.mod h1:n7DRkBerg/aorDM8kbduw5dN3oXGswK5liaSCx4T5NI= +github.com/gobuffalo/envy v1.7.0/go.mod h1:n7DRkBerg/aorDM8kbduw5dN3oXGswK5liaSCx4T5NI= +github.com/gobuffalo/flect v0.1.0/go.mod h1:d2ehjJqGOH/Kjqcoz+F7jHTBbmDb38yXA598Hb50EGs= +github.com/gobuffalo/flect v0.1.1/go.mod h1:8JCgGVbRjJhVgD6399mQr4fx5rRfGKVzFjbj6RE/9UI= +github.com/gobuffalo/flect v0.1.3/go.mod h1:8JCgGVbRjJhVgD6399mQr4fx5rRfGKVzFjbj6RE/9UI= +github.com/gobuffalo/genny v0.0.0-20190329151137-27723ad26ef9/go.mod h1:rWs4Z12d1Zbf19rlsn0nurr75KqhYp52EAGGxTbBhNk= +github.com/gobuffalo/genny v0.0.0-20190403191548-3ca520ef0d9e/go.mod h1:80lIj3kVJWwOrXWWMRzzdhW3DsrdjILVil/SFKBzF28= +github.com/gobuffalo/genny v0.1.0/go.mod h1:XidbUqzak3lHdS//TPu2OgiFB+51Ur5f7CSnXZ/JDvo= +github.com/gobuffalo/genny v0.1.1/go.mod h1:5TExbEyY48pfunL4QSXxlDOmdsD44RRq4mVZ0Ex28Xk= +github.com/gobuffalo/gitgen v0.0.0-20190315122116-cc086187d211/go.mod h1:vEHJk/E9DmhejeLeNt7UVvlSGv3ziL+djtTr3yyzcOw= +github.com/gobuffalo/gogen v0.0.0-20190315121717-8f38393713f5/go.mod h1:V9QVDIxsgKNZs6L2IYiGR8datgMhB577vzTDqypH360= +github.com/gobuffalo/gogen v0.1.0/go.mod h1:8NTelM5qd8RZ15VjQTFkAW6qOMx5wBbW4dSCS3BY8gg= +github.com/gobuffalo/gogen v0.1.1/go.mod h1:y8iBtmHmGc4qa3urIyo1shvOD8JftTtfcKi+71xfDNE= +github.com/gobuffalo/logger v0.0.0-20190315122211-86e12af44bc2/go.mod h1:QdxcLw541hSGtBnhUc4gaNIXRjiDppFGaDqzbrBd3v8= +github.com/gobuffalo/mapi v1.0.1/go.mod h1:4VAGh89y6rVOvm5A8fKFxYG+wIW6LO1FMTG9hnKStFc= +github.com/gobuffalo/mapi v1.0.2/go.mod h1:4VAGh89y6rVOvm5A8fKFxYG+wIW6LO1FMTG9hnKStFc= +github.com/gobuffalo/packd v0.0.0-20190315124812-a385830c7fc0/go.mod h1:M2Juc+hhDXf/PnmBANFCqx4DM3wRbgDvnVWeG2RIxq4= +github.com/gobuffalo/packd v0.1.0/go.mod h1:M2Juc+hhDXf/PnmBANFCqx4DM3wRbgDvnVWeG2RIxq4= +github.com/gobuffalo/packr/v2 v2.0.9/go.mod h1:emmyGweYTm6Kdper+iywB6YK5YzuKchGtJQZ0Odn4pQ= +github.com/gobuffalo/packr/v2 v2.2.0/go.mod h1:CaAwI0GPIAv+5wKLtv8Afwl+Cm78K/I/VCm/3ptBN+0= +github.com/gobuffalo/syncx v0.0.0-20190224160051-33c29581e754/go.mod h1:HhnNqWY95UYwwW3uSASeV7vtgYkT2t16hJgV3AEPUpw= +github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4= +github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +github.com/google/go-cmp v0.5.2 h1:X2ev0eStA3AbceY54o37/0PQ/UWqKEiiO2dKL5OPaFM= +github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= +github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg= +github.com/karrick/godirwalk v1.8.0/go.mod h1:H5KPZjojv4lE+QYImBI8xVtrBRgYrIVsaRPx4tDPEn4= +github.com/karrick/godirwalk v1.10.3/go.mod h1:RoGL9dQei4vP9ilrpETWE8CLOZ1kiN0LhBygSwrAsHA= +github.com/klauspost/compress v1.13.6 h1:P76CopJELS0TiO2mebmnzgWaajssP/EszplttgQxcgc= +github.com/klauspost/compress v1.13.6/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk= +github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= +github.com/markbates/oncer v0.0.0-20181203154359-bf2de49a0be2/go.mod h1:Ld9puTsIW75CHf65OeIOkyKbteujpZVXDpWK6YGZbxE= +github.com/markbates/safe v1.0.1/go.mod h1:nAqgmRi7cY2nqMc92/bSEeQA+R4OheNU2T1kNSCBdG0= +github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc= +github.com/pelletier/go-toml v1.7.0/go.mod h1:vwGMzjaWMwyfHwgIBhI2YUM4fB6nL6lVAvS1LBMMhTE= +github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= +github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/rogpeppe/go-internal v1.1.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/rogpeppe/go-internal v1.2.2/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/sirupsen/logrus v1.4.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= +github.com/sirupsen/logrus v1.4.1/go.mod h1:ni0Sbl8bgC9z8RoU9G6nDWqqs/fq4eDPysMBDgk/93Q= +github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= +github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= +github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= +github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= +github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= +github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/tidwall/pretty v1.0.0 h1:HsD+QiTn7sK6flMKIvNmpqz1qrpP3Ps6jOKIKMooyg4= +github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk= +github.com/xdg-go/pbkdf2 v1.0.0 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c= +github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= +github.com/xdg-go/scram v1.0.2 h1:akYIkZ28e6A96dkWNJQu3nmCzH3YfwMPQExUYDaRv7w= +github.com/xdg-go/scram v1.0.2/go.mod h1:1WAq6h33pAW+iRreB34OORO2Nf7qel3VV3fjBj+hCSs= +github.com/xdg-go/stringprep v1.0.2 h1:6iq84/ryjjeRmMJwxutI51F2GIPlP5BfTvXHeYjyhBc= +github.com/xdg-go/stringprep v1.0.2/go.mod h1:8F9zXuvzgwmyT5DUm4GUfZGDdT3W+LCvS6+da4O5kxM= +github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d h1:splanxYIlg+5LfHAM6xpdFEAYOk8iySO56hMFq6uLyA= +github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d/go.mod h1:rHwXgn7JulP+udvsHwJoVG1YGAP6VLg4y9I5dyZdqmA= +go.mongodb.org/mongo-driver v1.7.3 h1:G4l/eYY9VrQAK/AUgkV0koQKzQnyddnWxrd/Etf0jIs= +go.mongodb.org/mongo-driver v1.7.3/go.mod h1:NqaYOwnXWr5Pm7AOpO5QFxKJ503nbMse/R79oO62zWg= +golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190422162423-af44ce270edf/go.mod h1:WFFai1msRO1wXaEeE5yQxYXgSfI8pQAWXbQop6sCtWE= +golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073 h1:xMPOj6Pz6UipU1wXLkrtqpHbR0AVFnyPEQq/wRWz9lM= +golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190412183630-56d357773e84/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e h1:vcxGaoTs7kV8m5Np9uUNQin4BrLOthgV7252N8V+FwY= +golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190403152447-81d4e9dc473e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190419153524-e8e3143a4f4a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190531175056-4c3a928424d2/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.5 h1:i6eZZ+zk0SOf0xgBpEpPD18qWcJda6q1sxt3S0kzyUQ= +golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190329151228-23e29df326fe/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190416151739-9c9e1878f421/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190420181800-aa740d480789/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190531172133-b3315ee88b7d/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= +golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4= +golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= +gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= diff --git a/web/static/css/bs1in.css b/web/static/css/bs1in.css new file mode 100644 index 0000000..a97c3fc --- /dev/null +++ b/web/static/css/bs1in.css @@ -0,0 +1,76 @@ +/******************* DEFINITIONS *******************/ +@font-face { + font-family: "OpenSans"; + src: url("../fonts/OpenSans-Regular.ttf"); +} +:root { + --red: #b65058; + --blue: #a3d4d6; + --green: #8baf45; + --purple: #514a63; + --grey1: #fafafa; + --grey2: #f1f2f6; + --grey3: #d1d2d6; + --black: #292930; +} + +/******************* GENERAL *******************/ + +body { + font-family: "OpenSans", sans-serif; + color: var(--black); +} +span { + margin: none; + padding: none; +} + +label, input[type=radio] { + cursor: pointer; +} + +select, button { + text-transform: none; + border-radius: 4px; + border: none; + height: 40px; + margin: 0; + padding: 0 10px; + text-align: center; + line-height: 40px; + font-weight: bold; + cursor: pointer; + font-family: inherit; + font-size: 100%; + overflow: visible; + -webkit-appearance: button; + background: var(--grey2); +} + +button:hover { + opacity: .8; +} + +input[type=text], +input[type=password], +input[type=email], +textarea { + border-radius: 4px; + border: 2px solid var(--grey2); + box-sizing: border-box; + width: 100%; + margin: 5px 0; + padding: 4px 5px; + font-family: inherit; + font-size: 100%; +} + +input[type=text], +input[type=password], +input[type=email] { + height: 35px; +} + +.hidden { + visibility: hidden; +} diff --git a/web/static/js/helper.js b/web/static/js/helper.js new file mode 100644 index 0000000..111fe1e --- /dev/null +++ b/web/static/js/helper.js @@ -0,0 +1,21 @@ +function toggleDisabled(obj, state) { + if (state == "on") { + obj.setAttribute("disabled", ""); + } else { + obj.removeAttribute("disabled"); + } +} + +function toggleHidden(obj, state) { + if (state == "off") { + obj.classList.remove("hidden"); + } else { + obj.classList.add("hidden"); + } +} + +function setInfo(obj, txt, bcolor, fcolor) { + obj.innerText = txt; + obj.style.background = `var(--${bcolor})`; + obj.style.color = `var(--${fcolor})`; +} \ No newline at end of file diff --git a/web/templates/header.html b/web/templates/header.html new file mode 100644 index 0000000..080419c --- /dev/null +++ b/web/templates/header.html @@ -0,0 +1,4 @@ +{{define "header"}} +
+
+{{end}} diff --git a/web/templates/layouts/base.html b/web/templates/layouts/base.html new file mode 100644 index 0000000..e0bc8c9 --- /dev/null +++ b/web/templates/layouts/base.html @@ -0,0 +1,21 @@ + + + + + + + + + + + Stuff I am too lazy to do myself + + + + {{block "header" .HeaderData}}{{end}} +
+ {{block "body" .BodyData}}{{end}} +
+ + {{block "scripts" .BodyData}}{{end}} + diff --git a/web/templates/main.html b/web/templates/main.html new file mode 100644 index 0000000..22566a3 --- /dev/null +++ b/web/templates/main.html @@ -0,0 +1,8 @@ +{{define "body"}} +
+
+{{end}} + +{{define "scripts"}} + +{{end}} \ No newline at end of file diff --git a/web/templates/soon.html b/web/templates/soon.html new file mode 100644 index 0000000..26ca42c --- /dev/null +++ b/web/templates/soon.html @@ -0,0 +1,3 @@ +{{define "body"}} +

❤ COMING SOON ❤

+{{end}}