Pārlūkot izejas kodu

test code for new ui

volixer 6 mēneši atpakaļ
revīzija
746c6f322a
8 mainītis faili ar 702 papildinājumiem un 0 dzēšanām
  1. 37 0
      go.mod
  2. 91 0
      go.sum
  3. 401 0
      main.go
  4. 51 0
      templates/dashboard.html
  5. 54 0
      templates/edit.html
  6. 40 0
      templates/groups.html
  7. 28 0
      templates/login.html
  8. BIN
      users.db

+ 37 - 0
go.mod

@@ -0,0 +1,37 @@
+module app
+
+go 1.23.3
+
+require (
+	github.com/gin-gonic/gin v1.10.0
+	github.com/mattn/go-sqlite3 v1.14.24
+)
+
+require (
+	github.com/bytedance/sonic v1.11.6 // indirect
+	github.com/bytedance/sonic/loader v0.1.1 // indirect
+	github.com/cloudwego/base64x v0.1.4 // indirect
+	github.com/cloudwego/iasm v0.2.0 // indirect
+	github.com/gabriel-vasile/mimetype v1.4.3 // indirect
+	github.com/gin-contrib/sse v0.1.0 // indirect
+	github.com/go-playground/locales v0.14.1 // indirect
+	github.com/go-playground/universal-translator v0.18.1 // indirect
+	github.com/go-playground/validator/v10 v10.20.0 // indirect
+	github.com/goccy/go-json v0.10.2 // indirect
+	github.com/json-iterator/go v1.1.12 // indirect
+	github.com/klauspost/cpuid/v2 v2.2.7 // indirect
+	github.com/leodido/go-urn v1.4.0 // indirect
+	github.com/mattn/go-isatty v0.0.20 // indirect
+	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
+	github.com/modern-go/reflect2 v1.0.2 // indirect
+	github.com/pelletier/go-toml/v2 v2.2.2 // indirect
+	github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
+	github.com/ugorji/go/codec v1.2.12 // indirect
+	golang.org/x/arch v0.8.0 // indirect
+	golang.org/x/crypto v0.23.0 // indirect
+	golang.org/x/net v0.25.0 // indirect
+	golang.org/x/sys v0.20.0 // indirect
+	golang.org/x/text v0.15.0 // indirect
+	google.golang.org/protobuf v1.34.1 // indirect
+	gopkg.in/yaml.v3 v3.0.1 // indirect
+)

+ 91 - 0
go.sum

