2022-03-06 12:47:40 +07:00
|
|
|
package model
|
2021-11-23 16:45:38 +07:00
|
|
|
|
|
|
|
import (
|
2022-04-07 12:33:53 +07:00
|
|
|
"database/sql"
|
2021-11-23 16:45:38 +07:00
|
|
|
"database/sql/driver"
|
2022-03-03 18:20:43 +07:00
|
|
|
"encoding/base64"
|
2021-11-23 16:45:38 +07:00
|
|
|
"fmt"
|
|
|
|
"net"
|
2022-04-07 12:33:53 +07:00
|
|
|
|
2022-04-25 07:31:05 +07:00
|
|
|
"github.com/google/uuid"
|
|
|
|
|
2022-04-07 12:33:53 +07:00
|
|
|
"github.com/authelia/authelia/v4/internal/utils"
|
2021-11-23 16:45:38 +07:00
|
|
|
)
|
|
|
|
|
2022-04-25 07:31:05 +07:00
|
|
|
// NullUUID is a nullable uuid.UUID.
|
|
|
|
type NullUUID struct {
|
|
|
|
uuid.UUID
|
|
|
|
Valid bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// Value is the NullUUID implementation of the databases/sql driver.Valuer.
|
|
|
|
func (u NullUUID) Value() (value driver.Value, err error) {
|
|
|
|
if !u.Valid {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return u.UUID.Value()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan is the NullUUID implementation of the sql.Scanner.
|
|
|
|
func (u *NullUUID) Scan(src interface{}) (err error) {
|
|
|
|
if src == nil {
|
|
|
|
u.UUID, u.Valid = uuid.UUID{}, false
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return u.UUID.Scan(src)
|
|
|
|
}
|
|
|
|
|
2021-12-03 07:04:11 +07:00
|
|
|
// NewIP easily constructs a new IP.
|
|
|
|
func NewIP(value net.IP) (ip IP) {
|
|
|
|
return IP{IP: value}
|
2021-11-29 10:09:14 +07:00
|
|
|
}
|
|
|
|
|
2021-12-03 07:04:11 +07:00
|
|
|
// NewNullIP easily constructs a new NullIP.
|
|
|
|
func NewNullIP(value net.IP) (ip NullIP) {
|
|
|
|
return NullIP{IP: value}
|
2021-11-23 16:45:38 +07:00
|
|
|
}
|
|
|
|
|
2021-12-03 07:04:11 +07:00
|
|
|
// NewNullIPFromString easily constructs a new NullIP from a string.
|
|
|
|
func NewNullIPFromString(value string) (ip NullIP) {
|
|
|
|
if value == "" {
|
|
|
|
return ip
|
|
|
|
}
|
|
|
|
|
|
|
|
return NullIP{IP: net.ParseIP(value)}
|
|
|
|
}
|
|
|
|
|
2022-03-03 18:20:43 +07:00
|
|
|
// NewBase64 returns a new Base64.
|
|
|
|
func NewBase64(data []byte) Base64 {
|
|
|
|
return Base64{data: data}
|
|
|
|
}
|
|
|
|
|
2021-12-03 07:04:11 +07:00
|
|
|
// IP is a type specific for storage of a net.IP in the database which can't be NULL.
|
|
|
|
type IP struct {
|
|
|
|
IP net.IP
|
|
|
|
}
|
|
|
|
|
|
|
|
// Value is the IP implementation of the databases/sql driver.Valuer.
|
|
|
|
func (ip IP) Value() (value driver.Value, err error) {
|
|
|
|
if ip.IP == nil {
|
2022-03-03 18:20:43 +07:00
|
|
|
return nil, fmt.Errorf(errFmtValueNil, ip)
|
2021-12-03 07:04:11 +07:00
|
|
|
}
|
|
|
|
|
2022-03-03 18:20:43 +07:00
|
|
|
return ip.IP.String(), nil
|
2021-12-03 07:04:11 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Scan is the IP implementation of the sql.Scanner.
|
|
|
|
func (ip *IP) Scan(src interface{}) (err error) {
|
|
|
|
if src == nil {
|
2022-03-03 18:20:43 +07:00
|
|
|
return fmt.Errorf(errFmtScanNil, ip)
|
2021-12-03 07:04:11 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
var value string
|
|
|
|
|
|
|
|
switch v := src.(type) {
|
|
|
|
case string:
|
|
|
|
value = v
|
2021-12-04 11:34:20 +07:00
|
|
|
case []byte:
|
|
|
|
value = string(v)
|
2021-12-03 07:04:11 +07:00
|
|
|
default:
|
2022-03-03 18:20:43 +07:00
|
|
|
return fmt.Errorf(errFmtScanInvalidType, ip, src, src)
|
2021-12-03 07:04:11 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
ip.IP = net.ParseIP(value)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NullIP is a type specific for storage of a net.IP in the database which can also be NULL.
|
|
|
|
type NullIP struct {
|
|
|
|
IP net.IP
|
|
|
|
}
|
|
|
|
|
|
|
|
// Value is the NullIP implementation of the databases/sql driver.Valuer.
|
|
|
|
func (ip NullIP) Value() (value driver.Value, err error) {
|
2021-11-23 16:45:38 +07:00
|
|
|
if ip.IP == nil {
|
2022-03-03 18:20:43 +07:00
|
|
|
return nil, nil
|
2021-11-23 16:45:38 +07:00
|
|
|
}
|
|
|
|
|
2022-03-03 18:20:43 +07:00
|
|
|
return ip.IP.String(), nil
|
2021-11-23 16:45:38 +07:00
|
|
|
}
|
|
|
|
|
2021-12-03 07:04:11 +07:00
|
|
|
// Scan is the NullIP implementation of the sql.Scanner.
|
|
|
|
func (ip *NullIP) Scan(src interface{}) (err error) {
|
2021-11-23 16:45:38 +07:00
|
|
|
if src == nil {
|
|
|
|
ip.IP = nil
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var value string
|
|
|
|
|
|
|
|
switch v := src.(type) {
|
|
|
|
case string:
|
|
|
|
value = v
|
2021-12-04 11:34:20 +07:00
|
|
|
case []byte:
|
|
|
|
value = string(v)
|
2021-11-23 16:45:38 +07:00
|
|
|
default:
|
2022-03-03 18:20:43 +07:00
|
|
|
return fmt.Errorf(errFmtScanInvalidType, ip, src, src)
|
2021-11-23 16:45:38 +07:00
|
|
|
}
|
|
|
|
|
2021-12-03 07:04:11 +07:00
|
|
|
ip.IP = net.ParseIP(value)
|
2021-11-23 16:45:38 +07:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2021-12-01 19:11:29 +07:00
|
|
|
|
2022-03-03 18:20:43 +07:00
|
|
|
// Base64 saves bytes to the database as a base64 encoded string.
|
|
|
|
type Base64 struct {
|
|
|
|
data []byte
|
|
|
|
}
|
|
|
|
|
|
|
|
// String returns the Base64 string encoded as base64.
|
|
|
|
func (b Base64) String() string {
|
|
|
|
return base64.StdEncoding.EncodeToString(b.data)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bytes returns the Base64 string encoded as bytes.
|
|
|
|
func (b Base64) Bytes() []byte {
|
|
|
|
return b.data
|
|
|
|
}
|
|
|
|
|
|
|
|
// Value is the Base64 implementation of the databases/sql driver.Valuer.
|
|
|
|
func (b Base64) Value() (value driver.Value, err error) {
|
|
|
|
return b.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan is the Base64 implementation of the sql.Scanner.
|
|
|
|
func (b *Base64) Scan(src interface{}) (err error) {
|
|
|
|
if src == nil {
|
|
|
|
return fmt.Errorf(errFmtScanNil, b)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch v := src.(type) {
|
|
|
|
case string:
|
|
|
|
if b.data, err = base64.StdEncoding.DecodeString(v); err != nil {
|
|
|
|
return fmt.Errorf(errFmtScanInvalidTypeErr, b, src, src, err)
|
|
|
|
}
|
|
|
|
case []byte:
|
|
|
|
if b.data, err = base64.StdEncoding.DecodeString(string(v)); err != nil {
|
|
|
|
b.data = v
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return fmt.Errorf(errFmtScanInvalidType, b, src, src)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-12-01 19:11:29 +07:00
|
|
|
// StartupCheck represents a provider that has a startup check.
|
|
|
|
type StartupCheck interface {
|
|
|
|
StartupCheck() (err error)
|
|
|
|
}
|
2022-04-07 12:33:53 +07:00
|
|
|
|
|
|
|
// StringSlicePipeDelimited is a string slice that is stored in the database delimited by pipes.
|
|
|
|
type StringSlicePipeDelimited []string
|
|
|
|
|
|
|
|
// Scan is the StringSlicePipeDelimited implementation of the sql.Scanner.
|
|
|
|
func (s *StringSlicePipeDelimited) Scan(value interface{}) (err error) {
|
|
|
|
var nullStr sql.NullString
|
|
|
|
|
|
|
|
if err = nullStr.Scan(value); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if nullStr.Valid {
|
|
|
|
*s = utils.StringSplitDelimitedEscaped(nullStr.String, '|')
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Value is the StringSlicePipeDelimited implementation of the databases/sql driver.Valuer.
|
|
|
|
func (s StringSlicePipeDelimited) Value() (driver.Value, error) {
|
|
|
|
return utils.StringJoinDelimitedEscaped(s, '|'), nil
|
|
|
|
}
|