Initial setup

This commit is contained in:
$(pass /github/name)
2024-06-26 13:08:39 +02:00
parent 164c07e3d1
commit fbb3bdfcb3
16 changed files with 279 additions and 0 deletions

8
cmd/membership/main.go Normal file
View File

@@ -0,0 +1,8 @@
package main
import "GoMembership/internal/server"
func main() {
server.Run()
}

11
go.mod Normal file
View File

@@ -0,0 +1,11 @@
module GoMembership
go 1.18
require (
github.com/gorilla/mux v1.8.0
golang.org/x/crypto v0.0.0-20220411214040-896ed404bb0e
)
replace golang.org/x/crypto => golang.org/x/crypto v0.0.0-20220411214040-896ed404bb0e

5
go.sum Normal file
View File

@@ -0,0 +1,5 @@
github.com/gorilla/mux v1.8.0 h1:vGHr6G7CRByRCFwUmpxP8Cj8ljy1OV8ZrRH6qrOEzXM=
github.com/gorilla/mux v1.8.0/go.mod h1:vGHr6G7CRByRCFwUmpxP8Cj8ljy1OV8ZrRH6qrOEzXM=
golang.org/x/crypto v0.0.0-20220411214040-896ed404bb0e h1:aIjQLr9igSCqCxL1cOsTSG91TtOGStlwrPvHxGpQuWk=
golang.org/x/crypto v0.0.0

20
internal/config/config.go Normal file
View File

@@ -0,0 +1,20 @@
package config
import (
"os"
)
type Config struct {
DBUser string
DBPassword string
DBName string
}
func LoadConfig() *Config {
return &Config{
DBUser: os.Getenv("DB_USER"),
DBPassword: os.Getenv("DB_PASSWORD"),
DBName: os.Getenv("DB_NAME"),
}
}

View File

@@ -0,0 +1,31 @@
package controllers
import (
"encoding/json"
"net/http"
"GoMembership/internal/models"
"GoMembership/internal/services"
)
type UserController struct {
service services.UserService
}
func NewUserController(service services.UserService) *UserController {
return &UserController{service}
}
func (uc *UserController) RegisterUser(w http.ResponseWriter, r *http.Request) {
var user models.User
if err := json.NewDecoder(r.Body).Decode(&user); err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
if err := uc.service.RegisterUser(&user); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
w.WriteHeader(http.StatusCreated)
}

23
internal/database/db.go Normal file
View File

@@ -0,0 +1,23 @@
package database
import (
"database/sql"
"log"
"GoMembership/internal/config"
_ "github.com/go-sql-driver/mysql"
)
func Connect() *sql.DB {
cfg := config.LoadConfig()
dsn := cfg.DBUser + ":" + cfg.DBPassword + "@/" + cfg.DBName
db, err := sql.Open("mysql", dsn)
if err != nil {
log.Fatal(err)
}
if err := db.Ping(); err != nil {
log.Fatal(err)
}
return db
}

View File

@@ -0,0 +1,11 @@
package handlers
import (
"net/http"
"GoMembership/internal/controllers"
)
func RegisterUserHandler(controller *controllers.UserController) http.Handler {
return http.HandlerFunc(controller.RegisterUser)
}

View File

@@ -0,0 +1,13 @@
package middlewares
import (
"net/http"
)
func AuthMiddleware(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
// Authentication logic here
next.ServeHTTP(w, r)
})
}

14
internal/models/user.go Normal file
View File

@@ -0,0 +1,14 @@
package models
import "time"
type User struct {
ID int `json:"id"`
FirstName string `json:"first_name"`
LastName string `json:"last_name"`
Email string `json:"email"`
Password string `json:"password"`
CreatedAt time.Time `json:"created_at"`
UpdatedAt time.Time `json:"updated_at"`
}

View File

@@ -0,0 +1,25 @@
package repositories
import (
"database/sql"
"GoMembership/internal/models"
)
type UserRepository interface {
CreateUser(user *models.User) error
}
type userRepository struct {
db *sql.DB
}
func NewUserRepository(db *sql.DB) UserRepository {
return &userRepository{db}
}
func (r *userRepository) CreateUser(user *models.User) error {
query := "INSERT INTO users (first_name, last_name, email, password, created_at, updated_at) VALUES (?, ?, ?, ?, ?, ?)"
_, err := r.db.Exec(query, user.FirstName, user.LastName, user.Email, user.Password, user.CreatedAt, user.UpdatedAt)
return err
}

13
internal/routes/routes.go Normal file
View File

@@ -0,0 +1,13 @@
package routes
import (
"net/http"
"GoMembership/internal/controllers"
"github.com/gorilla/mux"
)
func RegisterRoutes(router *mux.Router, userController *controllers.UserController) {
router.HandleFunc("/register", userController.RegisterUser).Methods("POST")
}

32
internal/server/server.go Normal file
View File

@@ -0,0 +1,32 @@
package server
import (
"database/sql"
"log"
"GoMembership/internal/controllers"
"GoMembership/internal/repositories"
"GoMembership/internal/routes"
"GoMembership/internal/services"
"net/http"
"github.com/gorilla/mux"
_ "github.com/go-sql-driver/mysql"
)
func Run() {
db, err := sql.Open("mysql", "user:password@/dbname")
if err != nil {
log.Fatal(err)
}
defer db.Close()
userRepo := repositories.NewUserRepository(db)
userService := services.NewUserService(userRepo)
userController := controllers.NewUserController(userService)
router := mux.NewRouter()
routes.RegisterRoutes(router, userController)
log.Fatal(http.ListenAndServe(":8080", router))
}

View File

@@ -0,0 +1,29 @@
package services
import (
"GoMembership/internal/models"
"GoMembership/internal/repositories"
"golang.org/x/crypto/bcrypt"
)
type UserService interface {
RegisterUser(user *models.User) error
}
type userService struct {
repo repositories.UserRepository
}
func NewUserService(repo repositories.UserRepository) UserService {
return &userService{repo}
}
func (s *userService) RegisterUser(user *models.User) error {
hashedPassword, err := bcrypt.GenerateFromPassword([]byte(user.Password), bcrypt.DefaultCost)
if err != nil {
return err
}
user.Password = string(hashedPassword)
return s.repo.CreateUser(user)
}

View File

@@ -0,0 +1,10 @@
package utils
import "regexp"
func IsEmailValid(email string) bool {
regex := `^[a-z0-9._%+\-]+@[a-z0-9.\-]+\.[a-z]{2,}$`
re := regexp.MustCompile(regex)
return re.MatchString(email)
}

10
pkg/errors/errors.go Normal file
View File

@@ -0,0 +1,10 @@
package errors
import "errors"
var (
ErrUserNotFound = errors.New("user not found")
ErrInvalidEmail = errors.New("invalid email")
// Add other custom errors here
)

24
pkg/logger/logger.go Normal file
View File

@@ -0,0 +1,24 @@
package logger
import (
"log"
"os"
)
var (
Info *log.Logger
Warning *log.Logger
Error *log.Logger
)
func init() {
file, err := os.OpenFile("app.log", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0666)
if err != nil {
log.Fatal(err)
}
Info = log.New(file, "INFO: ", log.Ldate|log.Ltime|log.Lshortfile)
Warning = log.New(file, "WARNING: ", log.Ldate|log.Ltime|log.Lshortfile)
Error = log.New(file, "ERROR: ", log.Ldate|log.Ltime|log.Lshortfile)
}