@@ -0,0 +1,91 @@
+github.com/bytedance/sonic v1.11.6 h1:oUp34TzMlL+OY1OUWxHqsdkgC/Zfc85zGqw9siXjrc0=
+github.com/bytedance/sonic v1.11.6/go.mod h1:LysEHSvpvDySVdC2f87zGWf6CIKJcAvqab1ZaiQtds4=
+github.com/bytedance/sonic/loader v0.1.1 h1:c+e5Pt1k/cy5wMveRDyk2X4B9hF4g7an8N3zCYjJFNM=
+github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU=
+github.com/cloudwego/base64x v0.1.4 h1:jwCgWpFanWmN8xoIUHa2rtzmkd5J2plF/dnLS6Xd/0Y=
+github.com/cloudwego/base64x v0.1.4/go.mod h1:0zlkT4Wn5C6NdauXdJRhSKRlJvmclQ1hhJgA0rcu/8w=
+github.com/cloudwego/iasm v0.2.0 h1:1KNIy1I1H9hNNFEEH3DVnI4UujN+1zjpuk6gwHLTssg=
+github.com/cloudwego/iasm v0.2.0/go.mod h1:8rXZaNYT2n95jn+zTI1sDr+IgcD2GVs0nlbbQPiEFhY=
+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/gabriel-vasile/mimetype v1.4.3 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0=
+github.com/gabriel-vasile/mimetype v1.4.3/go.mod h1:d8uq/6HKRL6CGdk+aubisF/M5GcPfT7nKyLpA0lbSSk=
+github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
+github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
+github.com/gin-gonic/gin v1.10.0 h1:nTuyha1TYqgedzytsKYqna+DfLos46nTv2ygFy86HFU=
+github.com/gin-gonic/gin v1.10.0/go.mod h1:4PMNQiOhvDRa013RKVbsiNwoyezlm2rm0uX/T7kzp5Y=
+github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
+github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
+github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
+github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY=
+github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY=
+github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY=
+github.com/go-playground/validator/v10 v10.20.0 h1:K9ISHbSaI0lyB2eWMPJo+kOS/FBExVwjEviJTixqxL8=
+github.com/go-playground/validator/v10 v10.20.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM=
+github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU=
+github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I=
+github.com/google/go-cmp v0.5.5 h1:Khx7svrCpmxxtHBq5j2mp/xVjsi8hQMfNLvJFAlrGgU=
+github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
+github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
+github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
+github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
+github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg=
+github.com/klauspost/cpuid/v2 v2.2.7 h1:ZWSB3igEs+d0qvnxR/ZBzXVmxkgt8DdzP6m9pfuVLDM=
+github.com/klauspost/cpuid/v2 v2.2.7/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws=
+github.com/knz/go-libedit v1.10.1/go.mod h1:MZTVkCWyz0oBc7JOWP3wNAzd002ZbM/5hgShxwh4x8M=
+github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ=
+github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI=
+github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
+github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
+github.com/mattn/go-sqlite3 v1.14.24 h1:tpSp2G2KyMnnQu99ngJ47EIkWVmliIizyZBfPrBWDRM=
+github.com/mattn/go-sqlite3 v1.14.24/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
+github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
+github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
+github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
+github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
+github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
+github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM=
+github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs=
+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/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
+github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
+github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
+github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA=
+github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
+github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
+github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
+github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
+github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
+github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
+github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
+github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
+github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI=
+github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
+github.com/ugorji/go/codec v1.2.12 h1:9LC83zGrHhuUA9l16C9AHXAqEV/2wBQ4nkvumAE65EE=
+github.com/ugorji/go/codec v1.2.12/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg=
+golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
+golang.org/x/arch v0.8.0 h1:3wRIsP3pM4yUptoR96otTUOXI367OS0+c9eeRi9doIc=
+golang.org/x/arch v0.8.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys=
+golang.org/x/crypto v0.23.0 h1:dIJU/v2J8Mdglj/8rJ6UUOM3Zc9zLZxVZwwxMooUSAI=
+golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8=
+golang.org/x/net v0.25.0 h1:d/OCCoBEUq33pjydKrGQhw7IlUPI2Oylr+8qLx49kac=
+golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM=
+golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/sys v0.20.0 h1:Od9JTbYCk261bKm4M/mw7AklTlFYIa0bIp9BgSm1S8Y=
+golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
+golang.org/x/text v0.15.0 h1:h1V/4gjBv8v9cjcR6+AR5+/cIYK5N/WAgiv4xlsEtAk=
+golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
+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=
+google.golang.org/protobuf v1.34.1 h1:9ddQBjfCyZPOHPUiPxpYESBLc+T8P3E+Vo4IbKZgFWg=
+google.golang.org/protobuf v1.34.1/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos=
+gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
+gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
+gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
+gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
+gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
+nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50=
+rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=

+ 401 - 0
main.go

