From 018dd93fcb5985a8d24b703087ac1c5dadda4acf Mon Sep 17 00:00:00 2001 From: Denis Gukov Date: Mon, 22 Jul 2024 01:21:54 +0500 Subject: [PATCH 1/3] fix(be): convert time to utc before db operations --- db/Task.go | 19 +++++++++++++++++++ db/sql/task.go | 7 ++++++- 2 files changed, 25 insertions(+), 1 deletion(-) diff --git a/db/Task.go b/db/Task.go index cb4a2a29..fd8ee4fc 100644 --- a/db/Task.go +++ b/db/Task.go @@ -2,6 +2,7 @@ package db import ( "fmt" + "github.com/go-gorp/gorp/v3" "time" "github.com/ansible-semaphore/semaphore/pkg/task_logger" @@ -51,6 +52,24 @@ type Task struct { InventoryID *int `db:"inventory_id" json:"inventory_id"` } +func (task *Task) PreInsert(gorp.SqlExecutor) error { + task.Created = task.Created.UTC() + return nil +} + +func (task *Task) PreUpdate(gorp.SqlExecutor) error { + if task.Start != nil { + start := task.Start.UTC() + task.Start = &start + } + + if task.End != nil { + end := task.End.UTC() + task.End = &end + } + return nil +} + func (task *Task) GetIncomingVersion(d Store) *string { if task.BuildTaskID == nil { return nil diff --git a/db/sql/task.go b/db/sql/task.go index c3f71b64..4c2d206b 100644 --- a/db/sql/task.go +++ b/db/sql/task.go @@ -92,7 +92,12 @@ func (d *SqlDb) CreateTask(task db.Task, maxTasks int) (newTask db.Task, err err } func (d *SqlDb) UpdateTask(task db.Task) error { - _, err := d.exec( + err := task.PreUpdate(d.sql) + if err != nil { + return err + } + + _, err = d.exec( "update task set status=?, start=?, `end`=? where id=?", task.Status, task.Start, From ffeb65c2ec39e1939bdd4839d481444b40c47e79 Mon Sep 17 00:00:00 2001 From: Denis Gukov Date: Mon, 22 Jul 2024 11:14:06 +0500 Subject: [PATCH 2/3] feat(sql): use UTC for all dates --- db/sql/event.go | 2 +- db/sql/project.go | 2 +- db/sql/session.go | 4 ++-- db/sql/task.go | 2 +- db/sql/user.go | 4 ++-- 5 files changed, 7 insertions(+), 7 deletions(-) diff --git a/db/sql/event.go b/db/sql/event.go index 6f209255..65887a88 100644 --- a/db/sql/event.go +++ b/db/sql/event.go @@ -30,7 +30,7 @@ func (d *SqlDb) getEvents(q squirrel.SelectBuilder, params db.RetrieveQueryParam } func (d *SqlDb) CreateEvent(evt db.Event) (newEvent db.Event, err error) { - var created = time.Now() + var created = time.Now().UTC() _, err = d.exec( "insert into event(user_id, project_id, object_id, object_type, description, created) values (?, ?, ?, ?, ?, ?)", diff --git a/db/sql/project.go b/db/sql/project.go index 655edd91..32067d54 100644 --- a/db/sql/project.go +++ b/db/sql/project.go @@ -7,7 +7,7 @@ import ( ) func (d *SqlDb) CreateProject(project db.Project) (newProject db.Project, err error) { - project.Created = time.Now() + project.Created = time.Now().UTC() insertId, err := d.insert( "id", diff --git a/db/sql/session.go b/db/sql/session.go index f8baf743..363ad91a 100644 --- a/db/sql/session.go +++ b/db/sql/session.go @@ -12,7 +12,7 @@ func (d *SqlDb) CreateSession(session db.Session) (db.Session, error) { } func (d *SqlDb) CreateAPIToken(token db.APIToken) (db.APIToken, error) { - token.Created = db.GetParsedTime(time.Now()) + token.Created = db.GetParsedTime(time.Now().UTC()) err := d.sql.Insert(&token) return token, err } @@ -56,7 +56,7 @@ func (d *SqlDb) ExpireSession(userID int, sessionID int) error { } func (d *SqlDb) TouchSession(userID int, sessionID int) error { - _, err := d.exec("update session set last_active=? where id=? and user_id=?", time.Now(), sessionID, userID) + _, err := d.exec("update session set last_active=? where id=? and user_id=?", time.Now().UTC(), sessionID, userID) return err } diff --git a/db/sql/task.go b/db/sql/task.go index 4c2d206b..986912d8 100644 --- a/db/sql/task.go +++ b/db/sql/task.go @@ -112,7 +112,7 @@ func (d *SqlDb) CreateTaskOutput(output db.TaskOutput) (db.TaskOutput, error) { "insert into task__output (task_id, task, output, time) VALUES (?, '', ?, ?)", output.TaskID, output.Output, - output.Time) + output.Time.UTC()) return output, err } diff --git a/db/sql/user.go b/db/sql/user.go index 0a325c8b..f250a54f 100644 --- a/db/sql/user.go +++ b/db/sql/user.go @@ -16,7 +16,7 @@ func (d *SqlDb) CreateUserWithoutPassword(user db.User) (newUser db.User, err er } user.Password = "" - user.Created = db.GetParsedTime(time.Now()) + user.Created = db.GetParsedTime(time.Now().UTC()) err = d.sql.Insert(&user) @@ -42,7 +42,7 @@ func (d *SqlDb) CreateUser(user db.UserWithPwd) (newUser db.User, err error) { } user.Password = string(pwdHash) - user.Created = db.GetParsedTime(time.Now()) + user.Created = db.GetParsedTime(time.Now().UTC()) err = d.sql.Insert(&user.User) From 08cf4dd73c01c17f8b0eec5f2cd5c6f30f013d81 Mon Sep 17 00:00:00 2001 From: Denis Gukov Date: Mon, 22 Jul 2024 13:26:10 +0500 Subject: [PATCH 3/3] feat: time display --- db/sql/event.go | 4 ++-- db/sql/task.go | 2 +- web/src/main.js | 13 ++++++++++++- 3 files changed, 15 insertions(+), 4 deletions(-) diff --git a/db/sql/event.go b/db/sql/event.go index 65887a88..2247a040 100644 --- a/db/sql/event.go +++ b/db/sql/event.go @@ -54,7 +54,7 @@ func (d *SqlDb) GetUserEvents(userID int, params db.RetrieveQueryParams) ([]db.E q := squirrel.Select("event.*, p.name as project_name"). From("event"). LeftJoin("project as p on event.project_id=p.id"). - OrderBy("created desc"). + OrderBy("id desc"). LeftJoin("project__user as pu on pu.project_id=p.id"). Where("p.id IS NULL or pu.user_id=?", userID) @@ -65,7 +65,7 @@ func (d *SqlDb) GetEvents(projectID int, params db.RetrieveQueryParams) ([]db.Ev q := squirrel.Select("event.*, p.name as project_name"). From("event"). LeftJoin("project as p on event.project_id=p.id"). - OrderBy("created desc"). + OrderBy("id desc"). Where("event.project_id=?", projectID) return d.getEvents(q, params) diff --git a/db/sql/task.go b/db/sql/task.go index 986912d8..d5adb10e 100644 --- a/db/sql/task.go +++ b/db/sql/task.go @@ -128,7 +128,7 @@ func (d *SqlDb) getTasks(projectID int, templateID *int, taskIDs []int, params d From("task"). Join("project__template as tpl on task.template_id=tpl.id"). LeftJoin("`user` on task.user_id=`user`.id"). - OrderBy("task.created desc, id desc") + OrderBy("id desc") if templateID == nil { q = q.Where("tpl.project_id=?", projectID) diff --git a/web/src/main.js b/web/src/main.js index fd96ddfc..bf184829 100644 --- a/web/src/main.js +++ b/web/src/main.js @@ -13,7 +13,18 @@ const convert = new Convert(); axios.defaults.baseURL = document.baseURI; Vue.config.productionTip = false; -Vue.filter('formatDate', (value) => (value ? moment(String(value)).fromNow() : '—')); +Vue.filter('formatDate', (value) => { + if (!value) { + return '—'; + } + const date = moment(value); + const now = moment(); + + if (now.isSame(date, 'day')) { + return `${date.fromNow()} (${date.format('LT')})`; // Display only time if today + } + return date.format('L LT'); // Display only date otherwise +}); Vue.filter('formatTime', (value) => (value ? moment(String(value)).format('LTS') : '—')); Vue.filter('formatLog', (value) => (value ? convert.toHtml(String(value)) : value));