medication/main.go

285 lines
6.8 KiB
Go
Raw Normal View History

2024-08-22 17:23:11 -03:00
package main
import (
2024-08-22 19:25:14 -03:00
"crypto/rand"
2024-08-22 17:23:11 -03:00
"encoding/json"
"fmt"
"html/template"
2024-08-22 19:25:14 -03:00
"math"
"math/big"
2024-08-22 17:23:11 -03:00
"net/http"
"os"
2024-08-22 19:25:14 -03:00
"strconv"
2024-08-22 17:23:11 -03:00
"time"
)
type Timestamp struct {
Current int64 `json:"current"`
Next int64 `json:"next"`
}
type Store struct {
2024-08-22 19:25:14 -03:00
Username string `json:"username"`
Timestamps []Timestamp `json:"timestamps"`
2024-08-22 17:23:11 -03:00
}
type PageData struct {
2024-08-22 19:25:14 -03:00
Title string
Message string
ShowLogin bool
ShowInput bool
2024-08-22 17:23:11 -03:00
}
2024-08-22 19:25:14 -03:00
var Username string
var Password string
var DataDir string
var DataStore string
var Authorised []int64 // List of authorised "uuids"
2024-08-22 17:23:11 -03:00
func main() {
2024-08-22 19:25:14 -03:00
// Get environment variables
Username = os.Getenv("USERNAME")
fmt.Println("Username: ", Username)
Password = os.Getenv("API_KEY")
fmt.Println("API_KEY: ", Password)
DataDir = os.Getenv("DATA_DIR")
if DataDir == "" {
DataDir = "./data"
}
fmt.Println("DATA_DIR: ", DataDir)
DataStore = DataDir + "/data.json"
Reset := os.Getenv("RESET")
if Reset == "true" {
if err := os.Remove(DataStore); err != nil {
fmt.Println("Error removing file, check permissions")
}
if err := os.Remove(DataDir); err != nil {
fmt.Println("Error removing directory, check permissions")
}
}
2024-08-22 17:23:11 -03:00
// Check if directory exists, if not create it
2024-08-22 19:25:14 -03:00
if _, err := os.Stat(DataDir); os.IsNotExist(err) {
err := os.Mkdir(DataDir, 0755)
2024-08-22 17:23:11 -03:00
if err != nil {
fmt.Println("Error creating directory, check permissions")
os.Exit(1)
}
}
2024-08-22 19:25:14 -03:00
// Check if file exists, if not create it, and write empty json
if _, err := os.Stat(DataStore); os.IsNotExist(err) {
if _, err := os.Create(DataStore); err != nil {
fmt.Println("Error creating file, check permissions")
os.Exit(1)
}
if file, err := os.OpenFile(DataStore, os.O_RDWR, os.ModePerm); err != nil {
fmt.Println("Error opening file, check permissions")
os.Exit(1)
} else {
encoder := json.NewEncoder(file)
store := Store{
Username: Username,
Timestamps: []Timestamp{},
}
if err = encoder.Encode(store); err != nil {
fmt.Println("Error encoding file")
os.Exit(1)
}
if err = file.Close(); err != nil {
fmt.Println("Error closing file")
os.Exit(1)
}
2024-08-22 17:23:11 -03:00
}
}
// Start server
http.HandleFunc("/", index)
2024-08-22 19:25:14 -03:00
http.HandleFunc("/post", post)
http.HandleFunc("/login", login)
http.HandleFunc("/logout", logout)
2024-08-22 17:23:11 -03:00
if err := http.ListenAndServe(":8080", nil); err != nil {
fmt.Println("Error starting server")
os.Exit(1)
}
}
2024-08-22 19:25:14 -03:00
func commit(next int64) {
2024-08-22 17:23:11 -03:00
current := time.Now().Unix()
2024-08-22 19:25:14 -03:00
// Next is in hours, so convert to seconds
next = next * 60 * 60
// Next is the time when the form can be submitted again
next = current + next
2024-08-22 17:23:11 -03:00
2024-08-22 19:25:14 -03:00
// Read from file
if file, err := os.OpenFile(DataStore, os.O_RDWR, os.ModePerm); err != nil {
2024-08-22 17:23:11 -03:00
fmt.Println("Error opening file, check permissions")
os.Exit(1)
} else {
2024-08-22 19:25:14 -03:00
decoder := json.NewDecoder(file)
store := Store{}
if err = decoder.Decode(&store); err != nil {
fmt.Println("Error decoding file")
os.Exit(1)
}
store.Timestamps = append(store.Timestamps, Timestamp{current, next})
if _, err := file.Seek(0, 0); err != nil {
fmt.Println("Error seeking file")
os.Exit(1)
}
// Write to file
2024-08-22 17:23:11 -03:00
encoder := json.NewEncoder(file)
2024-08-22 19:25:14 -03:00
if err = encoder.Encode(store); err != nil {
2024-08-22 17:23:11 -03:00
fmt.Println("Error encoding file")
os.Exit(1)
}
if err = file.Close(); err != nil {
fmt.Println("Error closing file")
os.Exit(1)
}
}
}
2024-08-22 19:25:14 -03:00
func query() bool {
2024-08-22 17:23:11 -03:00
// Check if file exists
2024-08-22 19:25:14 -03:00
if file, err := os.Open(DataStore); err != nil {
fmt.Println("Error opening file, check permissions")
os.Exit(1)
} else {
// Decode file
decoder := json.NewDecoder(file)
store := Store{}
if err = decoder.Decode(&store); err != nil {
fmt.Println("Error decoding file")
os.Exit(1)
2024-08-22 17:23:11 -03:00
}
2024-08-22 19:25:14 -03:00
if err = file.Close(); err != nil {
fmt.Println("Error closing file")
os.Exit(1)
}
if len(store.Timestamps) == 0 {
return true
}
// Check if user has already submitted form today
current := time.Now().Unix()
// Get last timestamp
last := store.Timestamps[len(store.Timestamps)-1]
// If current time is greater than the next time the form can be submitted
if current > last.Next {
return true
2024-08-22 17:23:11 -03:00
}
}
2024-08-22 19:25:14 -03:00
return false
2024-08-22 17:23:11 -03:00
}
func index(w http.ResponseWriter, r *http.Request) {
tmpl := template.Must(template.ParseFiles("index.html"))
var data PageData
2024-08-22 19:25:14 -03:00
fmt.Println("GET request received")
unauthenticated := PageData{
Title: "Log in",
Message: "Please enter your username and password",
ShowLogin: true,
ShowInput: false,
}
authenticated := PageData{
Title: "View data",
Message: "Welcome back " + Username + ". Check in after the timeout",
ShowLogin: false,
ShowInput: true,
}
// Check for session cookie
cookie, err := r.Cookie("session")
if err != nil {
data = unauthenticated
} else {
// Parse cookie to int64
if session, err := strconv.ParseInt(cookie.Value, 10, 64); err != nil {
fmt.Println("Error parsing cookie")
data = unauthenticated
} else {
found := false
for _, uuid := range Authorised {
if uuid == session {
found = true
}
}
if found {
data = authenticated
} else {
// If cookie is not in Authorised list, remove it
data = unauthenticated
}
2024-08-22 17:23:11 -03:00
}
}
2024-08-22 19:25:14 -03:00
authenticated.ShowInput = query()
2024-08-22 17:23:11 -03:00
if err := tmpl.Execute(w, data); err != nil {
http.Error(w, "Error rendering template", http.StatusInternalServerError)
}
}
2024-08-22 19:25:14 -03:00
func post(w http.ResponseWriter, r *http.Request) {
fmt.Println("POST request received")
if err := r.ParseForm(); err != nil {
http.Error(w, "Error parsing form", http.StatusBadRequest)
return
}
next := r.Form.Get("next")
if next, err := strconv.ParseInt(next, 10, 64); err != nil {
http.Error(w, "Invalid time", http.StatusBadRequest)
return
} else {
commit(next)
}
http.Redirect(w, r, "/", http.StatusSeeOther)
return
}
func login(w http.ResponseWriter, r *http.Request) {
fmt.Println("LOGIN request received")
if err := r.ParseForm(); err != nil {
http.Error(w, "Error parsing form", http.StatusBadRequest)
return
}
username := r.Form.Get("username")
password := r.Form.Get("password")
if username != os.Getenv("USERNAME") || password != os.Getenv("PASSWORD") {
http.Error(w, "Invalid username or password", http.StatusUnauthorized)
return
}
// Create session cookie
if session, err := rand.Int(rand.Reader, big.NewInt(math.MaxInt64)); err != nil {
http.Error(w, "Error creating session", http.StatusInternalServerError)
return
} else {
Authorised = append(Authorised, session.Int64())
cookie := http.Cookie{
Name: "session",
Value: session.String(),
Expires: time.Now().Add(24 * time.Hour),
}
http.SetCookie(w, &cookie)
http.Redirect(w, r, "/", http.StatusSeeOther)
return
}
}
func logout(w http.ResponseWriter, r *http.Request) {
fmt.Println("LOGOUT request received")
cookie := http.Cookie{
Name: "session",
Value: "",
Expires: time.Now().Add(-1 * time.Hour),
}
http.SetCookie(w, &cookie)
http.Redirect(w, r, "/", http.StatusSeeOther)
return
}