@@ -0,0 +1,401 @@
+package main
+
+import (
+	"crypto/sha256"
+	"crypto/rand"
+	"math/big"
+	"database/sql"
+	"encoding/hex"
+	"log"
+	// "strconv"
+	"net/http"
+
+	"github.com/gin-gonic/gin"
+	_ "github.com/mattn/go-sqlite3"
+)
+
+type User struct {
+	ID           int
+	Name         string
+	UIDNumber    int
+	PrimaryGroup int
+	OtherGroups  string
+	GivenName    string
+	SN           string
+	Mail         string
+	LoginShell   string
+	HomeDirectory string
+	Disabled     int
+	PassSHA256   string
+	OTPSecret    string
+	YubiKey      string
+	SSHKeys      string
+	CustAttr     string
+}
+
+// Structure représentant un groupe LDAP
+type LDAPGroup struct {
+	ID       int    `json:"id"`
+	Name     string `json:"name"`
+	GIDNumber int   `json:"gidnumber"`
+}
+
+func createTable(db *sql.DB) {
+	query := `
+CREATE TABLE IF NOT EXISTS users (
+	id INTEGER PRIMARY KEY,
+	name TEXT NOT NULL,
+	uidnumber INTEGER NOT NULL,
+	primarygroup INTEGER NOT NULL,
+	othergroups TEXT DEFAULT '',
+	givenname TEXT DEFAULT '',
+	sn TEXT DEFAULT '',
+	mail TEXT DEFAULT '',
+	loginshell TEXT DEFAULT '',
+	homedirectory TEXT DEFAULT '',
+	disabled SMALLINT DEFAULT 0,
+	passsha256 TEXT DEFAULT '',
+	otpsecret TEXT DEFAULT '',
+	yubikey TEXT DEFAULT '',
+	sshkeys TEXT DEFAULT '',
+	custattr TEXT DEFAULT '{}'
+);
+`
+	_, err := db.Exec(query)
+	if err != nil {
+		log.Fatalf("Failed to create table: %v", err)
+	}
+}
+
+func hashPassword(password string) string {
+	hash := sha256.New()
+	hash.Write([]byte(password))
+	return hex.EncodeToString(hash.Sum(nil))
+}
+
+func generateRandomPassword(maxLength int) string {
+	const charset = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!@#$%^&*()-_+=<>?/|"
+	password := make([]byte, maxLength)
+	for i := range password {
+		charIndex, _ := rand.Int(rand.Reader, big.NewInt(int64(len(charset))))
+		password[i] = charset[charIndex.Int64()]
+	}
+	return string(password)
+}
+
+func generatePasswordSHA256(maxLength int) string {
+	randomPassword := generateRandomPassword(maxLength)
+	return hashPassword(randomPassword)
+}
+
+func authenticateUser(db *sql.DB, name, hashedPassword string) bool {
+	var user User
+	query := "SELECT * FROM users WHERE name = ? AND passsha256 = ?"
+	err := db.QueryRow(query, name, hashedPassword).Scan(
+		&user.ID, &user.Name, &user.UIDNumber, &user.PrimaryGroup, &user.OtherGroups,
+		&user.GivenName, &user.SN, &user.Mail, &user.LoginShell, &user.HomeDirectory,
+		&user.Disabled, &user.PassSHA256, &user.OTPSecret, &user.YubiKey,
+		&user.SSHKeys, &user.CustAttr,
+	)
+	return err == nil
+}
+
+// Fonction pour récupérer le dernier `uidnumber` dans la table `users`
+func getLastUIDNumber(db *sql.DB) (int, error) {
+	// Requête SQL pour récupérer le dernier uidnumber
+	var lastUIDNumber int
+	err := db.QueryRow("SELECT MAX(uidnumber) FROM users").Scan(&lastUIDNumber)
+	if err != nil {
+		if err == sql.ErrNoRows {
+			// Si aucune ligne n'est trouvée, retourner 0
+			return 0, nil
+		}
+		return 0, err
+	}
+	return lastUIDNumber, nil
+}
+
+func createUser(db *sql.DB, name, password string) bool {
+	hashedPassword := hashPassword(password)
+	query := "INSERT INTO users (name, passsha256) VALUES (?, ?)"
+	_, err := db.Exec(query, name, hashedPassword)
+	return err == nil
+}
+
+func fetchUsers(db *sql.DB) []User {
+	query := "SELECT id, name, sn, givenname, mail FROM users"
+	rows, err := db.Query(query)
+	if err != nil {
+		log.Fatalf("Failed to fetch users: %v", err)
+	}
+	defer rows.Close()
+
+	var users []User
+	for rows.Next() {
+		var user User
+		err := rows.Scan(&user.ID, &user.Name, &user.SN, &user.GivenName, &user.Mail)
+		if err != nil {
+			log.Fatalf("Failed to scan user: %v", err)
+		}
+		users = append(users, user)
+	}
+	return users
+}
+
+func fetchUserByID(db *sql.DB, id string) User {
+	query := "SELECT * FROM users WHERE id = ?"
+	var user User
+	err := db.QueryRow(query, id).Scan(
+		&user.ID, &user.Name, &user.UIDNumber, &user.PrimaryGroup, &user.OtherGroups,
+		&user.GivenName, &user.SN, &user.Mail, &user.LoginShell, &user.HomeDirectory,
+		&user.Disabled, &user.PassSHA256, &user.OTPSecret, &user.YubiKey,
+		&user.SSHKeys, &user.CustAttr,
+	)
+	if err != nil {
+		log.Fatalf("Failed to fetch user: %v", err)
+	}
+	return user
+}
+
+func updateUser(db *sql.DB, id, name, givenName, sn, mail string, sshkeys string) {
+	query := "UPDATE users SET name = ?, givenname = ?, sn = ?, mail = ?, sshkeys = ? WHERE id = ?"
+	_, err := db.Exec(query, name, givenName, sn, mail, sshkeys, id)
+	if err != nil {
+		log.Fatalf("Failed to update user: %v", err)
+	}
+}
+
+// Fonction pour récupérer tous les groupes LDAP
+func fetchLDAPGroups(db *sql.DB) ([]LDAPGroup, error) {
+	// Préparer la requête SQL pour récupérer les groupes
+	rows, err := db.Query("SELECT id, name, gidnumber FROM ldapgroups")
+	if err != nil {
+		return nil, err
+	}
+	defer rows.Close()
+
+	// Initialiser un tableau pour stocker les groupes
+	var groups []LDAPGroup
+
+	// Parcourir les résultats de la requête
+	for rows.Next() {
+		var group LDAPGroup
+		if err := rows.Scan(&group.ID, &group.Name, &group.GIDNumber); err != nil {
+			return nil, err
+		}
+		// Ajouter le groupe à la liste
+		groups = append(groups, group)
+	}
+
+	// Vérifier si des erreurs sont survenues lors du parcours des lignes
+	if err := rows.Err(); err != nil {
+		return nil, err
+	}
+
+	return groups, nil
+}
+
+func main() {
+	// Initialize database connection
+	db, err := sql.Open("sqlite3", "./users.db")
+	if err != nil {
+		log.Fatal(err)
+	}
+	defer db.Close()
+
+	// Create table if it doesn't exist
+	createTable(db)
+
+	// Initialize Gin router
+	router := gin.Default()
+
+	// Serve HTML templates
+	router.LoadHTMLGlob("templates/*")
+
+	// Middleware for session authentication
+	router.Use(func(c *gin.Context) {
+		// Skip authentication check for login page
+		if c.Request.URL.Path == "/" || c.Request.URL.Path == "/login" {
+			c.Next()
+			return
+		}
+		// Check if user is authenticated
+		session, err := c.Cookie("session")
+		if err != nil || session != "authenticated" {
+			c.Redirect(http.StatusFound, "/")
+			c.Abort()
+			return
+		}
+		c.Next()
+	})
+
+	// Routes
+	router.GET("/", func(c *gin.Context) {
+		c.HTML(http.StatusOK, "login.html", nil)
+	})
+
+	router.POST("/login", func(c *gin.Context) {
+		name := c.PostForm("name")
+		password := c.PostForm("password")
+
+		if authenticateUser(db, name, hashPassword(password)) {
+			// c.String(http.StatusOK, "Login successful!")
+			// Création d'un cookie de session
+			c.SetCookie("session", "authenticated", 3600, "/", "localhost", false, true)
+			c.Redirect(http.StatusFound, "/dashboard")
+		} else {
+			c.String(http.StatusUnauthorized, "Invalid credentials")
+		}
+	})
+	
+	router.GET("/dashboard", func(c *gin.Context) {
+		users := fetchUsers(db)
+		c.HTML(http.StatusOK, "dashboard.html", gin.H{"users": users})
+	})
+
+	router.POST("/edit/:id", func(c *gin.Context) {
+		id := c.Param("id")
+		name := c.PostForm("name")
+		givenName := c.PostForm("givenname")
+		sn := c.PostForm("sn")
+		mail := c.PostForm("mail")
+		sshkeys := c.PostForm("sshkeys")
+		updateUser(db, id, name, givenName, sn, mail, sshkeys)
+		c.Redirect(http.StatusFound, "/dashboard")
+	})
+	
+	router.GET("/edit/:id", func(c *gin.Context) {
+		id := c.Param("id")
+		user := fetchUserByID(db, id)
+		groups, err := fetchLDAPGroups(db)
+		if err != nil {
+			log.Println("Erreur lors de la récupération des groupes :", err)
+			c.String(http.StatusInternalServerError, "Erreur lors de la récupération des groupes")
+			return
+		}
+		c.HTML(http.StatusOK, "edit.html", gin.H{
+			"user": user,
+			"groups": groups,
+		})
+	})
+
+	// Route GET pour afficher le formulaire de création
+	router.GET("/new", func(c *gin.Context) {
+		// Rendre la page "edit.html" avec un utilisateur vide
+		c.HTML(http.StatusOK, "edit.html", gin.H{"user": map[string]string{
+			"ID":       "",
+			"Name":     "",
+			"GivenName": "",
+			"SN":       "",
+			"Mail":     "",
+			"SSHKeys":  "",
+		}})
+	})
+
+/*
+	// Route POST pour insérer un nouvel utilisateur
+	router.POST("/new", func(c *gin.Context) {
+		name := c.PostForm("name")
+		givenName := c.PostForm("givenname")
+		sn := c.PostForm("sn")
+		mail := c.PostForm("mail")
+		sshkeys := c.PostForm("sshkeys")
+
+		// Appeler la fonction pour récupérer le dernier uidnumber
+		lastUID, err := getLastUIDNumber(db)
+		if err != nil {
+			log.Fatal(err)
+		}
+		nextUID := lastUID+1
+
+		// Génération automatique d'un mot de passe aléatoire
+		randomPassword := generateRandomPassword(32)
+		hashedPassword := hashPassword(randomPassword)
+
+		// Insérer dans la base de données
+		_, err := db.Exec(`
+			INSERT INTO users (name, uidnumber, givenname, sn, mail, passsha256, sshkeys)
+			VALUES (?, ?, ?, ?, ?, ?, ?)`, name, nextUID, givenName, sn, mail, hashedPassword, sshkeys)
+
+		if err != nil {
+			log.Println("Erreur lors de l'insertion :", err)
+			c.String(http.StatusInternalServerError, "Erreur lors de l'ajout de l'utilisateur")
+			return
+		}
+
+		// Redirection vers le tableau de bord après la création
+		c.Redirect(http.StatusFound, "/dashboard")
+	})
+
+
+	router.POST("/update/:id", func(c *gin.Context) {
+		userID := c.Param("id")
+	
+		// Récupérer les valeurs du formulaire
+		name := c.PostForm("name")
+		sn := c.PostForm("sn")
+		givenname := c.PostForm("givenname")
+		mail := c.PostForm("mail")
+		primarygroup := c.PostForm("group")
+	
+		// Convertir groupID en entier
+		groupIDInt, err := strconv.Atoi(groupID)
+		if err != nil {
+			log.Println("Erreur lors de la conversion du groupID:", err)
+			c.String(http.StatusBadRequest, "Erreur dans la conversion du groupID")
+			return
+		}
+	
+		// Mettre à jour l'utilisateur dans la base de données
+		_, err = db.Exec(`
+			UPDATE users SET name = ?, sn = ?, givenname = ?, mail = ?, primarygroup = ?
+			WHERE id = ?`, name, sn, givenname, mail, groupIDInt, userID)
+		if err != nil {
+			log.Println("Erreur lors de la mise à jour de l'utilisateur :", err)
+			c.String(http.StatusInternalServerError, "Erreur lors de la mise à jour de l'utilisateur")
+			return
+		}
+	
+		// Rediriger vers la page dashboard après mise à jour
+		c.Redirect(http.StatusFound, "/dashboard")
+	})
+	*/
+	/*
+	// Route pour afficher la liste des utilisateurs
+	router.GET("/users", func(c *gin.Context) {
+		users := fetchUsers(db) // Assurez-vous de disposer de la fonction fetchUsers qui retourne les utilisateurs
+		if err != nil {
+			log.Println("Erreur lors de la récupération des utilisateurs :", err)
+			c.String(http.StatusInternalServerError, "Erreur lors de la récupération des utilisateurs")
+			return
+		}
+		c.HTML(http.StatusOK, "users.html", gin.H{
+			"users": users,
+		})
+	})
+	*/
+
+	// Route pour afficher la liste des groupes
+	router.GET("/groups", func(c *gin.Context) {
+		groups, err := fetchLDAPGroups(db) // Utilisez la fonction fetchLDAPGroups que vous avez déjà définie
+		if err != nil {
+			log.Println("Erreur lors de la récupération des groupes :", err)
+			c.String(http.StatusInternalServerError, "Erreur lors de la récupération des groupes")
+			return
+		}
+		c.HTML(http.StatusOK, "groups.html", gin.H{
+			"groups": groups,
+		})
+	})
+
+	// logout
+	router.GET("/logout", func(c *gin.Context) {
+		c.SetCookie("session", "", -1, "/", "localhost", false, true)
+		c.Redirect(http.StatusFound, "/")
+	})
+
+	// Start the server
+	router.Run(":8080")
+}
+
+

