Compare commits

..

No commits in common. "dev" and "v0.2.2" have entirely different histories.
dev ... v0.2.2

16 changed files with 134 additions and 220 deletions

View file

@ -1,9 +1,9 @@
package main package main
import ( import (
"cgnolink/pkg/cgnolink" "cgnolink"
"cgnolink/pkg/cgnolink/database" "cgnolink/database"
appserver "cgnolink/pkg/cgnolink/server" appserver "cgnolink/server"
"github.com/rs/zerolog" "github.com/rs/zerolog"
"github.com/rs/zerolog/log" "github.com/rs/zerolog/log"
"os" "os"

3
go.mod
View file

@ -13,9 +13,8 @@ require (
github.com/knadh/koanf v0.15.0 github.com/knadh/koanf v0.15.0
github.com/labstack/echo/v4 v4.2.1 github.com/labstack/echo/v4 v4.2.1
github.com/mitchellh/copystructure v1.1.1 // indirect github.com/mitchellh/copystructure v1.1.1 // indirect
github.com/patrickmn/go-cache v2.1.0+incompatible github.com/patrickmn/go-cache v2.1.0+incompatible // indirect
github.com/rs/zerolog v1.20.0 github.com/rs/zerolog v1.20.0
golang.org/x/crypto v0.0.0-20210220033148-5ea612d1eb83
golang.org/x/sys v0.0.0-20210309074719-68d13333faf2 // indirect golang.org/x/sys v0.0.0-20210309074719-68d13333faf2 // indirect
golang.org/x/text v0.3.5 // indirect golang.org/x/text v0.3.5 // indirect
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 // indirect golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 // indirect

View file

@ -9,7 +9,6 @@ type CreationModel struct {
Id string `json:"id"` Id string `json:"id"`
Name string `json:"name"` Name string `json:"name"`
RedirectURL string `json:"redirectUrl"` RedirectURL string `json:"redirectUrl"`
Password string `json:"password"`
} }
type ResourceModel struct { type ResourceModel struct {
@ -22,7 +21,6 @@ type ResourceModel struct {
type UpdateModel struct { type UpdateModel struct {
Name string `json:"name,omitempty"` Name string `json:"name,omitempty"`
RedirectURL string `json:"redirectUrl,omitempty"` RedirectURL string `json:"redirectUrl,omitempty"`
Password string `json:"password"`
} }
func (m *CreationModel) MapModelToEntity() (*Link, error) { func (m *CreationModel) MapModelToEntity() (*Link, error) {
@ -35,7 +33,6 @@ func (m *CreationModel) MapModelToEntity() (*Link, error) {
Id: m.Id, Id: m.Id,
Name: m.Name, Name: m.Name,
RedirectURL: *u, RedirectURL: *u,
Password: m.Password,
CreationTime: time.Now().UTC(), CreationTime: time.Now().UTC(),
}, nil }, nil
} }

View file

@ -9,7 +9,6 @@ type Link struct {
Id string Id string
Name string Name string
RedirectURL url.URL RedirectURL url.URL
Password string
CreationTime time.Time CreationTime time.Time
} }

View file

@ -11,14 +11,13 @@ import (
func redirectHandler(ctx echo.Context, serv Service) error { func redirectHandler(ctx echo.Context, serv Service) error {
linkId := ctx.Param("id") linkId := ctx.Param("id")
linkPassword := ctx.QueryParam("password")
redirectUrl, err := serv.AccessLink(linkId, linkPassword) link, err := serv.GetById(linkId)
if err != nil { if err != nil {
return err return err
} }
return ctx.Redirect(http.StatusSeeOther, redirectUrl.String()) return ctx.Redirect(http.StatusSeeOther, link.RedirectURL.String())
} }
func creationHandler(ctx echo.Context, serv Service) error { func creationHandler(ctx echo.Context, serv Service) error {
@ -54,7 +53,7 @@ func allRetrievalHandler(ctx echo.Context, serv Service) error {
limit := 20 limit := 20
if v := ctx.QueryParam("limit"); v != "" { if v := ctx.QueryParam("limit"); v != "" {
num, err := strconv.Atoi(v) num, err := strconv.Atoi(v)
if err != nil || num < 0 { if err != nil {
return echo.NewHTTPError(http.StatusBadRequest, "Invalid limit value.") return echo.NewHTTPError(http.StatusBadRequest, "Invalid limit value.")
} }
@ -64,7 +63,7 @@ func allRetrievalHandler(ctx echo.Context, serv Service) error {
offset := 0 offset := 0
if v := ctx.QueryParam("offset"); v != "" { if v := ctx.QueryParam("offset"); v != "" {
num, err := strconv.Atoi(v) num, err := strconv.Atoi(v)
if err != nil || num < 0 { if err != nil {
return echo.NewHTTPError(http.StatusBadRequest, "Invalid offset value.") return echo.NewHTTPError(http.StatusBadRequest, "Invalid offset value.")
} }
@ -92,18 +91,31 @@ func updateHandler(ctx echo.Context, serv Service) error {
return echo.NewHTTPError(http.StatusBadRequest, "Invalid data format.") return echo.NewHTTPError(http.StatusBadRequest, "Invalid data format.")
} }
var parsedUrl *url.URL updatingLink, err := serv.GetById(linkId)
parsedUrl, err := url.Parse(model.RedirectURL)
if err != nil { if err != nil {
return echo.NewHTTPError(http.StatusBadRequest, "Invalid URL value.") return err
} }
if err := serv.UpdateById(linkId, struct { hasChanges := false
Name string switch {
Password string case model.Name != "" && model.Name != updatingLink.Name:
RedirectURL *url.URL updatingLink.Name = model.Name
}{Name: model.Name, Password: model.Password, RedirectURL: parsedUrl}); err != nil {
return err hasChanges = true
case model.RedirectURL != "" && model.RedirectURL != updatingLink.RedirectURL.String():
if parsedUrl, err := url.Parse(model.RedirectURL); err != nil {
return echo.NewHTTPError(http.StatusBadRequest, "Invalid URL value.")
} else {
updatingLink.RedirectURL = *parsedUrl
}
hasChanges = true
}
if hasChanges {
if err = serv.Update(updatingLink); err != nil {
return err
}
} }
return ctx.NoContent(http.StatusOK) return ctx.NoContent(http.StatusOK)

View file

@ -1,7 +1,7 @@
package link package link
import ( import (
"cgnolink/pkg/cgnolink/database" "cgnolink/database"
"context" "context"
"errors" "errors"
"github.com/jackc/pgtype" "github.com/jackc/pgtype"
@ -34,12 +34,12 @@ func (r *PgRepository) Save(link *Link) error {
defer cancel() defer cancel()
sql := ` sql := `
INSERT INTO links (id, name, redirect_url, password, creation_time) INSERT INTO links (id, name, redirect_url, creation_time)
VALUES ($1, $2, $3, $4, $5::timestamp) VALUES ($1, $2, $3, $4::timestamp)
` `
database.LogPoolState(r.pool, "Saving link") database.LogPoolState(r.pool, "Saving link")
_, err := r.pool.Exec(ctx, sql, link.Id, link.Name, link.RedirectURL.String(), link.Password, link.CreationTime.Format("2006-01-02 15:04:05")) _, err := r.pool.Exec(ctx, sql, link.Id, link.Name, link.RedirectURL.String(), link.CreationTime.Format("2006-01-02 15:04:05"))
if err != nil { if err != nil {
return err return err
} }
@ -52,7 +52,7 @@ func (r *PgRepository) FindById(id string) (*Link, error) {
defer cancel() defer cancel()
sql := ` sql := `
SELECT id, name, redirect_url, password, creation_time SELECT id, name, redirect_url, creation_time
FROM links FROM links
WHERE id = $1 WHERE id = $1
` `
@ -82,7 +82,7 @@ func (r *PgRepository) GetAll(limit int, offset int) (Links, error) {
defer cancel() defer cancel()
sql := ` sql := `
SELECT id, name, redirect_url, password, creation_time SELECT id, name, redirect_url, creation_time
FROM links FROM links
LIMIT $1 LIMIT $1
OFFSET $2 OFFSET $2
@ -122,12 +122,12 @@ func (r *PgRepository) Update(link *Link) error {
sql := ` sql := `
UPDATE links UPDATE links
SET name = $1, redirect_url = $2, password = $3 SET name = $1, redirect_url = $2
WHERE id = $4 WHERE id = $3
` `
database.LogPoolState(r.pool, "Updating link") database.LogPoolState(r.pool, "Updating link")
_, err := r.pool.Exec(ctx, sql, link.Name, link.RedirectURL.String(), link.Password, link.Id) _, err := r.pool.Exec(ctx, sql, link.Name, link.RedirectURL.String(), link.Id)
if err != nil { if err != nil {
return err return err
} }
@ -159,11 +159,11 @@ func mapRowToEntity(r interface{}) (*Link, error) {
switch v := r.(type) { switch v := r.(type) {
case pgx.Row: case pgx.Row:
if err := v.Scan(&entity.Id, &entity.Name, &urlStr, &entity.Password, &t); err != nil { if err := v.Scan(&entity.Id, &entity.Name, &urlStr, &t); err != nil {
return nil, err return nil, err
} }
case pgx.Rows: case pgx.Rows:
if err := v.Scan(&entity.Id, &entity.Name, &urlStr, &entity.Password, &t); err != nil { if err := v.Scan(&entity.Id, &entity.Name, &urlStr, &t); err != nil {
return nil, err return nil, err
} }
default: default:

91
link/service.go Normal file
View file

@ -0,0 +1,91 @@
package link
import (
apperrors "cgnolink/errors"
"github.com/patrickmn/go-cache"
)
type Service interface {
Create(link *Link) error
GetById(id string) (*Link, error)
GetAll(limit int, offset int) (Links, error)
Update(data *Link) error
DeleteById(id string) error
}
type PgService struct {
rep Repository
cache *cache.Cache
}
func NewService(rep Repository) Service {
return &PgService{
rep: rep,
cache: cache.New(cache.NoExpiration, 0),
}
}
func (s *PgService) Create(link *Link) error {
existingLink, err := s.rep.FindById(link.Id)
if err != nil {
return apperrors.UnknownError{Err: err}
}
if existingLink != nil {
return apperrors.AlreadyExistsError{Message: "Link with given ID already exists."}
}
if err = s.rep.Save(link); err != nil {
return apperrors.UnknownError{Err: err}
}
return nil
}
func (s *PgService) GetById(id string) (*Link, error) {
if v, found := s.cache.Get(id); found {
if link, ok := v.(*Link); ok {
return link, nil
}
}
link, err := s.rep.FindById(id)
if err != nil {
return nil, apperrors.UnknownError{Err: err}
}
if link == nil {
return nil, apperrors.NotFoundError{Message: "Link with given ID was not found."}
}
s.cache.Set(id, link, cache.DefaultExpiration)
return link, nil
}
func (s *PgService) GetAll(limit int, offset int) (Links, error) {
links, err := s.rep.GetAll(limit, offset)
if err != nil {
return nil, apperrors.UnknownError{Err: err}
}
return links, nil
}
func (s *PgService) Update(data *Link) error {
if err := s.rep.Update(data); err != nil {
return apperrors.UnknownError{Err: err}
}
return nil
}
func (s *PgService) DeleteById(id string) error {
s.cache.Delete(id)
if err := s.rep.DeleteById(id); err != nil {
return apperrors.UnknownError{Err: err}
}
return nil
}

View file

@ -1,13 +0,0 @@
ALTER TABLE links
ADD COLUMN password varchar;
UPDATE links
SET password = '';
ALTER TABLE links
ALTER COLUMN password SET not null;
---- create above / drop below ----
ALTER TABLE links
DROP COLUMN password;

View file

@ -1,171 +0,0 @@
package link
import (
apperrors "cgnolink/pkg/cgnolink/errors"
"github.com/patrickmn/go-cache"
"golang.org/x/crypto/bcrypt"
"net/url"
)
var linkNotFoundError = apperrors.NotFoundError{Message: "Link with given ID was not found."}
type Service interface {
Create(link *Link) error
GetById(id string) (*Link, error)
AccessLink(id string, password string) (*url.URL, error)
GetAll(limit int, offset int) (Links, error)
UpdateById(
id string,
data struct {
Name string
Password string
RedirectURL *url.URL
},
) error
DeleteById(id string) error
}
type PgService struct {
rep Repository
cache *cache.Cache
}
func (service *PgService) AccessLink(id string, password string) (*url.URL, error) {
link, err := service.GetById(id)
if err != nil {
return nil, err
}
if link.Password != "" {
if password == "" || bcrypt.CompareHashAndPassword([]byte(link.Password), []byte(password)) != nil {
return nil, linkNotFoundError
}
}
return &link.RedirectURL, nil
}
func NewService(rep Repository) Service {
return &PgService{
rep: rep,
cache: cache.New(cache.NoExpiration, 0),
}
}
func (service *PgService) Create(link *Link) error {
existingLink, err := service.rep.FindById(link.Id)
if err != nil {
return apperrors.UnknownError{Err: err}
}
if existingLink != nil {
return apperrors.AlreadyExistsError{Message: "Link with given ID already exists."}
}
if link.Password != "" {
hashedPassword, err := HashPassword(link.Password)
if err != nil {
return err
}
link.Password = hashedPassword
}
if err = service.rep.Save(link); err != nil {
return apperrors.UnknownError{Err: err}
}
return nil
}
func (service *PgService) GetById(id string) (*Link, error) {
if v, found := service.cache.Get(id); found {
if link, ok := v.(*Link); ok {
return link, nil
}
}
link, err := service.rep.FindById(id)
if err != nil {
return nil, apperrors.UnknownError{Err: err}
}
if link == nil {
return nil, linkNotFoundError
}
service.cache.Set(id, link, cache.DefaultExpiration)
return link, nil
}
func (service *PgService) GetAll(limit int, offset int) (Links, error) {
links, err := service.rep.GetAll(limit, offset)
if err != nil {
return nil, apperrors.UnknownError{Err: err}
}
return links, nil
}
func (service *PgService) UpdateById(
id string,
data struct {
Name string
Password string
RedirectURL *url.URL
},
) error {
link, err := service.GetById(id)
if err != nil {
return err
}
hasChanges := false
switch {
case data.Name != "":
link.Name = data.Name
hasChanges = true
case data.RedirectURL != nil:
link.RedirectURL = *data.RedirectURL
hasChanges = true
case data.Password != "":
hashedPw, err := HashPassword(data.Password)
if err != nil {
return err
}
link.Password = hashedPw
hasChanges = true
}
if hasChanges {
if err := service.rep.Update(link); err != nil {
return apperrors.UnknownError{Err: err}
}
service.cache.Delete(link.Id)
}
return nil
}
func (service *PgService) DeleteById(id string) error {
if err := service.rep.DeleteById(id); err != nil {
return apperrors.UnknownError{Err: err}
}
service.cache.Delete(id)
return nil
}
func HashPassword(password string) (string, error) {
hashedPw, err := bcrypt.GenerateFromPassword([]byte(password), bcrypt.DefaultCost)
if err != nil {
return "", apperrors.UnknownError{Err: err}
}
return string(hashedPw), nil
}

View file

@ -1,8 +1,8 @@
package server package server
import ( import (
apperrors "cgnolink/pkg/cgnolink/errors" apperrors "cgnolink/errors"
"cgnolink/pkg/cgnolink/link" "cgnolink/link"
"github.com/jackc/pgx/v4/pgxpool" "github.com/jackc/pgx/v4/pgxpool"
"github.com/knadh/koanf" "github.com/knadh/koanf"
"github.com/labstack/echo/v4" "github.com/labstack/echo/v4"