mirror of
https://github.com/lordmathis/lemma.git
synced 2025-11-05 15:44:21 +00:00
Use InsertStruct
This commit is contained in:
@@ -10,11 +10,12 @@ import (
|
|||||||
|
|
||||||
// CreateSession inserts a new session record into the database
|
// CreateSession inserts a new session record into the database
|
||||||
func (db *database) CreateSession(session *models.Session) error {
|
func (db *database) CreateSession(session *models.Session) error {
|
||||||
query := NewQuery(db.dbType).
|
query, err := NewQuery(db.dbType).
|
||||||
Insert("sessions", "id", "user_id", "refresh_token", "expires_at", "created_at").
|
InsertStruct(session, "sessions")
|
||||||
Values(5).
|
if err != nil {
|
||||||
AddArgs(session.ID, session.UserID, session.RefreshToken, session.ExpiresAt, session.CreatedAt)
|
return fmt.Errorf("failed to create query: %w", err)
|
||||||
_, err := db.Exec(query.String(), query.Args()...)
|
}
|
||||||
|
_, err = db.Exec(query.String(), query.Args()...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to store session: %w", err)
|
return fmt.Errorf("failed to store session: %w", err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,9 +8,10 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type DBField struct {
|
type DBField struct {
|
||||||
Name string
|
Name string
|
||||||
Value any
|
Value any
|
||||||
Type reflect.Type
|
Type reflect.Type
|
||||||
|
useDefault bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func StructTagsToFields(s any) ([]DBField, error) {
|
func StructTagsToFields(s any) ([]DBField, error) {
|
||||||
@@ -47,14 +48,28 @@ func StructTagsToFields(s any) ([]DBField, error) {
|
|||||||
tag = toSnakeCase(f.Name)
|
tag = toSnakeCase(f.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
if strings.Contains(tag, "omitempty") && reflect.DeepEqual(v.Field(i).Interface(), reflect.Zero(f.Type).Interface()) {
|
useDefault := false
|
||||||
continue
|
if strings.Contains(tag, ",") {
|
||||||
|
parts := strings.Split(tag, ",")
|
||||||
|
tag = parts[0]
|
||||||
|
|
||||||
|
for _, opt := range parts[1:] {
|
||||||
|
switch opt {
|
||||||
|
case "omitempty":
|
||||||
|
if reflect.DeepEqual(v.Field(i).Interface(), reflect.Zero(f.Type).Interface()) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
case "default":
|
||||||
|
useDefault = true
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fields = append(fields, DBField{
|
fields = append(fields, DBField{
|
||||||
Name: tag,
|
Name: tag,
|
||||||
Value: v.Field(i).Interface(),
|
Value: v.Field(i).Interface(),
|
||||||
Type: f.Type,
|
Type: f.Type,
|
||||||
|
useDefault: useDefault,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return fields, nil
|
return fields, nil
|
||||||
@@ -86,6 +101,10 @@ func (q *Query) InsertStruct(s any, table string) (*Query, error) {
|
|||||||
values := make([]any, 0, len(fields))
|
values := make([]any, 0, len(fields))
|
||||||
|
|
||||||
for _, f := range fields {
|
for _, f := range fields {
|
||||||
|
if f.useDefault {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
columns = append(columns, f.Name)
|
columns = append(columns, f.Name)
|
||||||
values = append(values, f.Value)
|
values = append(values, f.Value)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,11 +17,14 @@ func (db *database) CreateUser(user *models.User) (*models.User, error) {
|
|||||||
}
|
}
|
||||||
defer tx.Rollback()
|
defer tx.Rollback()
|
||||||
|
|
||||||
query := NewQuery(db.dbType).
|
query, err := NewQuery(db.dbType).
|
||||||
Insert("users", "email", "display_name", "password_hash", "role").
|
InsertStruct(user, "users")
|
||||||
Values(4).
|
|
||||||
AddArgs(user.Email, user.DisplayName, user.PasswordHash, user.Role).
|
if err != nil {
|
||||||
Returning("id", "created_at")
|
return nil, fmt.Errorf("failed to create query: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
query.Returning("id", "created_at")
|
||||||
|
|
||||||
err = tx.QueryRow(query.String(), query.Args()...).
|
err = tx.QueryRow(query.String(), query.Args()...).
|
||||||
Scan(&user.ID, &user.CreatedAt)
|
Scan(&user.ID, &user.CreatedAt)
|
||||||
@@ -69,23 +72,16 @@ func (db *database) CreateUser(user *models.User) (*models.User, error) {
|
|||||||
func (db *database) createWorkspaceTx(tx *sql.Tx, workspace *models.Workspace) error {
|
func (db *database) createWorkspaceTx(tx *sql.Tx, workspace *models.Workspace) error {
|
||||||
log := getLogger().WithGroup("users")
|
log := getLogger().WithGroup("users")
|
||||||
|
|
||||||
insertQuery := NewQuery(db.dbType).
|
insertQuery, err := NewQuery(db.dbType).
|
||||||
Insert("workspaces",
|
InsertStruct(workspace, "workspaces")
|
||||||
"user_id", "name",
|
|
||||||
"theme", "auto_save", "show_hidden_files",
|
|
||||||
"git_enabled", "git_url", "git_user", "git_token",
|
|
||||||
"git_auto_commit", "git_commit_msg_template",
|
|
||||||
"git_commit_name", "git_commit_email").
|
|
||||||
Values(13).
|
|
||||||
AddArgs(
|
|
||||||
workspace.UserID, workspace.Name,
|
|
||||||
workspace.Theme, workspace.AutoSave, workspace.ShowHiddenFiles,
|
|
||||||
workspace.GitEnabled, workspace.GitURL, workspace.GitUser, workspace.GitToken,
|
|
||||||
workspace.GitAutoCommit, workspace.GitCommitMsgTemplate,
|
|
||||||
workspace.GitCommitName, workspace.GitCommitEmail).
|
|
||||||
Returning("id")
|
|
||||||
|
|
||||||
err := tx.QueryRow(insertQuery.String(), insertQuery.Args()...).Scan(&workspace.ID)
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to create query: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
insertQuery.Returning("id")
|
||||||
|
|
||||||
|
err = tx.QueryRow(insertQuery.String(), insertQuery.Args()...).Scan(&workspace.ID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to insert workspace: %w", err)
|
return fmt.Errorf("failed to insert workspace: %w", err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,19 +24,16 @@ func (db *database) CreateWorkspace(workspace *models.Workspace) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to encrypt token: %w", err)
|
return fmt.Errorf("failed to encrypt token: %w", err)
|
||||||
}
|
}
|
||||||
|
workspace.GitToken = encryptedToken
|
||||||
|
|
||||||
query := NewQuery(db.dbType).
|
query, err := NewQuery(db.dbType).
|
||||||
Insert("workspaces",
|
InsertStruct(workspace, "workspaces")
|
||||||
"user_id", "name", "theme", "auto_save", "show_hidden_files",
|
|
||||||
"git_enabled", "git_url", "git_user", "git_token",
|
if err != nil {
|
||||||
"git_auto_commit", "git_commit_msg_template",
|
return fmt.Errorf("failed to create query: %w", err)
|
||||||
"git_commit_name", "git_commit_email").
|
}
|
||||||
Values(13).
|
|
||||||
AddArgs(
|
query.Returning("id", "created_at")
|
||||||
workspace.UserID, workspace.Name, workspace.Theme, workspace.AutoSave, workspace.ShowHiddenFiles,
|
|
||||||
workspace.GitEnabled, workspace.GitURL, workspace.GitUser, encryptedToken,
|
|
||||||
workspace.GitAutoCommit, workspace.GitCommitMsgTemplate, workspace.GitCommitName, workspace.GitCommitEmail).
|
|
||||||
Returning("id", "created_at")
|
|
||||||
|
|
||||||
err = db.QueryRow(query.String(), query.Args()...).
|
err = db.QueryRow(query.String(), query.Args()...).
|
||||||
Scan(&workspace.ID, &workspace.CreatedAt)
|
Scan(&workspace.ID, &workspace.CreatedAt)
|
||||||
|
|||||||
@@ -5,9 +5,9 @@ import "time"
|
|||||||
|
|
||||||
// Session represents a user session in the database
|
// Session represents a user session in the database
|
||||||
type Session struct {
|
type Session struct {
|
||||||
ID string // Unique session identifier
|
ID string `db:"id,default"` // Unique session identifier
|
||||||
UserID int // ID of the user this session belongs to
|
UserID int `db:"user_id"` // ID of the user this session belongs to
|
||||||
RefreshToken string // The refresh token associated with this session
|
RefreshToken string `db:"refresh_token"` // The refresh token associated with this session
|
||||||
ExpiresAt time.Time // When this session expires
|
ExpiresAt time.Time `db:"expires_at"` // When this session expires
|
||||||
CreatedAt time.Time // When this session was created
|
CreatedAt time.Time `db:"created_at,default"` // When this session was created
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,13 +20,13 @@ const (
|
|||||||
|
|
||||||
// User represents a user in the system
|
// User represents a user in the system
|
||||||
type User struct {
|
type User struct {
|
||||||
ID int `json:"id" validate:"required,min=1"`
|
ID int `json:"id" db:"id,default" validate:"required,min=1"`
|
||||||
Email string `json:"email" validate:"required,email"`
|
Email string `json:"email" db:"email" validate:"required,email"`
|
||||||
DisplayName string `json:"displayName"`
|
DisplayName string `json:"displayName" db:"display_name"`
|
||||||
PasswordHash string `json:"-"`
|
PasswordHash string `json:"-" db:"password_hash"`
|
||||||
Role UserRole `json:"role" validate:"required,oneof=admin editor viewer"`
|
Role UserRole `json:"role" db:"role" validate:"required,oneof=admin editor viewer"`
|
||||||
CreatedAt time.Time `json:"createdAt"`
|
CreatedAt time.Time `json:"createdAt" db:"created_at,default"`
|
||||||
LastWorkspaceID int `json:"lastWorkspaceId"`
|
LastWorkspaceID int `json:"lastWorkspaceId" db:"last_workspace_id"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate validates the user struct
|
// Validate validates the user struct
|
||||||
|
|||||||
@@ -6,24 +6,24 @@ import (
|
|||||||
|
|
||||||
// Workspace represents a user's workspace in the system
|
// Workspace represents a user's workspace in the system
|
||||||
type Workspace struct {
|
type Workspace struct {
|
||||||
ID int `json:"id" validate:"required,min=1"`
|
ID int `json:"id" db:"id,default" validate:"required,min=1"`
|
||||||
UserID int `json:"userId" validate:"required,min=1"`
|
UserID int `json:"userId" db:"user_id" validate:"required,min=1"`
|
||||||
Name string `json:"name" validate:"required"`
|
Name string `json:"name" db:"name" validate:"required"`
|
||||||
CreatedAt time.Time `json:"createdAt"`
|
CreatedAt time.Time `json:"createdAt" db:"created_at,default"`
|
||||||
LastOpenedFilePath string `json:"lastOpenedFilePath"`
|
LastOpenedFilePath string `json:"lastOpenedFilePath" db:"last_opened_file_path"`
|
||||||
|
|
||||||
// Integrated settings
|
// Integrated settings
|
||||||
Theme string `json:"theme" validate:"oneof=light dark"`
|
Theme string `json:"theme" db:"theme" validate:"oneof=light dark"`
|
||||||
AutoSave bool `json:"autoSave"`
|
AutoSave bool `json:"autoSave" db:"auto_save"`
|
||||||
ShowHiddenFiles bool `json:"showHiddenFiles"`
|
ShowHiddenFiles bool `json:"showHiddenFiles" db:"show_hidden_files"`
|
||||||
GitEnabled bool `json:"gitEnabled"`
|
GitEnabled bool `json:"gitEnabled" db:"git_enabled"`
|
||||||
GitURL string `json:"gitUrl" validate:"required_if=GitEnabled true"`
|
GitURL string `json:"gitUrl" db:"git_url" validate:"required_if=GitEnabled true"`
|
||||||
GitUser string `json:"gitUser" validate:"required_if=GitEnabled true"`
|
GitUser string `json:"gitUser" db:"git_user" validate:"required_if=GitEnabled true"`
|
||||||
GitToken string `json:"gitToken" validate:"required_if=GitEnabled true"`
|
GitToken string `json:"gitToken" db:"git_token" validate:"required_if=GitEnabled true"`
|
||||||
GitAutoCommit bool `json:"gitAutoCommit"`
|
GitAutoCommit bool `json:"gitAutoCommit" db:"git_auto_commit"`
|
||||||
GitCommitMsgTemplate string `json:"gitCommitMsgTemplate"`
|
GitCommitMsgTemplate string `json:"gitCommitMsgTemplate" db:"git_commit_msg_template"`
|
||||||
GitCommitName string `json:"gitCommitName"`
|
GitCommitName string `json:"gitCommitName" db:"git_commit_name"`
|
||||||
GitCommitEmail string `json:"gitCommitEmail" validate:"omitempty,required_if=GitEnabled true,email"`
|
GitCommitEmail string `json:"gitCommitEmail" db:"git_commit_email" validate:"omitempty,required_if=GitEnabled true,email"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate validates the workspace struct
|
// Validate validates the workspace struct
|
||||||
|
|||||||
Reference in New Issue
Block a user