+ 51 - 0
templates/dashboard.html

@@ -0,0 +1,51 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>Dashboard</title>
+    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0-alpha3/dist/css/bootstrap.min.css" rel="stylesheet">
+</head>
+<body class="bg-light">
+    <nav class="navbar navbar-expand-lg navbar-light bg-white border-bottom mb-4">
+        <div class="container">
+            <a class="navbar-brand" href="/dashboard">User Management</a>
+            <div class="d-flex">
+                <a href="/logout" class="btn btn-danger btn-sm">Logout</a>
+            </div>
+        </div>
+    </nav>
+    <div class="container mt-5">
+        <div class="d-flex justify-content-between align-items-center mb-4">
+            <h1>Dashboard</h1>
+            <a href="/new" class="btn btn-success">New User</a>
+        </div>
+        <table class="table table-striped">
+            <thead>
+                <tr>
+                    <th>ID</th>
+                    <th>Name</th>
+                    <th>Surname</th>
+                    <th>Given Name</th>
+                    <th>Email</th>
+                    <th>Actions</th>
+                </tr>
+            </thead>
+            <tbody>
+                {{range .users}}
+                <tr>
+                    <td>{{.ID}}</td>
+                    <td>{{.Name}}</td>
+                    <td>{{.SN}}</td>
+                    <td>{{.GivenName}}</td>
+                    <td>{{.Mail}}</td>
+                    <td>
+                        <a href="/edit/{{.ID}}" class="btn btn-sm btn-primary">Edit</a>
+                    </td>
+                </tr>
+                {{end}}
+            </tbody>
+        </table>
+    </div>
+</body>
+</html>

