package storage import ( "context" "errors" "fmt" "github.com/jackc/pgx/v5" "log" "madsky.ru/go-tracker/internal/database" "madsky.ru/go-tracker/internal/model/project" "strings" ) type ProjectRepository interface { Find(ctx context.Context, userId uint32, isAdmin bool, filter *project.FilterDTO) ([]*project.Project, error) FindOne(ctx context.Context, projectId uint64, userId uint32, isAdmin bool) (*project.Project, error) Create(ctx context.Context, dto *project.CreateProjectDTO) (*project.Project, error) Update(ctx context.Context, id uint64, issue *project.UpdateProjectDTO) (*project.Project, error) Remove(ctx context.Context, id uint64) (uint64, error) } type ProjectStore struct { client database.Client } func (ps *ProjectStore) Find(ctx context.Context, userId uint32, isAdmin bool, filter *project.FilterDTO) ([]*project.Project, error) { query := `select p.id, p.name, p.description, p.key from projects p ` var args []interface{} if isAdmin == false { query += `left join user_to_project up on up.project_id = p.id where up.user_id = $1` args = append(args, userId) } rows, err := ps.client.Query(ctx, query, args...) if err != nil { log.Println("ProjectStore rows", err) return nil, err } defer rows.Close() // another variant //projects, err := pgx.CollectRows(rows, pgx.RowToStructByName[project.Project]) projects := make([]*project.Project, 0) for rows.Next() { var p project.Project err = rows.Scan(&p.ID, &p.Name, &p.Description, &p.Key) if err != nil { log.Println("scan", err) return nil, err } projects = append(projects, &p) } if err = rows.Err(); err != nil { log.Println("rows err", err) return nil, err } return projects, nil } func (ps *ProjectStore) FindOne(ctx context.Context, projectId uint64, userId uint32, isAdmin bool) (*project.Project, error) { args := []interface{}{projectId} query := `select p.id, p.name, p.description, p.key from projects p left join user_to_project up on up.project_id = p.id where p.id = $1 ` if isAdmin == false { query += "and up.user_id = $2" args = append(args, userId) } var p project.Project rows, _ := ps.client.Query(ctx, query, args...) defer rows.Close() p, err := pgx.CollectOneRow(rows, pgx.RowToStructByName[project.Project]) if err != nil { fmt.Println("CollectOneRow FindById project", err) return nil, err } return &p, nil } func (ps *ProjectStore) Create(ctx context.Context, dto *project.CreateProjectDTO) (*project.Project, error) { //q := "insert into projects (name, description, key) values ($1, $2, $3) returning id, name, description, key" query := `insert into projects (name, description, key) values (@projName, @projDescription, @projKey) returning id, name, description, key` key := dto.Key if dto.Key == "" { key = trimString(dto.Name) } args := pgx.NamedArgs{ "projName": dto.Name, "projDescription": dto.Description, "projKey": key, } //rows, _ := ps.client.Query(ctx, q, dto.Name, dto.Description, key) rows, _ := ps.client.Query(ctx, query, args) defer rows.Close() p, err := pgx.CollectOneRow(rows, pgx.RowToStructByName[project.Project]) if err != nil { fmt.Println("CollectOneRow Create Project", err) //return nil, fmt.Errorf("unable to insert row: %w", err) return nil, fmt.Errorf("KEY must be 4 characters long") } return &p, nil } func (ps *ProjectStore) Update(ctx context.Context, id uint64, dto *project.UpdateProjectDTO) (*project.Project, error) { q := `update projects set name = $1, description = $2 where id = $3 returning id, name, description, key` var p project.Project if err := ps.client.QueryRow(ctx, q, dto.Name, dto.Description, id).Scan(&p.ID, &p.Name, &p.Description, &p.Key); err != nil { fmt.Println(fmt.Sprintf("error %v", err)) return nil, err } return &p, nil } func (ps *ProjectStore) Remove(ctx context.Context, id uint64) (uint64, error) { q := `delete from projects where id=$1` tag, err := ps.client.Exec(ctx, q, id) if err != nil { log.Println("exec error", err) return 0, err } rowsAffected := tag.RowsAffected() if rowsAffected == 0 { return 0, errors.New("project not found") } return uint64(rowsAffected), nil } func trimString(str string) string { runeStr := []rune(str) var res string if len(runeStr) > 4 { res = string(runeStr[:3]) } else { res = string(runeStr) } return strings.ToUpper(res) }