244 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
| package util
 | |
| 
 | |
| import (
 | |
| 	"encoding/base64"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"path"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/ngoduykhanh/wireguard-ui/model"
 | |
| 	"github.com/sdomino/scribble"
 | |
| 	"github.com/skip2/go-qrcode"
 | |
| 	"golang.zx2c4.com/wireguard/wgctrl/wgtypes"
 | |
| )
 | |
| 
 | |
| const dbPath = "./db"
 | |
| const defaultUsername = "admin"
 | |
| const defaultPassword = "admin"
 | |
| const defaultServerAddress = "10.252.1.0/24"
 | |
| const defaultServerPort = 51820
 | |
| const defaultDNS = "1.1.1.1"
 | |
| const defaultMTU = 1450
 | |
| const defaultPersistentKeepalive = 15
 | |
| const defaultConfigFilePath = "/etc/wireguard/wg0.conf"
 | |
| 
 | |
| // DBConn to initialize the database connection
 | |
| func DBConn() (*scribble.Driver, error) {
 | |
| 	db, err := scribble.New(dbPath, nil)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return db, nil
 | |
| }
 | |
| 
 | |
| // InitDB to create the default database
 | |
| func InitDB() error {
 | |
| 	var clientPath string = path.Join(dbPath, "clients")
 | |
| 	var serverPath string = path.Join(dbPath, "server")
 | |
| 	var serverInterfacePath string = path.Join(serverPath, "interfaces.json")
 | |
| 	var serverKeyPairPath string = path.Join(serverPath, "keypair.json")
 | |
| 	var globalSettingPath string = path.Join(serverPath, "global_settings.json")
 | |
| 	var userPath string = path.Join(serverPath, "users.json")
 | |
| 
 | |
| 	// create directories if they do not exist
 | |
| 	if _, err := os.Stat(clientPath); os.IsNotExist(err) {
 | |
| 		os.Mkdir(clientPath, os.ModePerm)
 | |
| 	}
 | |
| 	if _, err := os.Stat(serverPath); os.IsNotExist(err) {
 | |
| 		os.Mkdir(serverPath, os.ModePerm)
 | |
| 	}
 | |
| 
 | |
| 	// server's interface
 | |
| 	if _, err := os.Stat(serverInterfacePath); os.IsNotExist(err) {
 | |
| 		db, err := DBConn()
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		serverInterface := new(model.ServerInterface)
 | |
| 		serverInterface.Addresses = []string{defaultServerAddress}
 | |
| 		serverInterface.ListenPort = defaultServerPort
 | |
| 		serverInterface.UpdatedAt = time.Now().UTC()
 | |
| 		db.Write("server", "interfaces", serverInterface)
 | |
| 	}
 | |
| 
 | |
| 	// server's key pair
 | |
| 	if _, err := os.Stat(serverKeyPairPath); os.IsNotExist(err) {
 | |
| 		db, err := DBConn()
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		key, err := wgtypes.GeneratePrivateKey()
 | |
| 		if err != nil {
 | |
| 			return scribble.ErrMissingCollection
 | |
| 		}
 | |
| 		serverKeyPair := new(model.ServerKeypair)
 | |
| 		serverKeyPair.PrivateKey = key.String()
 | |
| 		serverKeyPair.PublicKey = key.PublicKey().String()
 | |
| 		serverKeyPair.UpdatedAt = time.Now().UTC()
 | |
| 		db.Write("server", "keypair", serverKeyPair)
 | |
| 	}
 | |
| 
 | |
| 	// global settings
 | |
| 	if _, err := os.Stat(globalSettingPath); os.IsNotExist(err) {
 | |
| 		db, err := DBConn()
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		publicInterface, err := GetPublicIP()
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		globalSetting := new(model.GlobalSetting)
 | |
| 		globalSetting.EndpointAddress = publicInterface.IPAddress
 | |
| 		globalSetting.DNSServers = []string{defaultDNS}
 | |
| 		globalSetting.MTU = defaultMTU
 | |
| 		globalSetting.PersistentKeepalive = defaultPersistentKeepalive
 | |
| 		globalSetting.ConfigFilePath = defaultConfigFilePath
 | |
| 		globalSetting.UpdatedAt = time.Now().UTC()
 | |
| 		db.Write("server", "global_settings", globalSetting)
 | |
| 	}
 | |
| 
 | |
| 	// user info
 | |
| 	if _, err := os.Stat(userPath); os.IsNotExist(err) {
 | |
| 		db, err := DBConn()
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		user := new(model.User)
 | |
| 		user.Username = defaultUsername
 | |
| 		user.Password = defaultPassword
 | |
| 		db.Write("server", "users", user)
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // GetUser func to query user info from the database
 | |
| func GetUser() (model.User, error) {
 | |
| 	user := model.User{}
 | |
| 
 | |
| 	db, err := DBConn()
 | |
| 	if err != nil {
 | |
| 		return user, err
 | |
| 	}
 | |
| 
 | |
| 	if err := db.Read("server", "users", &user); err != nil {
 | |
| 		return user, err
 | |
| 	}
 | |
| 
 | |
| 	return user, nil
 | |
| }
 | |
| 
 | |
| // GetGlobalSettings func to query global settings from the database
 | |
| func GetGlobalSettings() (model.GlobalSetting, error) {
 | |
| 	settings := model.GlobalSetting{}
 | |
| 
 | |
| 	db, err := DBConn()
 | |
| 	if err != nil {
 | |
| 		return settings, err
 | |
| 	}
 | |
| 
 | |
| 	if err := db.Read("server", "global_settings", &settings); err != nil {
 | |
| 		return settings, err
 | |
| 	}
 | |
| 
 | |
| 	return settings, nil
 | |
| }
 | |
| 
 | |
| // GetServer func to query Server setting from the database
 | |
| func GetServer() (model.Server, error) {
 | |
| 	server := model.Server{}
 | |
| 
 | |
| 	db, err := DBConn()
 | |
| 	if err != nil {
 | |
| 		return server, err
 | |
| 	}
 | |
| 
 | |
| 	// read server interface information
 | |
| 	serverInterface := model.ServerInterface{}
 | |
| 	if err := db.Read("server", "interfaces", &serverInterface); err != nil {
 | |
| 		return server, err
 | |
| 	}
 | |
| 
 | |
| 	// read server key pair information
 | |
| 	serverKeyPair := model.ServerKeypair{}
 | |
| 	if err := db.Read("server", "keypair", &serverKeyPair); err != nil {
 | |
| 		return server, err
 | |
| 	}
 | |
| 
 | |
| 	// create Server object and return
 | |
| 	server.Interface = &serverInterface
 | |
| 	server.KeyPair = &serverKeyPair
 | |
| 	return server, nil
 | |
| }
 | |
| 
 | |
| // GetClients to get all clients from the database
 | |
| func GetClients(hasQRCode bool) ([]model.ClientData, error) {
 | |
| 	var clients []model.ClientData
 | |
| 
 | |
| 	db, err := DBConn()
 | |
| 	if err != nil {
 | |
| 		return clients, err
 | |
| 	}
 | |
| 
 | |
| 	// read all client json file in "clients" directory
 | |
| 	records, err := db.ReadAll("clients")
 | |
| 	if err != nil {
 | |
| 		return clients, err
 | |
| 	}
 | |
| 
 | |
| 	// build the ClientData list
 | |
| 	for _, f := range records {
 | |
| 		client := model.Client{}
 | |
| 		clientData := model.ClientData{}
 | |
| 
 | |
| 		// get client info
 | |
| 		if err := json.Unmarshal([]byte(f), &client); err != nil {
 | |
| 			return clients, fmt.Errorf("cannot decode client json structure: %v", err)
 | |
| 		}
 | |
| 
 | |
| 		// generate client qrcode image in base64
 | |
| 		if hasQRCode {
 | |
| 			server, _ := GetServer()
 | |
| 			globalSettings, _ := GetGlobalSettings()
 | |
| 
 | |
| 			png, _ := qrcode.Encode(BuildClientConfig(client, server, globalSettings), qrcode.Medium, 256)
 | |
| 			if err == nil {
 | |
| 				clientData.QRCode = "data:image/png;base64," + base64.StdEncoding.EncodeToString([]byte(png))
 | |
| 			} else {
 | |
| 				fmt.Print("Cannot generate QR code: ", err)
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// create the list of clients and their qrcode data
 | |
| 		clientData.Client = &client
 | |
| 		clients = append(clients, clientData)
 | |
| 	}
 | |
| 
 | |
| 	return clients, nil
 | |
| }
 | |
| 
 | |
| // GetClientByID func to query a client from the database
 | |
| func GetClientByID(clientID string) (model.Client, error) {
 | |
| 	client := model.Client{}
 | |
| 
 | |
| 	db, err := DBConn()
 | |
| 	if err != nil {
 | |
| 		return client, err
 | |
| 	}
 | |
| 
 | |
| 	// read client information
 | |
| 	if err := db.Read("clients", clientID, &client); err != nil {
 | |
| 		return client, err
 | |
| 	}
 | |
| 
 | |
| 	return client, nil
 | |
| }
 |