+ 54 - 0
templates/edit.html

@@ -0,0 +1,54 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>New/Edit User</title>
+    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0-alpha3/dist/css/bootstrap.min.css" rel="stylesheet">
+</head>
+<body class="bg-light">
+    <nav class="navbar navbar-expand-lg navbar-light bg-white border-bottom mb-4">
+        <div class="container">
+            <a class="navbar-brand" href="/dashboard">User Management</a>
+            <div class="d-flex">
+                <a href="/logout" class="btn btn-danger btn-sm">Logout</a>
+            </div>
+        </div>
+    </nav>
+    <div class="container mt-5">
+        <h1 class="mb-4">{{if .user.ID}}Edit User{{else}}New User{{end}}</h1>
+        <form action="{{if .user.ID}}/edit/{{.user.ID}}{{else}}/new{{end}}" method="post">
+            <div class="mb-3">
+                <label for="name" class="form-label">Name</label>
+                <input type="text" class="form-control" id="name" name="name" value="{{.user.Name}}" required>
+            </div>
+            <div class="mb-3">
+                <label for="givenname" class="form-label">Given Name</label>
+                <input type="text" class="form-control" id="givenname" name="givenname" value="{{.user.GivenName}}" required>
+            </div>
+            <div class="mb-3">
+                <label for="sn" class="form-label">Surname</label>
+                <input type="text" class="form-control" id="sn" name="sn" value="{{.user.SN}}" required>
+            </div>
+            <div class="mb-3">
+                <label for="mail" class="form-label">Email</label>
+                <input type="email" class="form-control" id="mail" name="mail" value="{{.user.Mail}}" required>
+            </div>
+            <div class="mb-3">
+                <label for="sshkeys" class="form-label">SSHKeys</label>
+                <input type="sshkeys" class="form-control" id="sshkeys" name="sshkeys" value="{{.user.SSHKeys}}" required>
+            </div>
+            <div class="mb-3">
+                <label for="group" class="form-label">Group</label>
+                <select class="form-select" id="group" name="group" required>
+                    <option value="">Select a Group</option>
+                    {{range .groups}}
+                        <option value="{{.ID}}" {{if eq .ID}}selected{{end}}>{{.Name}}</option>
+                    {{end}}
+                </select>
+            </div>
+            <button type="submit" class="btn btn-primary">{{if .user.ID}}Update{{else}}Create{{end}} User</button>
+        </form>
+    </div>
+</body>
+</html>

