parent
							
								
									df7fa4e995
								
							
						
					
					
						commit
						4c89a9c33c
					
				| 
						 | 
					@ -915,7 +915,10 @@ func Issues(opts *IssuesOptions) ([]*Issue, error) {
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if len(opts.Labels) > 0 && opts.Labels != "0" {
 | 
						if len(opts.Labels) > 0 && opts.Labels != "0" {
 | 
				
			||||||
		labelIDs := base.StringsToInt64s(strings.Split(opts.Labels, ","))
 | 
							labelIDs, err := base.StringsToInt64s(strings.Split(opts.Labels, ","))
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return nil, err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
		if len(labelIDs) > 0 {
 | 
							if len(labelIDs) > 0 {
 | 
				
			||||||
			sess.
 | 
								sess.
 | 
				
			||||||
				Join("INNER", "issue_label", "issue.id = issue_label.issue_id").
 | 
									Join("INNER", "issue_label", "issue.id = issue_label.issue_id").
 | 
				
			||||||
| 
						 | 
					@ -1171,10 +1174,11 @@ func GetIssueStats(opts *IssueStatsOptions) *IssueStats {
 | 
				
			||||||
			And("is_pull = ?", opts.IsPull)
 | 
								And("is_pull = ?", opts.IsPull)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if len(opts.Labels) > 0 && opts.Labels != "0" {
 | 
							if len(opts.Labels) > 0 && opts.Labels != "0" {
 | 
				
			||||||
			labelIDs := base.StringsToInt64s(strings.Split(opts.Labels, ","))
 | 
								labelIDs, err := base.StringsToInt64s(strings.Split(opts.Labels, ","))
 | 
				
			||||||
			if len(labelIDs) > 0 {
 | 
								if err != nil {
 | 
				
			||||||
				sess.
 | 
									log.Warn("Malformed Labels argument: %s", opts.Labels)
 | 
				
			||||||
					Join("INNER", "issue_label", "issue.id = issue_id").
 | 
								} else if len(labelIDs) > 0 {
 | 
				
			||||||
 | 
									sess.Join("INNER", "issue_label", "issue.id = issue_id").
 | 
				
			||||||
					In("label_id", labelIDs)
 | 
										In("label_id", labelIDs)
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -56,6 +56,9 @@ func DetectEncoding(content []byte) (string, error) {
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	result, err := chardet.NewTextDetector().DetectBest(content)
 | 
						result, err := chardet.NewTextDetector().DetectBest(content)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return "", err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	if result.Charset != "UTF-8" && len(setting.Repository.AnsiCharset) > 0 {
 | 
						if result.Charset != "UTF-8" && len(setting.Repository.AnsiCharset) > 0 {
 | 
				
			||||||
		log.Debug("Using default AnsiCharset: %s", setting.Repository.AnsiCharset)
 | 
							log.Debug("Using default AnsiCharset: %s", setting.Repository.AnsiCharset)
 | 
				
			||||||
		return setting.Repository.AnsiCharset, err
 | 
							return setting.Repository.AnsiCharset, err
 | 
				
			||||||
| 
						 | 
					@ -256,19 +259,25 @@ func computeTimeDiff(diff int64) (int64, string) {
 | 
				
			||||||
		diffStr = "1 year"
 | 
							diffStr = "1 year"
 | 
				
			||||||
	default:
 | 
						default:
 | 
				
			||||||
		diffStr = fmt.Sprintf("%d years", diff/Year)
 | 
							diffStr = fmt.Sprintf("%d years", diff/Year)
 | 
				
			||||||
		diff = 0
 | 
							diff -= (diff / Year) * Year
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	return diff, diffStr
 | 
						return diff, diffStr
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TimeSincePro calculates the time interval and generate full user-friendly string.
 | 
					// TimeSincePro calculates the time interval and generate full user-friendly string.
 | 
				
			||||||
func TimeSincePro(then time.Time) string {
 | 
					func TimeSincePro(then time.Time) string {
 | 
				
			||||||
	now := time.Now()
 | 
						return timeSincePro(then, time.Now())
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func timeSincePro(then, now time.Time) string {
 | 
				
			||||||
	diff := now.Unix() - then.Unix()
 | 
						diff := now.Unix() - then.Unix()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if then.After(now) {
 | 
						if then.After(now) {
 | 
				
			||||||
		return "future"
 | 
							return "future"
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						if diff == 0 {
 | 
				
			||||||
 | 
							return "now"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var timeStr, diffStr string
 | 
						var timeStr, diffStr string
 | 
				
			||||||
	for {
 | 
						for {
 | 
				
			||||||
| 
						 | 
					@ -282,9 +291,7 @@ func TimeSincePro(then time.Time) string {
 | 
				
			||||||
	return strings.TrimPrefix(timeStr, ", ")
 | 
						return strings.TrimPrefix(timeStr, ", ")
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func timeSince(then time.Time, lang string) string {
 | 
					func timeSince(then, now time.Time, lang string) string {
 | 
				
			||||||
	now := time.Now()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	lbl := i18n.Tr(lang, "tool.ago")
 | 
						lbl := i18n.Tr(lang, "tool.ago")
 | 
				
			||||||
	diff := now.Unix() - then.Unix()
 | 
						diff := now.Unix() - then.Unix()
 | 
				
			||||||
	if then.After(now) {
 | 
						if then.After(now) {
 | 
				
			||||||
| 
						 | 
					@ -295,7 +302,7 @@ func timeSince(then time.Time, lang string) string {
 | 
				
			||||||
	switch {
 | 
						switch {
 | 
				
			||||||
	case diff <= 0:
 | 
						case diff <= 0:
 | 
				
			||||||
		return i18n.Tr(lang, "tool.now")
 | 
							return i18n.Tr(lang, "tool.now")
 | 
				
			||||||
	case diff <= 2:
 | 
						case diff <= 1:
 | 
				
			||||||
		return i18n.Tr(lang, "tool.1s", lbl)
 | 
							return i18n.Tr(lang, "tool.1s", lbl)
 | 
				
			||||||
	case diff < 1*Minute:
 | 
						case diff < 1*Minute:
 | 
				
			||||||
		return i18n.Tr(lang, "tool.seconds", diff, lbl)
 | 
							return i18n.Tr(lang, "tool.seconds", diff, lbl)
 | 
				
			||||||
| 
						 | 
					@ -334,12 +341,18 @@ func timeSince(then time.Time, lang string) string {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// RawTimeSince retrieves i18n key of time since t
 | 
					// RawTimeSince retrieves i18n key of time since t
 | 
				
			||||||
func RawTimeSince(t time.Time, lang string) string {
 | 
					func RawTimeSince(t time.Time, lang string) string {
 | 
				
			||||||
	return timeSince(t, lang)
 | 
						return timeSince(t, time.Now(), lang)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TimeSince calculates the time interval and generate user-friendly string.
 | 
					// TimeSince calculates the time interval and generate user-friendly string.
 | 
				
			||||||
func TimeSince(t time.Time, lang string) template.HTML {
 | 
					func TimeSince(then time.Time, lang string) template.HTML {
 | 
				
			||||||
	return template.HTML(fmt.Sprintf(`<span class="time-since" title="%s">%s</span>`, t.Format(setting.TimeFormat), timeSince(t, lang)))
 | 
						return htmlTimeSince(then, time.Now(), lang)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func htmlTimeSince(then, now time.Time, lang string) template.HTML {
 | 
				
			||||||
 | 
						return template.HTML(fmt.Sprintf(`<span class="time-since" title="%s">%s</span>`,
 | 
				
			||||||
 | 
							then.Format(setting.TimeFormat),
 | 
				
			||||||
 | 
							timeSince(then, now, lang)))
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Storage space size types
 | 
					// Storage space size types
 | 
				
			||||||
| 
						 | 
					@ -424,10 +437,10 @@ func Subtract(left interface{}, right interface{}) interface{} {
 | 
				
			||||||
	case int64:
 | 
						case int64:
 | 
				
			||||||
		rright = right.(int64)
 | 
							rright = right.(int64)
 | 
				
			||||||
	case float32:
 | 
						case float32:
 | 
				
			||||||
		fright = float64(left.(float32))
 | 
							fright = float64(right.(float32))
 | 
				
			||||||
		isInt = false
 | 
							isInt = false
 | 
				
			||||||
	case float64:
 | 
						case float64:
 | 
				
			||||||
		fleft = left.(float64)
 | 
							fright = right.(float64)
 | 
				
			||||||
		isInt = false
 | 
							isInt = false
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -459,12 +472,16 @@ func TruncateString(str string, limit int) string {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// StringsToInt64s converts a slice of string to a slice of int64.
 | 
					// StringsToInt64s converts a slice of string to a slice of int64.
 | 
				
			||||||
func StringsToInt64s(strs []string) []int64 {
 | 
					func StringsToInt64s(strs []string) ([]int64, error) {
 | 
				
			||||||
	ints := make([]int64, len(strs))
 | 
						ints := make([]int64, len(strs))
 | 
				
			||||||
	for i := range strs {
 | 
						for i := range strs {
 | 
				
			||||||
		ints[i] = com.StrTo(strs[i]).MustInt64()
 | 
							n, err := com.StrTo(strs[i]).Int64()
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return ints, err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							ints[i] = n
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	return ints
 | 
						return ints, nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Int64sToStrings converts a slice of int64 to a slice of string.
 | 
					// Int64sToStrings converts a slice of int64 to a slice of string.
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -4,10 +4,40 @@ import (
 | 
				
			||||||
	"testing"
 | 
						"testing"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	"code.gitea.io/gitea/modules/setting"
 | 
						"code.gitea.io/gitea/modules/setting"
 | 
				
			||||||
 | 
						"github.com/Unknwon/i18n"
 | 
				
			||||||
 | 
						macaroni18n "github.com/go-macaron/i18n"
 | 
				
			||||||
	"github.com/stretchr/testify/assert"
 | 
						"github.com/stretchr/testify/assert"
 | 
				
			||||||
 | 
						"os"
 | 
				
			||||||
	"strk.kbt.io/projects/go/libravatar"
 | 
						"strk.kbt.io/projects/go/libravatar"
 | 
				
			||||||
 | 
						"time"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var BaseDate time.Time
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// time durations
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						DayDur   = 24 * time.Hour
 | 
				
			||||||
 | 
						WeekDur  = 7 * DayDur
 | 
				
			||||||
 | 
						MonthDur = 30 * DayDur
 | 
				
			||||||
 | 
						YearDur  = 12 * MonthDur
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestMain(m *testing.M) {
 | 
				
			||||||
 | 
						// setup
 | 
				
			||||||
 | 
						macaroni18n.I18n(macaroni18n.Options{
 | 
				
			||||||
 | 
							Directory:   "../../conf/locale/",
 | 
				
			||||||
 | 
							DefaultLang: "en-US",
 | 
				
			||||||
 | 
							Langs:       []string{"en-US"},
 | 
				
			||||||
 | 
							Names:       []string{"english"},
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
						BaseDate = time.Date(2000, time.January, 1, 0, 0, 0, 0, time.UTC)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// run the tests
 | 
				
			||||||
 | 
						retVal := m.Run()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						os.Exit(retVal)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestEncodeMD5(t *testing.T) {
 | 
					func TestEncodeMD5(t *testing.T) {
 | 
				
			||||||
	assert.Equal(t,
 | 
						assert.Equal(t,
 | 
				
			||||||
		"3858f62230ac3c915f300c664312c63f",
 | 
							"3858f62230ac3c915f300c664312c63f",
 | 
				
			||||||
| 
						 | 
					@ -26,7 +56,36 @@ func TestShortSha(t *testing.T) {
 | 
				
			||||||
	assert.Equal(t, "veryverylo", ShortSha("veryverylong"))
 | 
						assert.Equal(t, "veryverylo", ShortSha("veryverylong"))
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TODO: Test DetectEncoding()
 | 
					func TestDetectEncoding(t *testing.T) {
 | 
				
			||||||
 | 
						testSuccess := func(b []byte, expected string) {
 | 
				
			||||||
 | 
							encoding, err := DetectEncoding(b)
 | 
				
			||||||
 | 
							assert.NoError(t, err)
 | 
				
			||||||
 | 
							assert.Equal(t, expected, encoding)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						// utf-8
 | 
				
			||||||
 | 
						b := []byte("just some ascii")
 | 
				
			||||||
 | 
						testSuccess(b, "UTF-8")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// utf-8-sig: "hey" (with BOM)
 | 
				
			||||||
 | 
						b = []byte{0xef, 0xbb, 0xbf, 0x68, 0x65, 0x79}
 | 
				
			||||||
 | 
						testSuccess(b, "UTF-8")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// utf-16: "hey<accented G>"
 | 
				
			||||||
 | 
						b = []byte{0xff, 0xfe, 0x68, 0x00, 0x65, 0x00, 0x79, 0x00, 0xf4, 0x01}
 | 
				
			||||||
 | 
						testSuccess(b, "UTF-16LE")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// iso-8859-1: d<accented e>cor<newline>
 | 
				
			||||||
 | 
						b = []byte{0x44, 0xe9, 0x63, 0x6f, 0x72, 0x0a}
 | 
				
			||||||
 | 
						testSuccess(b, "ISO-8859-1")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						setting.Repository.AnsiCharset = "placeholder"
 | 
				
			||||||
 | 
						testSuccess(b, "placeholder")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// invalid bytes
 | 
				
			||||||
 | 
						b = []byte{0xfa}
 | 
				
			||||||
 | 
						_, err := DetectEncoding(b)
 | 
				
			||||||
 | 
						assert.Error(t, err)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestBasicAuthDecode(t *testing.T) {
 | 
					func TestBasicAuthDecode(t *testing.T) {
 | 
				
			||||||
	_, _, err := BasicAuthDecode("?")
 | 
						_, _, err := BasicAuthDecode("?")
 | 
				
			||||||
| 
						 | 
					@ -88,11 +147,112 @@ func TestAvatarLink(t *testing.T) {
 | 
				
			||||||
	)
 | 
						)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TODO: computeTimeDiff()
 | 
					func TestComputeTimeDiff(t *testing.T) {
 | 
				
			||||||
// TODO: TimeSincePro()
 | 
						// test that for each offset in offsets,
 | 
				
			||||||
// TODO: timeSince()
 | 
						// computeTimeDiff(base + offset) == (offset, str)
 | 
				
			||||||
// TODO: RawTimeSince()
 | 
						test := func(base int64, str string, offsets ...int64) {
 | 
				
			||||||
// TODO: TimeSince()
 | 
							for _, offset := range offsets {
 | 
				
			||||||
 | 
								diff, diffStr := computeTimeDiff(base + offset)
 | 
				
			||||||
 | 
								assert.Equal(t, offset, diff)
 | 
				
			||||||
 | 
								assert.Equal(t, str, diffStr)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						test(0, "now", 0)
 | 
				
			||||||
 | 
						test(1, "1 second", 0)
 | 
				
			||||||
 | 
						test(2, "2 seconds", 0)
 | 
				
			||||||
 | 
						test(Minute, "1 minute", 0, 1, 30, Minute-1)
 | 
				
			||||||
 | 
						test(2*Minute, "2 minutes", 0, Minute-1)
 | 
				
			||||||
 | 
						test(Hour, "1 hour", 0, 1, Hour-1)
 | 
				
			||||||
 | 
						test(5*Hour, "5 hours", 0, Hour-1)
 | 
				
			||||||
 | 
						test(Day, "1 day", 0, 1, Day-1)
 | 
				
			||||||
 | 
						test(5*Day, "5 days", 0, Day-1)
 | 
				
			||||||
 | 
						test(Week, "1 week", 0, 1, Week-1)
 | 
				
			||||||
 | 
						test(3*Week, "3 weeks", 0, 4*Day+25000)
 | 
				
			||||||
 | 
						test(Month, "1 month", 0, 1, Month-1)
 | 
				
			||||||
 | 
						test(10*Month, "10 months", 0, Month-1)
 | 
				
			||||||
 | 
						test(Year, "1 year", 0, Year-1)
 | 
				
			||||||
 | 
						test(3*Year, "3 years", 0, Year-1)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestTimeSince(t *testing.T) {
 | 
				
			||||||
 | 
						assert.Equal(t, "now", timeSince(BaseDate, BaseDate, "en"))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// test that each diff in `diffs` yields the expected string
 | 
				
			||||||
 | 
						test := func(expected string, diffs ...time.Duration) {
 | 
				
			||||||
 | 
							ago := i18n.Tr("en", "tool.ago")
 | 
				
			||||||
 | 
							fromNow := i18n.Tr("en", "tool.from_now")
 | 
				
			||||||
 | 
							for _, diff := range diffs {
 | 
				
			||||||
 | 
								actual := timeSince(BaseDate, BaseDate.Add(diff), "en")
 | 
				
			||||||
 | 
								assert.Equal(t, expected+" "+ago, actual)
 | 
				
			||||||
 | 
								actual = timeSince(BaseDate.Add(diff), BaseDate, "en")
 | 
				
			||||||
 | 
								assert.Equal(t, expected+" "+fromNow, actual)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						test("1 second", time.Second, time.Second+50*time.Millisecond)
 | 
				
			||||||
 | 
						test("2 seconds", 2*time.Second, 2*time.Second+50*time.Millisecond)
 | 
				
			||||||
 | 
						test("1 minute", time.Minute, time.Minute+30*time.Second)
 | 
				
			||||||
 | 
						test("2 minutes", 2*time.Minute, 2*time.Minute+30*time.Second)
 | 
				
			||||||
 | 
						test("1 hour", time.Hour, time.Hour+30*time.Minute)
 | 
				
			||||||
 | 
						test("2 hours", 2*time.Hour, 2*time.Hour+30*time.Minute)
 | 
				
			||||||
 | 
						test("1 day", DayDur, DayDur+12*time.Hour)
 | 
				
			||||||
 | 
						test("2 days", 2*DayDur, 2*DayDur+12*time.Hour)
 | 
				
			||||||
 | 
						test("1 week", WeekDur, WeekDur+3*DayDur)
 | 
				
			||||||
 | 
						test("2 weeks", 2*WeekDur, 2*WeekDur+3*DayDur)
 | 
				
			||||||
 | 
						test("1 month", MonthDur, MonthDur+15*DayDur)
 | 
				
			||||||
 | 
						test("2 months", 2*MonthDur, 2*MonthDur+15*DayDur)
 | 
				
			||||||
 | 
						test("1 year", YearDur, YearDur+6*MonthDur)
 | 
				
			||||||
 | 
						test("2 years", 2*YearDur, 2*YearDur+6*MonthDur)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestTimeSincePro(t *testing.T) {
 | 
				
			||||||
 | 
						assert.Equal(t, "now", timeSincePro(BaseDate, BaseDate))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// test that a difference of `diff` yields the expected string
 | 
				
			||||||
 | 
						test := func(expected string, diff time.Duration) {
 | 
				
			||||||
 | 
							actual := timeSincePro(BaseDate, BaseDate.Add(diff))
 | 
				
			||||||
 | 
							assert.Equal(t, expected, actual)
 | 
				
			||||||
 | 
							assert.Equal(t, "future", timeSincePro(BaseDate.Add(diff), BaseDate))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						test("1 second", time.Second)
 | 
				
			||||||
 | 
						test("2 seconds", 2*time.Second)
 | 
				
			||||||
 | 
						test("1 minute", time.Minute)
 | 
				
			||||||
 | 
						test("1 minute, 1 second", time.Minute+time.Second)
 | 
				
			||||||
 | 
						test("1 minute, 59 seconds", time.Minute+59*time.Second)
 | 
				
			||||||
 | 
						test("2 minutes", 2*time.Minute)
 | 
				
			||||||
 | 
						test("1 hour", time.Hour)
 | 
				
			||||||
 | 
						test("1 hour, 1 second", time.Hour+time.Second)
 | 
				
			||||||
 | 
						test("1 hour, 59 minutes, 59 seconds", time.Hour+59*time.Minute+59*time.Second)
 | 
				
			||||||
 | 
						test("2 hours", 2*time.Hour)
 | 
				
			||||||
 | 
						test("1 day", DayDur)
 | 
				
			||||||
 | 
						test("1 day, 23 hours, 59 minutes, 59 seconds",
 | 
				
			||||||
 | 
							DayDur+23*time.Hour+59*time.Minute+59*time.Second)
 | 
				
			||||||
 | 
						test("2 days", 2*DayDur)
 | 
				
			||||||
 | 
						test("1 week", WeekDur)
 | 
				
			||||||
 | 
						test("2 weeks", 2*WeekDur)
 | 
				
			||||||
 | 
						test("1 month", MonthDur)
 | 
				
			||||||
 | 
						test("3 months", 3*MonthDur)
 | 
				
			||||||
 | 
						test("1 year", YearDur)
 | 
				
			||||||
 | 
						test("2 years, 3 months, 1 week, 2 days, 4 hours, 12 minutes, 17 seconds",
 | 
				
			||||||
 | 
							2*YearDur+3*MonthDur+WeekDur+2*DayDur+4*time.Hour+
 | 
				
			||||||
 | 
								12*time.Minute+17*time.Second)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestHtmlTimeSince(t *testing.T) {
 | 
				
			||||||
 | 
						setting.TimeFormat = time.UnixDate
 | 
				
			||||||
 | 
						// test that `diff` yields a result containing `expected`
 | 
				
			||||||
 | 
						test := func(expected string, diff time.Duration) {
 | 
				
			||||||
 | 
							actual := htmlTimeSince(BaseDate, BaseDate.Add(diff), "en")
 | 
				
			||||||
 | 
							assert.Contains(t, actual, `title="Sat Jan  1 00:00:00 UTC 2000"`)
 | 
				
			||||||
 | 
							assert.Contains(t, actual, expected)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						test("1 second", time.Second)
 | 
				
			||||||
 | 
						test("3 minutes", 3*time.Minute+5*time.Second)
 | 
				
			||||||
 | 
						test("1 day", DayDur+18*time.Hour)
 | 
				
			||||||
 | 
						test("1 week", WeekDur+6*DayDur)
 | 
				
			||||||
 | 
						test("3 months", 3*MonthDur+3*WeekDur)
 | 
				
			||||||
 | 
						test("2 years", 2*YearDur)
 | 
				
			||||||
 | 
						test("3 years", 3*YearDur+11*MonthDur+4*WeekDur)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestFileSize(t *testing.T) {
 | 
					func TestFileSize(t *testing.T) {
 | 
				
			||||||
	var size int64
 | 
						var size int64
 | 
				
			||||||
| 
						 | 
					@ -108,11 +268,48 @@ func TestFileSize(t *testing.T) {
 | 
				
			||||||
	assert.Equal(t, "512TB", FileSize(size))
 | 
						assert.Equal(t, "512TB", FileSize(size))
 | 
				
			||||||
	size = size * 1024
 | 
						size = size * 1024
 | 
				
			||||||
	assert.Equal(t, "512PB", FileSize(size))
 | 
						assert.Equal(t, "512PB", FileSize(size))
 | 
				
			||||||
	//size = size * 1024 TODO: Fix bug for EB
 | 
						size = size * 4
 | 
				
			||||||
	//assert.Equal(t, "512EB", FileSize(size))
 | 
						assert.Equal(t, "2.0EB", FileSize(size))
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TODO: Subtract()
 | 
					func TestSubtract(t *testing.T) {
 | 
				
			||||||
 | 
						toFloat64 := func(n interface{}) float64 {
 | 
				
			||||||
 | 
							switch n.(type) {
 | 
				
			||||||
 | 
							case int:
 | 
				
			||||||
 | 
								return float64(n.(int))
 | 
				
			||||||
 | 
							case int8:
 | 
				
			||||||
 | 
								return float64(n.(int8))
 | 
				
			||||||
 | 
							case int16:
 | 
				
			||||||
 | 
								return float64(n.(int16))
 | 
				
			||||||
 | 
							case int32:
 | 
				
			||||||
 | 
								return float64(n.(int32))
 | 
				
			||||||
 | 
							case int64:
 | 
				
			||||||
 | 
								return float64(n.(int64))
 | 
				
			||||||
 | 
							case float32:
 | 
				
			||||||
 | 
								return float64(n.(float32))
 | 
				
			||||||
 | 
							case float64:
 | 
				
			||||||
 | 
								return n.(float64)
 | 
				
			||||||
 | 
							default:
 | 
				
			||||||
 | 
								return 0.0
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						values := []interface{}{
 | 
				
			||||||
 | 
							int(-3),
 | 
				
			||||||
 | 
							int8(14),
 | 
				
			||||||
 | 
							int16(81),
 | 
				
			||||||
 | 
							int32(-156),
 | 
				
			||||||
 | 
							int64(1528),
 | 
				
			||||||
 | 
							float32(3.5),
 | 
				
			||||||
 | 
							float64(-15.348),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						for _, left := range values {
 | 
				
			||||||
 | 
							for _, right := range values {
 | 
				
			||||||
 | 
								expected := toFloat64(left) - toFloat64(right)
 | 
				
			||||||
 | 
								sub := Subtract(left, right)
 | 
				
			||||||
 | 
								assert.InDelta(t, expected, sub, 1e-3)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestEllipsisString(t *testing.T) {
 | 
					func TestEllipsisString(t *testing.T) {
 | 
				
			||||||
	assert.Equal(t, "...", EllipsisString("foobar", 0))
 | 
						assert.Equal(t, "...", EllipsisString("foobar", 0))
 | 
				
			||||||
| 
						 | 
					@ -137,14 +334,18 @@ func TestTruncateString(t *testing.T) {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestStringsToInt64s(t *testing.T) {
 | 
					func TestStringsToInt64s(t *testing.T) {
 | 
				
			||||||
	assert.Equal(t, []int64{}, StringsToInt64s([]string{}))
 | 
						testSuccess := func(input []string, expected []int64) {
 | 
				
			||||||
	assert.Equal(t,
 | 
							result, err := StringsToInt64s(input)
 | 
				
			||||||
		[]int64{1, 4, 16, 64, 256},
 | 
							assert.NoError(t, err)
 | 
				
			||||||
		StringsToInt64s([]string{"1", "4", "16", "64", "256"}),
 | 
							assert.Equal(t, expected, result)
 | 
				
			||||||
	)
 | 
						}
 | 
				
			||||||
 | 
						testSuccess([]string{}, []int64{})
 | 
				
			||||||
 | 
						testSuccess([]string{"-1234"}, []int64{-1234})
 | 
				
			||||||
 | 
						testSuccess([]string{"1", "4", "16", "64", "256"},
 | 
				
			||||||
 | 
							[]int64{1, 4, 16, 64, 256})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// TODO: StringsToInt64s should return ([]int64, error)
 | 
						_, err := StringsToInt64s([]string{"-1", "a", "$"})
 | 
				
			||||||
	assert.Equal(t, []int64{-1, 0, 0}, StringsToInt64s([]string{"-1", "a", "$"}))
 | 
						assert.Error(t, err)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestInt64sToStrings(t *testing.T) {
 | 
					func TestInt64sToStrings(t *testing.T) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -374,7 +374,10 @@ func ValidateRepoMetas(ctx *context.Context, form auth.CreateIssueForm) ([]int64
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check labels.
 | 
						// Check labels.
 | 
				
			||||||
	labelIDs := base.StringsToInt64s(strings.Split(form.LabelIDs, ","))
 | 
						labelIDs, err := base.StringsToInt64s(strings.Split(form.LabelIDs, ","))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, 0, 0
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	labelIDMark := base.Int64sToMap(labelIDs)
 | 
						labelIDMark := base.Int64sToMap(labelIDs)
 | 
				
			||||||
	hasSelected := false
 | 
						hasSelected := false
 | 
				
			||||||
	for i := range labels {
 | 
						for i := range labels {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue