Compare commits
4 Commits
Author | SHA1 | Date |
---|---|---|
|
50c2c27060 | |
|
69be6e3bee | |
|
35e0da91ca | |
|
99ccac5948 |
|
@ -0,0 +1,33 @@
|
|||
package versioning
|
||||
|
||||
import "encoding/json"
|
||||
|
||||
func (v *Version) UnmarshalJSON(data []byte) error {
|
||||
// try parse from string
|
||||
if data[0] == '"' {
|
||||
tmp, err := ParseVersion(string(data[1 : len(data)-1]))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
*v = tmp
|
||||
return nil
|
||||
}
|
||||
|
||||
var code uint32
|
||||
if err := json.Unmarshal(data, &code); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
tmp, err := ParseVersionFromCode(code)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
*v = tmp
|
||||
return nil
|
||||
}
|
||||
|
||||
func (v Version) MarshalJSON() ([]byte, error) {
|
||||
return json.Marshal(v.String())
|
||||
}
|
|
@ -0,0 +1,105 @@
|
|||
package versioning_test
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
|
||||
"git.bit5.ru/backend/versioning"
|
||||
)
|
||||
|
||||
type structWithVersion struct {
|
||||
Version versioning.Version `json:"v"`
|
||||
}
|
||||
|
||||
func TestMarshalJSON(t *testing.T) {
|
||||
cases := []struct {
|
||||
version structWithVersion
|
||||
expectedJson string
|
||||
}{
|
||||
{
|
||||
version: structWithVersion{versioning.MustParseVersion("")},
|
||||
expectedJson: `{"v":"0.0.0"}`,
|
||||
},
|
||||
|
||||
{
|
||||
version: structWithVersion{versioning.MustParseVersion("10")},
|
||||
expectedJson: `{"v":"10.0.0"}`,
|
||||
},
|
||||
|
||||
{
|
||||
version: structWithVersion{versioning.MustParseVersion("11.2")},
|
||||
expectedJson: `{"v":"11.2.0"}`,
|
||||
},
|
||||
|
||||
{
|
||||
version: structWithVersion{versioning.MustParseVersion("11.2.33")},
|
||||
expectedJson: `{"v":"11.2.33"}`,
|
||||
},
|
||||
}
|
||||
|
||||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
actualJson, err := json.Marshal(c.version)
|
||||
require.NoError(t, err, "case #%d", caseNum)
|
||||
require.EqualValues(t, []byte(c.expectedJson), actualJson, "case #%d", caseNum)
|
||||
}
|
||||
}
|
||||
|
||||
func TestUnmarshalJSON(t *testing.T) {
|
||||
cases := []struct {
|
||||
versionJson string
|
||||
expectedVal structWithVersion
|
||||
}{
|
||||
{
|
||||
versionJson: `{"v":"0.0.0"}`,
|
||||
expectedVal: structWithVersion{versioning.MustParseVersion("")},
|
||||
},
|
||||
|
||||
{
|
||||
versionJson: `{"v":"0.0.1"}`,
|
||||
expectedVal: structWithVersion{versioning.MustParseVersion("0.0.1")},
|
||||
},
|
||||
|
||||
{
|
||||
versionJson: `{"v":"0.1"}`,
|
||||
expectedVal: structWithVersion{versioning.MustParseVersion("0.1.0")},
|
||||
},
|
||||
|
||||
{
|
||||
versionJson: `{"v":"1"}`,
|
||||
expectedVal: structWithVersion{versioning.MustParseVersion("1.0.0")},
|
||||
},
|
||||
|
||||
{
|
||||
versionJson: `{"v":0}`,
|
||||
expectedVal: structWithVersion{versioning.MustParseVersion("0.0.0")},
|
||||
},
|
||||
|
||||
{
|
||||
versionJson: `{"v":99}`,
|
||||
expectedVal: structWithVersion{versioning.MustParseVersion("0.0.99")},
|
||||
},
|
||||
|
||||
{
|
||||
versionJson: `{"v":88899}`,
|
||||
expectedVal: structWithVersion{versioning.MustParseVersion("0.888.99")},
|
||||
},
|
||||
|
||||
{
|
||||
versionJson: `{"v":7788899}`,
|
||||
expectedVal: structWithVersion{versioning.MustParseVersion("77.888.99")},
|
||||
},
|
||||
}
|
||||
|
||||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
var actualVal structWithVersion
|
||||
err := json.Unmarshal([]byte(c.versionJson), &actualVal)
|
||||
require.NoError(t, err, "case #%d", caseNum)
|
||||
require.EqualValues(t, c.expectedVal, actualVal, "case #%d", caseNum)
|
||||
}
|
||||
}
|
|
@ -0,0 +1,39 @@
|
|||
package versioning
|
||||
|
||||
import (
|
||||
"database/sql/driver"
|
||||
"strconv"
|
||||
|
||||
"github.com/pkg/errors"
|
||||
)
|
||||
|
||||
func (v *Version) Scan(value interface{}) error {
|
||||
var code uint32
|
||||
|
||||
switch val := value.(type) {
|
||||
case []byte:
|
||||
c, err := strconv.ParseUint(string(val), 10, 32)
|
||||
if err != nil {
|
||||
return errors.WithStack(err)
|
||||
}
|
||||
code = uint32(c)
|
||||
|
||||
case int64:
|
||||
code = uint32(val)
|
||||
|
||||
default:
|
||||
return errors.Errorf("incompatible version code type - %T", value)
|
||||
}
|
||||
|
||||
parsed, err := ParseVersionFromCode(code)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
*v = parsed
|
||||
return nil
|
||||
}
|
||||
|
||||
func (v Version) Value() (driver.Value, error) {
|
||||
return int64(v.Code()), nil
|
||||
}
|
|
@ -16,13 +16,6 @@ const (
|
|||
|
||||
var maxVersion = newVersion(MaxMajor, MaxMinor, MaxPatch)
|
||||
|
||||
type Layout uint8
|
||||
|
||||
const (
|
||||
Full Layout = iota
|
||||
WithoutPatch
|
||||
)
|
||||
|
||||
type Version struct {
|
||||
major uint16
|
||||
minor uint16
|
||||
|
@ -30,29 +23,47 @@ type Version struct {
|
|||
code uint32
|
||||
}
|
||||
|
||||
func (v Version) String() string {
|
||||
return v.Format(Full)
|
||||
func (v Version) Major() uint16 {
|
||||
return v.major
|
||||
}
|
||||
|
||||
func (v Version) Format(l Layout) string {
|
||||
switch l {
|
||||
default:
|
||||
return fmt.Sprintf("%d.%d.%d", v.major, v.minor, v.patch)
|
||||
func (v Version) Minor() uint16 {
|
||||
return v.minor
|
||||
}
|
||||
|
||||
case WithoutPatch:
|
||||
return fmt.Sprintf("%d.%d", v.major, v.minor)
|
||||
}
|
||||
func (v Version) Patch() uint8 {
|
||||
return v.patch
|
||||
}
|
||||
|
||||
func (v Version) Code() uint32 {
|
||||
return v.code
|
||||
}
|
||||
|
||||
func (a Version) Equal(b Version) bool {
|
||||
return a.Code() == b.Code()
|
||||
}
|
||||
|
||||
func (a Version) Less(b Version) bool {
|
||||
return a.Code() < b.Code()
|
||||
}
|
||||
|
||||
func Parse(str string) (Version, error) {
|
||||
func (a Version) Lte(b Version) bool {
|
||||
return a.Code() <= b.Code()
|
||||
}
|
||||
|
||||
func (a Version) Greater(b Version) bool {
|
||||
return a.Code() > b.Code()
|
||||
}
|
||||
|
||||
func (a Version) Gte(b Version) bool {
|
||||
return a.Code() >= b.Code()
|
||||
}
|
||||
|
||||
func (v Version) String() string {
|
||||
return fmt.Sprintf("%d.%d.%d", v.major, v.minor, v.patch)
|
||||
}
|
||||
|
||||
func ParseVersion(str string) (Version, error) {
|
||||
parts := strings.Split(str, ".")
|
||||
partsLen := len(parts)
|
||||
if partsLen > 3 {
|
||||
|
@ -87,7 +98,15 @@ func Parse(str string) (Version, error) {
|
|||
return newVersion(maj, min, patch), nil
|
||||
}
|
||||
|
||||
func ParseFromCode(code uint32) (Version, error) {
|
||||
func MustParseVersion(str string) Version {
|
||||
v, err := ParseVersion(str)
|
||||
if err != nil {
|
||||
return Version{}
|
||||
}
|
||||
return v
|
||||
}
|
||||
|
||||
func ParseVersionFromCode(code uint32) (Version, error) {
|
||||
patch := code % 100
|
||||
minor := code%100000 - patch
|
||||
major := code - minor - patch
|
||||
|
|
|
@ -8,24 +8,194 @@ import (
|
|||
"github.com/stretchr/testify/require"
|
||||
)
|
||||
|
||||
func TestParse(t *testing.T) {
|
||||
cases := []struct {
|
||||
versionStr string
|
||||
valid bool
|
||||
expectedStr string
|
||||
expectedCode uint32
|
||||
}{
|
||||
{versionStr: "0.0.0", valid: true, expectedStr: "0.0.0", expectedCode: 0},
|
||||
{versionStr: "0.0.1", valid: true, expectedStr: "0.0.1", expectedCode: 1},
|
||||
{versionStr: "0.1.0", valid: true, expectedStr: "0.1.0", expectedCode: 100},
|
||||
{versionStr: "0.1.1", valid: true, expectedStr: "0.1.1", expectedCode: 101},
|
||||
{versionStr: "1.0.0", valid: true, expectedStr: "1.0.0", expectedCode: 100000},
|
||||
{versionStr: "1.0.1", valid: true, expectedStr: "1.0.1", expectedCode: 100001},
|
||||
{versionStr: "1.1.0", valid: true, expectedStr: "1.1.0", expectedCode: 100100},
|
||||
{versionStr: "1.1.1", valid: true, expectedStr: "1.1.1", expectedCode: 100101},
|
||||
func TestVersion(t *testing.T) {
|
||||
t.Run("Equal", func(t *testing.T) {
|
||||
cases := []struct {
|
||||
a versioning.Version
|
||||
b versioning.Version
|
||||
expectedEqual bool
|
||||
}{
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedEqual: true,
|
||||
},
|
||||
|
||||
{versionStr: "12345.123.12", valid: true, expectedStr: "12345.123.12", expectedCode: 1234512312},
|
||||
{versionStr: "42948.999.99", valid: true, expectedStr: "42948.999.99", expectedCode: 4294899999},
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0.1"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedEqual: false,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("0.999.99"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedEqual: false,
|
||||
},
|
||||
}
|
||||
|
||||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
actualEqual := c.a.Equal(c.b)
|
||||
require.EqualValues(t, c.expectedEqual, actualEqual, "case#%d", caseNum)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("Less", func(t *testing.T) {
|
||||
cases := []struct {
|
||||
a versioning.Version
|
||||
b versioning.Version
|
||||
expectedLess bool
|
||||
}{
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedLess: false,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0.1"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedLess: false,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("0.999.99"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedLess: true,
|
||||
},
|
||||
}
|
||||
|
||||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
actualLess := c.a.Less(c.b)
|
||||
require.EqualValues(t, c.expectedLess, actualLess, "case#%d", caseNum)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("Lte", func(t *testing.T) {
|
||||
cases := []struct {
|
||||
a versioning.Version
|
||||
b versioning.Version
|
||||
expectedLte bool
|
||||
}{
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedLte: true,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0.1"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedLte: false,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("0.999.99"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedLte: true,
|
||||
},
|
||||
}
|
||||
|
||||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
actualLte := c.a.Lte(c.b)
|
||||
require.EqualValues(t, c.expectedLte, actualLte, "case#%d", caseNum)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("Greater", func(t *testing.T) {
|
||||
cases := []struct {
|
||||
a versioning.Version
|
||||
b versioning.Version
|
||||
expectedGreater bool
|
||||
}{
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedGreater: false,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0.1"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedGreater: true,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("0.999.99"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedGreater: false,
|
||||
},
|
||||
}
|
||||
|
||||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
actualGreater := c.a.Greater(c.b)
|
||||
require.EqualValues(t, c.expectedGreater, actualGreater, "case#%d", caseNum)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("Gte", func(t *testing.T) {
|
||||
cases := []struct {
|
||||
a versioning.Version
|
||||
b versioning.Version
|
||||
expectedGte bool
|
||||
}{
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedGte: true,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("1.0.1"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedGte: true,
|
||||
},
|
||||
|
||||
{
|
||||
a: versioning.MustParseVersion("0.999.99"),
|
||||
b: versioning.MustParseVersion("1.0.0"),
|
||||
expectedGte: false,
|
||||
},
|
||||
}
|
||||
|
||||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
actualGte := c.a.Gte(c.b)
|
||||
require.EqualValues(t, c.expectedGte, actualGte, "case#%d", caseNum)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func TestParseVersion(t *testing.T) {
|
||||
cases := []struct {
|
||||
versionStr string
|
||||
valid bool
|
||||
expectedMajor uint16
|
||||
expectedMinor uint16
|
||||
expectedPatch uint8
|
||||
expectedStr string
|
||||
expectedCode uint32
|
||||
}{
|
||||
{versionStr: "0.0.0", valid: true, expectedMajor: 0, expectedMinor: 0, expectedPatch: 0, expectedStr: "0.0.0", expectedCode: 0},
|
||||
{versionStr: "0.0.1", valid: true, expectedMajor: 0, expectedMinor: 0, expectedPatch: 1, expectedStr: "0.0.1", expectedCode: 1},
|
||||
{versionStr: "0.1.0", valid: true, expectedMajor: 0, expectedMinor: 1, expectedPatch: 0, expectedStr: "0.1.0", expectedCode: 100},
|
||||
{versionStr: "0.1.1", valid: true, expectedMajor: 0, expectedMinor: 1, expectedPatch: 1, expectedStr: "0.1.1", expectedCode: 101},
|
||||
{versionStr: "1.0.0", valid: true, expectedMajor: 1, expectedMinor: 0, expectedPatch: 0, expectedStr: "1.0.0", expectedCode: 100000},
|
||||
{versionStr: "1.0.1", valid: true, expectedMajor: 1, expectedMinor: 0, expectedPatch: 1, expectedStr: "1.0.1", expectedCode: 100001},
|
||||
{versionStr: "1.1.0", valid: true, expectedMajor: 1, expectedMinor: 1, expectedPatch: 0, expectedStr: "1.1.0", expectedCode: 100100},
|
||||
{versionStr: "1.1.1", valid: true, expectedMajor: 1, expectedMinor: 1, expectedPatch: 1, expectedStr: "1.1.1", expectedCode: 100101},
|
||||
|
||||
{versionStr: "12345.123.12", valid: true, expectedMajor: 12345, expectedMinor: 123, expectedPatch: 12, expectedStr: "12345.123.12", expectedCode: 1234512312},
|
||||
{versionStr: "42948.999.99", valid: true, expectedMajor: 42948, expectedMinor: 999, expectedPatch: 99, expectedStr: "42948.999.99", expectedCode: 4294899999},
|
||||
{versionStr: "42948.999.100", valid: false},
|
||||
{versionStr: "42948.1000.99", valid: false},
|
||||
{versionStr: "42948.1000.100", valid: false},
|
||||
|
@ -34,22 +204,22 @@ func TestParse(t *testing.T) {
|
|||
{versionStr: "42949.1000.99", valid: false},
|
||||
{versionStr: "42949.1000.100", valid: false},
|
||||
|
||||
{versionStr: "0.0", valid: true, expectedStr: "0.0.0", expectedCode: 0},
|
||||
{versionStr: "0.1", valid: true, expectedStr: "0.1.0", expectedCode: 100},
|
||||
{versionStr: "1.0", valid: true, expectedStr: "1.0.0", expectedCode: 100000},
|
||||
{versionStr: "1.1", valid: true, expectedStr: "1.1.0", expectedCode: 100100},
|
||||
{versionStr: "0.0", valid: true, expectedMajor: 0, expectedMinor: 0, expectedPatch: 0, expectedStr: "0.0.0", expectedCode: 0},
|
||||
{versionStr: "0.1", valid: true, expectedMajor: 0, expectedMinor: 1, expectedPatch: 0, expectedStr: "0.1.0", expectedCode: 100},
|
||||
{versionStr: "1.0", valid: true, expectedMajor: 1, expectedMinor: 0, expectedPatch: 0, expectedStr: "1.0.0", expectedCode: 100000},
|
||||
{versionStr: "1.1", valid: true, expectedMajor: 1, expectedMinor: 1, expectedPatch: 0, expectedStr: "1.1.0", expectedCode: 100100},
|
||||
|
||||
{versionStr: "12345.123", valid: true, expectedStr: "12345.123.0", expectedCode: 1234512300},
|
||||
{versionStr: "42948.999", valid: true, expectedStr: "42948.999.0", expectedCode: 4294899900},
|
||||
{versionStr: "12345.123", valid: true, expectedMajor: 12345, expectedMinor: 123, expectedPatch: 0, expectedStr: "12345.123.0", expectedCode: 1234512300},
|
||||
{versionStr: "42948.999", valid: true, expectedMajor: 42948, expectedMinor: 999, expectedPatch: 0, expectedStr: "42948.999.0", expectedCode: 4294899900},
|
||||
{versionStr: "42948.1000", valid: false},
|
||||
{versionStr: "42949.999", valid: false},
|
||||
{versionStr: "42949.1000", valid: false},
|
||||
|
||||
{versionStr: "0", valid: true, expectedStr: "0.0.0", expectedCode: 0},
|
||||
{versionStr: "1", valid: true, expectedStr: "1.0.0", expectedCode: 100000},
|
||||
{versionStr: "0", valid: true, expectedMajor: 0, expectedMinor: 0, expectedPatch: 0, expectedStr: "0.0.0", expectedCode: 0},
|
||||
{versionStr: "1", valid: true, expectedMajor: 1, expectedMinor: 0, expectedPatch: 0, expectedStr: "1.0.0", expectedCode: 100000},
|
||||
|
||||
{versionStr: "12345", valid: true, expectedStr: "12345.0.0", expectedCode: 1234500000},
|
||||
{versionStr: "42948", valid: true, expectedStr: "42948.0.0", expectedCode: 4294800000},
|
||||
{versionStr: "12345", valid: true, expectedMajor: 12345, expectedMinor: 0, expectedPatch: 0, expectedStr: "12345.0.0", expectedCode: 1234500000},
|
||||
{versionStr: "42948", valid: true, expectedMajor: 42948, expectedMinor: 0, expectedPatch: 0, expectedStr: "42948.0.0", expectedCode: 4294800000},
|
||||
{versionStr: "42949", valid: false},
|
||||
|
||||
{versionStr: "1 ", valid: false},
|
||||
|
@ -74,7 +244,7 @@ func TestParse(t *testing.T) {
|
|||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
v, err := versioning.Parse(c.versionStr)
|
||||
v, err := versioning.ParseVersion(c.versionStr)
|
||||
|
||||
if !c.valid {
|
||||
require.Error(t, err, "case#%d", caseNum)
|
||||
|
@ -82,12 +252,15 @@ func TestParse(t *testing.T) {
|
|||
}
|
||||
|
||||
require.NoError(t, err, "case#%d", caseNum)
|
||||
require.EqualValues(t, c.expectedMajor, v.Major(), "case#%d", caseNum)
|
||||
require.EqualValues(t, c.expectedMinor, v.Minor(), "case#%d", caseNum)
|
||||
require.EqualValues(t, c.expectedPatch, v.Patch(), "case#%d", caseNum)
|
||||
require.EqualValues(t, c.expectedStr, v.String(), "case#%d", caseNum)
|
||||
require.EqualValues(t, c.expectedCode, v.Code(), "case#%d", caseNum)
|
||||
}
|
||||
}
|
||||
|
||||
func TestParseFromCode(t *testing.T) {
|
||||
func TestParseVersionFromCode(t *testing.T) {
|
||||
cases := []struct {
|
||||
code uint32
|
||||
valid bool
|
||||
|
@ -129,7 +302,7 @@ func TestParseFromCode(t *testing.T) {
|
|||
for i, c := range cases {
|
||||
caseNum := i + 1
|
||||
|
||||
v, err := versioning.ParseFromCode(c.code)
|
||||
v, err := versioning.ParseVersionFromCode(c.code)
|
||||
|
||||
if !c.valid {
|
||||
require.Error(t, err, "case#%d", caseNum)
|
||||
|
|
Loading…
Reference in New Issue