+ 40 - 0
templates/groups.html

@@ -0,0 +1,40 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>LDAP Groups</title>
+    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0-alpha3/dist/css/bootstrap.min.css" rel="stylesheet">
+</head>
+<body class="bg-light">
+    <nav class="navbar navbar-expand-lg navbar-light bg-white border-bottom mb-4">
+        <div class="container">
+            <a class="navbar-brand" href="/dashboard">User Management</a>
+            <div class="d-flex">
+                <a href="/logout" class="btn btn-danger btn-sm">Logout</a>
+            </div>
+        </div>
+    </nav>
+    <div class="container mt-5">
+        <h1 class="mb-4">Groups</h1>
+        <table class="table table-striped">
+            <thead>
+                <tr>
+                    <th>ID</th>
+                    <th>Name</th>
+                    <th>GID Number</th>
+                </tr>
+            </thead>
+            <tbody>
+                {{range .groups}}
+                <tr>
+                    <td>{{.ID}}</td>
+                    <td>{{.Name}}</td>
+                    <td>{{.GIDNumber}}</td>
+                </tr>
+                {{end}}
+            </tbody>
+        </table>
+    </div>
+</body>
+</html>

+ 28 - 0
templates/login.html

@@ -0,0 +1,28 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>Login</title>
+    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0-alpha3/dist/css/bootstrap.min.css" rel="stylesheet">
+</head>
+<body class="bg-light">
+    <div class="container d-flex justify-content-center align-items-center vh-100">
+        <div class="card p-4 shadow-lg" style="width: 100%; max-width: 400px;">
+            <h2 class="text-center mb-4">Login</h2>
+            <form action="/login" method="post">
+                <div class="mb-3">
+                    <label for="name" class="form-label">Name</label>
+                    <input type="name" class="form-control" id="name" name="name" required>
+                </div>
+                <div class="mb-3">
+                    <label for="password" class="form-label">Password</label>
+                    <input type="password" class="form-control" id="password" name="password" required>
+                </div>
+                <button type="submit" class="btn btn-primary w-100">Login</button>
+            </form>
+        </div>
+    </div>
+    <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0-alpha3/dist/js/bootstrap.bundle.min.js"></script>
+</body>
+</html>

BIN
users.db