388 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			388 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Go
		
	
	
	
| package main
 | |
| 
 | |
| import (
 | |
| 	"crypto/rand"
 | |
| 	"fmt"
 | |
| 	"github.com/gomodule/redigo/redis"
 | |
| 	"golang.org/x/crypto/bcrypt"
 | |
| 	"log"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| )
 | |
| 
 | |
| var globalPool *redis.Pool
 | |
| 
 | |
| const userPrefix string = "user:"
 | |
| const devPrefix string = "dev:"
 | |
| const confirmPrefix string = "confirm:"
 | |
| 
 | |
| func newPool() *redis.Pool {
 | |
| 	return &redis.Pool{
 | |
| 		// Maximum number of idle connections in the pool.
 | |
| 		MaxIdle: 80,
 | |
| 		// max number of connections
 | |
| 		MaxActive: 12000,
 | |
| 		// Dial is an application supplied function for creating and
 | |
| 		// configuring a connection.
 | |
| 		Dial: func() (redis.Conn, error) {
 | |
| 			c, err := redis.Dial("tcp", ":6379")
 | |
| 			if err != nil {
 | |
| 				panic(err.Error())
 | |
| 			}
 | |
| 			return c, err
 | |
| 		},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // ping tests connectivity for redis (PONG should be returned)
 | |
| func ping(c redis.Conn) error {
 | |
| 	// Send PING command to Redis
 | |
| 	// PING command returns a Redis "Simple String"
 | |
| 	// Use redis.String to convert the interface type to string
 | |
| 	s, err := redis.String(c.Do("PING"))
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	logit("PING Response = " + s)
 | |
| 	// Output: PONG
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| type Dev struct {
 | |
| 	Deveui      string
 | |
| 	Alias       string
 | |
| 	ActiveUntil string // Abo bezahlt bis TT.MM.YYYY
 | |
| }
 | |
| 
 | |
| func initDB() {
 | |
| 
 | |
| 	// newPool returns a pointer to a redis.Pool
 | |
| 	pool := newPool()
 | |
| 	// get a connection from the globalPool (redis.Conn)
 | |
| 	conn := pool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	globalPool = pool
 | |
| 
 | |
| 	// wir machen einen Connection Test
 | |
| 	ping(conn)
 | |
| 
 | |
| 	// Wir legen einen initialen Admin User an, falls es diesen noch nicht gibt
 | |
| 	if checkUserAvailable("joerg.lehmann@nbit.ch") {
 | |
| 		insertUser("joerg.lehmann@nbit.ch", "changeme123", "Y")
 | |
| 	}
 | |
| 
 | |
| }
 | |
| 
 | |
| func closeDB() {
 | |
| 	globalPool.Close()
 | |
| }
 | |
| 
 | |
| func getUsers() []string {
 | |
| 	res := []string{}
 | |
| 
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	logit("getUsers")
 | |
| 	users, err := redis.Strings(conn.Do("KEYS", userPrefix+"*"))
 | |
| 	if err == nil {
 | |
| 		logit("getUsers successful!")
 | |
| 		res = users
 | |
| 	} else {
 | |
| 		log.Print(err)
 | |
| 	}
 | |
| 
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| func updateTrackerSettings(trackerSettings Dev) error {
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	// SET object
 | |
| 	_, err := conn.Do("HMSET", devPrefix+trackerSettings.Deveui, "alias", trackerSettings.Alias)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func checkUserAvailable(username string) bool {
 | |
| 	logit("checkUserAvailable: User: " + username)
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	res, err := redis.Int(conn.Do("EXISTS", userPrefix+username))
 | |
| 	if err == nil {
 | |
| 		logit("Result of EXISTS: " + strconv.Itoa(res))
 | |
| 		return res == 0
 | |
| 	} else {
 | |
| 		logit("checkUserAvailable: Error to query Key Value Store")
 | |
| 		return false
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func getMyDevs(username string) []string {
 | |
| 	res := []string{}
 | |
| 
 | |
| 	if username == "" {
 | |
| 		return res
 | |
| 	}
 | |
| 
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	logit("getMyDevs: User: " + username)
 | |
| 	mydevs, err := redis.String(conn.Do("HGET", userPrefix+username, "my_devs"))
 | |
| 	if err == nil {
 | |
| 		logit("getMyDevs: mydevs: " + mydevs)
 | |
| 		res = strings.Split(mydevs, ",")
 | |
| 	} else {
 | |
| 		log.Print(err)
 | |
| 	}
 | |
| 
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| func getDevAlias(deveui string) string {
 | |
| 	res := deveui
 | |
| 
 | |
| 	if deveui == "" {
 | |
| 		return res
 | |
| 	}
 | |
| 
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	logit("getDevAlias: Deveui: " + deveui)
 | |
| 	alias, err := redis.String(conn.Do("HGET", devPrefix+deveui, "alias"))
 | |
| 	if err == nil {
 | |
| 		logit("getDevAlias: alias: " + alias)
 | |
| 		res = alias
 | |
| 	} else {
 | |
| 		log.Print(err)
 | |
| 	}
 | |
| 
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| func getActiveUntil(deveui string) string {
 | |
| 	res := ""
 | |
| 
 | |
| 	if deveui == "" {
 | |
| 		return res
 | |
| 	}
 | |
| 
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	logit("getActiveUntil: Deveui: " + deveui)
 | |
| 	activeuntil, err := redis.String(conn.Do("HGET", devPrefix+deveui, "active_until"))
 | |
| 	if err == nil {
 | |
| 		logit("getActiveUntil: activeuntil: " + activeuntil)
 | |
| 		res = activeuntil
 | |
| 	} else {
 | |
| 		log.Print(err)
 | |
| 	}
 | |
| 
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| func getYearlyAboCost(deveui string) int {
 | |
| 	res := 0
 | |
| 
 | |
| 	logit("getYearlyAboCost: Deveui: " + deveui)
 | |
| 
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	// first we get the default
 | |
| 	yearlyAboCostDefault, err := redis.Int(conn.Do("HGET", "settings", "default_yearly_abo_cost"))
 | |
| 	if err == nil {
 | |
| 		yearlyAboCost, err := redis.Int(conn.Do("HGET", devPrefix+deveui, "yearly_abo_cost"))
 | |
| 		if err == nil {
 | |
| 			res = yearlyAboCost
 | |
| 		} else {
 | |
| 			res = yearlyAboCostDefault
 | |
| 		}
 | |
| 	} else {
 | |
| 		log.Print(err)
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| func AboExpired(deveui string) bool {
 | |
| 	active_until := getActiveUntil(deveui)
 | |
| 
 | |
| 	layout := "02.01.2006"
 | |
| 	t, _ := time.Parse(layout, active_until)
 | |
| 
 | |
| 	return t.Before(time.Now())
 | |
| }
 | |
| 
 | |
| func prolongActivation(deveui string, years int) (string, error) {
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	active_until_old, err := redis.String(conn.Do("HGET", devPrefix+deveui, "active_until"))
 | |
| 	if err == nil {
 | |
| 		logit("prolongActivation: active_until: " + active_until_old)
 | |
| 	} else {
 | |
| 		log.Print(err)
 | |
| 	}
 | |
| 
 | |
| 	layout := "02.01.2006"
 | |
| 	t, err := time.Parse(layout, active_until_old)
 | |
| 
 | |
| 	if err != nil {
 | |
| 		fmt.Println(err)
 | |
| 	}
 | |
| 	fmt.Println(t.Unix())
 | |
| 
 | |
| 	var t_new time.Time
 | |
| 	if t.Before(time.Now()) {
 | |
| 		t_new = time.Now().AddDate(years, 0, 0)
 | |
| 	} else {
 | |
| 		t_new = t.AddDate(years, 0, 0)
 | |
| 	}
 | |
| 	active_until_new := t_new.Format(layout)
 | |
| 
 | |
| 	// SET object
 | |
| 	_, err1 := conn.Do("HMSET", devPrefix+deveui, "active_until", active_until_new)
 | |
| 	if err1 != nil {
 | |
| 		return "", err1
 | |
| 	}
 | |
| 
 | |
| 	return active_until_new, nil
 | |
| }
 | |
| 
 | |
| func randString(n int) string {
 | |
| 	const alphanum = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"
 | |
| 	var bytes = make([]byte, n)
 | |
| 	rand.Read(bytes)
 | |
| 	for i, b := range bytes {
 | |
| 		bytes[i] = alphanum[b%byte(len(alphanum))]
 | |
| 	}
 | |
| 	return string(bytes)
 | |
| }
 | |
| 
 | |
| func insertUser(username, password, is_admin string) {
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 	logit("insertUser: " + username)
 | |
| 	pwd := []byte(password)
 | |
| 	hashedPassword, err := bcrypt.GenerateFromPassword(pwd, bcrypt.DefaultCost)
 | |
| 	if err != nil {
 | |
| 		logit("insertUser: Error with bcrypt.GenerateFromPassword, User: " + username)
 | |
| 		return
 | |
| 	}
 | |
| 	confirm_id := ""
 | |
| 
 | |
| 	_, err = conn.Do("HMSET", userPrefix+username, "password", string(hashedPassword), "new_password", string(hashedPassword), "confirm_id", confirm_id, "last_login", "", "my_devs", "")
 | |
| 
 | |
| 	if err != nil {
 | |
| 		logit("insertUser: Error inserting User: " + username)
 | |
| 		return
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func updateUser(username, password string) {
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 	logit("updateUser: " + username)
 | |
| 	pwd := []byte(password)
 | |
| 	hashedPassword, err := bcrypt.GenerateFromPassword(pwd, bcrypt.DefaultCost)
 | |
| 	if err != nil {
 | |
| 		logit("updateUser: Error with bcrypt.GenerateFromPassword, User: " + username)
 | |
| 		return
 | |
| 	}
 | |
| 	confirm_id := randString(30)
 | |
| 
 | |
| 	_, err = conn.Do("HMSET", userPrefix+username, "new_password", string(hashedPassword), "confirm_id", confirm_id)
 | |
| 	if err != nil {
 | |
| 		logit("updateUser: Error updateing User: " + username)
 | |
| 		return
 | |
| 	}
 | |
| 	_, err = conn.Do("SET", confirmPrefix+confirm_id, username)
 | |
| 	if err != nil {
 | |
| 		logit("updateUser: Error inserting confirm_id: " + confirm_id + ": " + username)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	sendEmail(username, confirm_id)
 | |
| }
 | |
| 
 | |
| func checkLoginCredentials(username, password string) bool {
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	logit("checkLoginCredentials: called with username,password: " + username + "," + password)
 | |
| 	pwd, err := redis.String(conn.Do("HGET", userPrefix+username, "password"))
 | |
| 	if err == nil {
 | |
| 		logit("checkLoginCredentials: pwd: " + pwd + " CMD: HGET " + userPrefix + username + userPrefix + username + " confirm_id")
 | |
| 		cid, err := redis.String(conn.Do("HGET", userPrefix+username, "confirm_id"))
 | |
| 		if err == nil {
 | |
| 			logit("checkLoginCredentials: cid: " + cid)
 | |
| 			if !(err != nil && cid != "") {
 | |
| 				logit("checkLoginCredentials: pwd: " + pwd)
 | |
| 				if err != nil {
 | |
| 					return false
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		log.Print(err)
 | |
| 		return false
 | |
| 	}
 | |
| 
 | |
| 	hashedPassword := []byte(pwd)
 | |
| 	err = bcrypt.CompareHashAndPassword(hashedPassword, []byte(password))
 | |
| 	return err == nil
 | |
| }
 | |
| 
 | |
| func updateLoginTime(username string) {
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	_, err := conn.Do("HSET", userPrefix+username, "last_login", time.Now().UTC().Format("2006-01-02 15:04:05"))
 | |
| 	if err != nil {
 | |
| 		logit("updateUser: Error updateing User: " + username)
 | |
| 		return
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func confirmUser(confirm_id string) bool {
 | |
| 	conn := globalPool.Get()
 | |
| 	defer conn.Close()
 | |
| 
 | |
| 	u, err := redis.String(conn.Do("GET", confirmPrefix+confirm_id))
 | |
| 	if err != nil {
 | |
| 		logit("confirmUser: Error with searching confirm_id: " + confirm_id)
 | |
| 		return false
 | |
| 	}
 | |
| 	new_password, err := redis.String(conn.Do("HGET", userPrefix+u, "new_password"))
 | |
| 	if err != nil {
 | |
| 		logit("confirmUser: Error with getting new_password: " + u)
 | |
| 		return false
 | |
| 	}
 | |
| 	_, err = conn.Do("HMSET", userPrefix+u, "confirm_id", "", "password", new_password)
 | |
| 	if err != nil {
 | |
| 		logit("confirmUser: Error updateing User: " + u)
 | |
| 		return false
 | |
| 	}
 | |
| 	_, err = conn.Do("DEL", confirmPrefix+confirm_id)
 | |
| 	if err != nil {
 | |
| 		logit("confirmUser: Error deleting confirm_id: " + confirm_id)
 | |
| 		return false
 | |
| 	}
 | |
| 
 | |
| 	return true
 | |
| }
 |