Initial setup
This commit is contained in:
8
cmd/membership/main.go
Normal file
8
cmd/membership/main.go
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import "GoMembership/internal/server"
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
server.Run()
|
||||||
|
}
|
||||||
|
|
||||||
11
go.mod
Normal file
11
go.mod
Normal 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
5
go.sum
Normal 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
20
internal/config/config.go
Normal 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"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
31
internal/controllers/user_controller.go
Normal file
31
internal/controllers/user_controller.go
Normal 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
23
internal/database/db.go
Normal 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
|
||||||
|
}
|
||||||
|
|
||||||
11
internal/handlers/user_handler.go
Normal file
11
internal/handlers/user_handler.go
Normal 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)
|
||||||
|
}
|
||||||
|
|
||||||
13
internal/middlewares/auth_middleware.go
Normal file
13
internal/middlewares/auth_middleware.go
Normal 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
14
internal/models/user.go
Normal 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"`
|
||||||
|
}
|
||||||
|
|
||||||
25
internal/repositories/user_repository.go
Normal file
25
internal/repositories/user_repository.go
Normal 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
13
internal/routes/routes.go
Normal 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
32
internal/server/server.go
Normal 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))
|
||||||
|
}
|
||||||
|
|
||||||
29
internal/services/user_service.go
Normal file
29
internal/services/user_service.go
Normal 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)
|
||||||
|
}
|
||||||
|
|
||||||
10
internal/utils/validators.go
Normal file
10
internal/utils/validators.go
Normal 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
10
pkg/errors/errors.go
Normal 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
24
pkg/logger/logger.go
Normal 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)
|
||||||
|
}
|
||||||
|
|
||||||
Reference in New Issue
Block a user