mirror of
https://github.com/gitpod-io/gitpod.git
synced 2025-12-08 17:36:30 +00:00
141 lines
2.6 KiB
Go
141 lines
2.6 KiB
Go
// Copyright (c) 2022 Gitpod GmbH. All rights reserved.
|
|
// Licensed under the GNU Affero General Public License (AGPL).
|
|
// See License.AGPL.txt in the project root for license information.
|
|
|
|
package db
|
|
|
|
import (
|
|
"database/sql/driver"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/relvacode/iso8601"
|
|
"google.golang.org/protobuf/types/known/timestamppb"
|
|
)
|
|
|
|
func NewVarCharTime(t time.Time) VarcharTime {
|
|
return VarcharTime{
|
|
t: t.UTC(),
|
|
valid: true,
|
|
}
|
|
}
|
|
|
|
func NewVarCharTimeFromStr(s string) (VarcharTime, error) {
|
|
var vt VarcharTime
|
|
err := vt.Scan(s)
|
|
return vt, err
|
|
}
|
|
|
|
// VarcharTime exists for cases where records are inserted into the DB as VARCHAR but actually contain a timestamp which is time.RFC3339
|
|
type VarcharTime struct {
|
|
t time.Time
|
|
valid bool
|
|
}
|
|
|
|
// Scan implements the Scanner interface.
|
|
func (n *VarcharTime) Scan(value interface{}) error {
|
|
if value == nil {
|
|
n.valid = false
|
|
return nil
|
|
}
|
|
|
|
switch s := value.(type) {
|
|
case []uint8:
|
|
return n.parseString(string(s))
|
|
case string:
|
|
return n.parseString(s)
|
|
}
|
|
return fmt.Errorf("unknown scan value for VarcharTime with value: %v", value)
|
|
}
|
|
|
|
func (n *VarcharTime) parseString(s string) error {
|
|
// Null value - empty string mean value is not set
|
|
if len(s) == 0 {
|
|
n.valid = false
|
|
return nil
|
|
}
|
|
|
|
parsed, err := iso8601.ParseString(s)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to parse %v into ISO8601: %w", s, err)
|
|
}
|
|
|
|
if parsed.UTC().IsZero() {
|
|
n.t = time.Time{}.UTC()
|
|
n.valid = false
|
|
return nil
|
|
}
|
|
|
|
n.valid = true
|
|
n.t = parsed.UTC()
|
|
return nil
|
|
}
|
|
|
|
func (n VarcharTime) Time() time.Time {
|
|
return n.t
|
|
}
|
|
|
|
func (n VarcharTime) IsSet() bool {
|
|
return n.valid
|
|
}
|
|
|
|
// Value implements the driver Valuer interface.
|
|
func (n VarcharTime) Value() (driver.Value, error) {
|
|
if n.IsSet() {
|
|
return TimeToISO8601(n.t), nil
|
|
}
|
|
return "", nil
|
|
}
|
|
|
|
func (n VarcharTime) String() string {
|
|
if n.IsSet() {
|
|
return TimeToISO8601(n.t)
|
|
}
|
|
return ""
|
|
}
|
|
|
|
var null = "null"
|
|
|
|
func (n VarcharTime) MarshalJSON() ([]byte, error) {
|
|
if !n.IsSet() {
|
|
return []byte(null), nil
|
|
}
|
|
|
|
return n.Time().UTC().MarshalJSON()
|
|
}
|
|
|
|
func (n *VarcharTime) UnmarshalJSON(data []byte) error {
|
|
if string(data) == null {
|
|
n.valid = false
|
|
return nil
|
|
}
|
|
|
|
t := time.Time{}
|
|
if err := t.UnmarshalJSON(data); err != nil {
|
|
return fmt.Errorf("failed to unmarshal VarcharTime %s: %w", string(data), err)
|
|
}
|
|
|
|
if t.IsZero() {
|
|
return nil
|
|
}
|
|
|
|
n.valid = true
|
|
n.t = t.UTC()
|
|
|
|
return nil
|
|
}
|
|
|
|
const ISO8601Format = "2006-01-02T15:04:05.000Z"
|
|
|
|
func TimeToISO8601(t time.Time) string {
|
|
return t.UTC().Format(ISO8601Format)
|
|
}
|
|
|
|
func VarcharTimeToTimestamppb(t VarcharTime) *timestamppb.Timestamp {
|
|
if !t.IsSet() {
|
|
return nil
|
|
}
|
|
|
|
return timestamppb.New(t.Time())
|
|
}
|