| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503 | // Copyright 2014 The Gogs Authors. All rights reserved.// Use of this source code is governed by a MIT-style// license that can be found in the LICENSE file.package baseimport (	"crypto/hmac"	"crypto/md5"	"crypto/rand"	"crypto/sha1"	"encoding/base64"	"encoding/hex"	"errors"	"fmt"	"hash"	"html/template"	"math"	"regexp"	"strings"	"time"	"github.com/Unknwon/com"	"github.com/Unknwon/i18n"	"github.com/gogits/gogs/modules/avatar"	"github.com/gogits/gogs/modules/setting")// Encode string to md5 hex value.func EncodeMd5(str string) string {	m := md5.New()	m.Write([]byte(str))	return hex.EncodeToString(m.Sum(nil))}// Encode string to sha1 hex value.func EncodeSha1(str string) string {	h := sha1.New()	h.Write([]byte(str))	return hex.EncodeToString(h.Sum(nil))}func BasicAuthDecode(encoded string) (user string, name string, err error) {	var s []byte	s, err = base64.StdEncoding.DecodeString(encoded)	if err != nil {		return user, name, err	}	auth := strings.SplitN(string(s), ":", 2)	return auth[0], auth[1], err}func BasicAuthEncode(username, password string) string {	return base64.StdEncoding.EncodeToString([]byte(username + ":" + password))}// GetRandomString generate random string by specify chars.func GetRandomString(n int, alphabets ...byte) string {	const alphanum = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"	var bytes = make([]byte, n)	rand.Read(bytes)	for i, b := range bytes {		if len(alphabets) == 0 {			bytes[i] = alphanum[b%byte(len(alphanum))]		} else {			bytes[i] = alphabets[b%byte(len(alphabets))]		}	}	return string(bytes)}// http://code.google.com/p/go/source/browse/pbkdf2/pbkdf2.go?repo=cryptofunc PBKDF2(password, salt []byte, iter, keyLen int, h func() hash.Hash) []byte {	prf := hmac.New(h, password)	hashLen := prf.Size()	numBlocks := (keyLen + hashLen - 1) / hashLen	var buf [4]byte	dk := make([]byte, 0, numBlocks*hashLen)	U := make([]byte, hashLen)	for block := 1; block <= numBlocks; block++ {		// N.B.: || means concatenation, ^ means XOR		// for each block T_i = U_1 ^ U_2 ^ ... ^ U_iter		// U_1 = PRF(password, salt || uint(i))		prf.Reset()		prf.Write(salt)		buf[0] = byte(block >> 24)		buf[1] = byte(block >> 16)		buf[2] = byte(block >> 8)		buf[3] = byte(block)		prf.Write(buf[:4])		dk = prf.Sum(dk)		T := dk[len(dk)-hashLen:]		copy(U, T)		// U_n = PRF(password, U_(n-1))		for n := 2; n <= iter; n++ {			prf.Reset()			prf.Write(U)			U = U[:0]			U = prf.Sum(U)			for x := range U {				T[x] ^= U[x]			}		}	}	return dk[:keyLen]}// verify time limit codefunc VerifyTimeLimitCode(data string, minutes int, code string) bool {	if len(code) <= 18 {		return false	}	// split code	start := code[:12]	lives := code[12:18]	if d, err := com.StrTo(lives).Int(); err == nil {		minutes = d	}	// right active code	retCode := CreateTimeLimitCode(data, minutes, start)	if retCode == code && minutes > 0 {		// check time is expired or not		before, _ := DateParse(start, "YmdHi")		now := time.Now()		if before.Add(time.Minute*time.Duration(minutes)).Unix() > now.Unix() {			return true		}	}	return false}const TimeLimitCodeLength = 12 + 6 + 40// create a time limit code// code format: 12 length date time string + 6 minutes string + 40 sha1 encoded stringfunc CreateTimeLimitCode(data string, minutes int, startInf interface{}) string {	format := "YmdHi"	var start, end time.Time	var startStr, endStr string	if startInf == nil {		// Use now time create code		start = time.Now()		startStr = DateFormat(start, format)	} else {		// use start string create code		startStr = startInf.(string)		start, _ = DateParse(startStr, format)		startStr = DateFormat(start, format)	}	end = start.Add(time.Minute * time.Duration(minutes))	endStr = DateFormat(end, format)	// create sha1 encode string	sh := sha1.New()	sh.Write([]byte(data + setting.SecretKey + startStr + endStr + com.ToStr(minutes)))	encoded := hex.EncodeToString(sh.Sum(nil))	code := fmt.Sprintf("%s%06d%s", startStr, minutes, encoded)	return code}// AvatarLink returns avatar link by given e-mail.func AvatarLink(email string) string {	if setting.DisableGravatar {		return setting.AppSubUrl + "/img/avatar_default.jpg"	}	gravatarHash := avatar.HashEmail(email)	if setting.Service.EnableCacheAvatar {		return setting.AppSubUrl + "/avatar/" + gravatarHash	}	return setting.GravatarSource + gravatarHash}// Seconds-based time unitsconst (	Minute = 60	Hour   = 60 * Minute	Day    = 24 * Hour	Week   = 7 * Day	Month  = 30 * Day	Year   = 12 * Month)func computeTimeDiff(diff int64) (int64, string) {	diffStr := ""	switch {	case diff <= 0:		diff = 0		diffStr = "now"	case diff < 2:		diff = 0		diffStr = "1 second"	case diff < 1*Minute:		diffStr = fmt.Sprintf("%d seconds", diff)		diff = 0	case diff < 2*Minute:		diff -= 1 * Minute		diffStr = "1 minute"	case diff < 1*Hour:		diffStr = fmt.Sprintf("%d minutes", diff/Minute)		diff -= diff / Minute * Minute	case diff < 2*Hour:		diff -= 1 * Hour		diffStr = "1 hour"	case diff < 1*Day:		diffStr = fmt.Sprintf("%d hours", diff/Hour)		diff -= diff / Hour * Hour	case diff < 2*Day:		diff -= 1 * Day		diffStr = "1 day"	case diff < 1*Week:		diffStr = fmt.Sprintf("%d days", diff/Day)		diff -= diff / Day * Day	case diff < 2*Week:		diff -= 1 * Week		diffStr = "1 week"	case diff < 1*Month:		diffStr = fmt.Sprintf("%d weeks", diff/Week)		diff -= diff / Week * Week	case diff < 2*Month:		diff -= 1 * Month		diffStr = "1 month"	case diff < 1*Year:		diffStr = fmt.Sprintf("%d months", diff/Month)		diff -= diff / Month * Month	case diff < 2*Year:		diff -= 1 * Year		diffStr = "1 year"	default:		diffStr = fmt.Sprintf("%d years", diff/Year)		diff = 0	}	return diff, diffStr}// TimeSincePro calculates the time interval and generate full user-friendly string.func TimeSincePro(then time.Time) string {	now := time.Now()	diff := now.Unix() - then.Unix()	if then.After(now) {		return "future"	}	var timeStr, diffStr string	for {		if diff == 0 {			break		}		diff, diffStr = computeTimeDiff(diff)		timeStr += ", " + diffStr	}	return strings.TrimPrefix(timeStr, ", ")}func timeSince(then time.Time, lang string) string {	now := time.Now()	lbl := i18n.Tr(lang, "tool.ago")	diff := now.Unix() - then.Unix()	if then.After(now) {		lbl = i18n.Tr(lang, "tool.from_now")		diff = then.Unix() - now.Unix()	}	switch {	case diff <= 0:		return i18n.Tr(lang, "tool.now")	case diff <= 2:		return i18n.Tr(lang, "tool.1s", lbl)	case diff < 1*Minute:		return i18n.Tr(lang, "tool.seconds", diff, lbl)	case diff < 2*Minute:		return i18n.Tr(lang, "tool.1m", lbl)	case diff < 1*Hour:		return i18n.Tr(lang, "tool.minutes", diff/Minute, lbl)	case diff < 2*Hour:		return i18n.Tr(lang, "tool.1h", lbl)	case diff < 1*Day:		return i18n.Tr(lang, "tool.hours", diff/Hour, lbl)	case diff < 2*Day:		return i18n.Tr(lang, "tool.1d", lbl)	case diff < 1*Week:		return i18n.Tr(lang, "tool.days", diff/Day, lbl)	case diff < 2*Week:		return i18n.Tr(lang, "tool.1w", lbl)	case diff < 1*Month:		return i18n.Tr(lang, "tool.weeks", diff/Week, lbl)	case diff < 2*Month:		return i18n.Tr(lang, "tool.1mon", lbl)	case diff < 1*Year:		return i18n.Tr(lang, "tool.months", diff/Month, lbl)	case diff < 2*Year:		return i18n.Tr(lang, "tool.1y", lbl)	default:		return i18n.Tr(lang, "tool.years", diff/Year, lbl)	}}// TimeSince calculates the time interval and generate user-friendly string.func TimeSince(t 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)))}const (	Byte  = 1	KByte = Byte * 1024	MByte = KByte * 1024	GByte = MByte * 1024	TByte = GByte * 1024	PByte = TByte * 1024	EByte = PByte * 1024)var bytesSizeTable = map[string]uint64{	"b":  Byte,	"kb": KByte,	"mb": MByte,	"gb": GByte,	"tb": TByte,	"pb": PByte,	"eb": EByte,}func logn(n, b float64) float64 {	return math.Log(n) / math.Log(b)}func humanateBytes(s uint64, base float64, sizes []string) string {	if s < 10 {		return fmt.Sprintf("%dB", s)	}	e := math.Floor(logn(float64(s), base))	suffix := sizes[int(e)]	val := float64(s) / math.Pow(base, math.Floor(e))	f := "%.0f"	if val < 10 {		f = "%.1f"	}	return fmt.Sprintf(f+"%s", val, suffix)}// FileSize calculates the file size and generate user-friendly string.func FileSize(s int64) string {	sizes := []string{"B", "KB", "MB", "GB", "TB", "PB", "EB"}	return humanateBytes(uint64(s), 1024, sizes)}// Subtract deals with subtraction of all types of number.func Subtract(left interface{}, right interface{}) interface{} {	var rleft, rright int64	var fleft, fright float64	var isInt bool = true	switch left.(type) {	case int:		rleft = int64(left.(int))	case int8:		rleft = int64(left.(int8))	case int16:		rleft = int64(left.(int16))	case int32:		rleft = int64(left.(int32))	case int64:		rleft = left.(int64)	case float32:		fleft = float64(left.(float32))		isInt = false	case float64:		fleft = left.(float64)		isInt = false	}	switch right.(type) {	case int:		rright = int64(right.(int))	case int8:		rright = int64(right.(int8))	case int16:		rright = int64(right.(int16))	case int32:		rright = int64(right.(int32))	case int64:		rright = right.(int64)	case float32:		fright = float64(left.(float32))		isInt = false	case float64:		fleft = left.(float64)		isInt = false	}	if isInt {		return rleft - rright	} else {		return fleft + float64(rleft) - (fright + float64(rright))	}}// DateFormat pattern rules.var datePatterns = []string{	// year	"Y", "2006", // A full numeric representation of a year, 4 digits   Examples: 1999 or 2003	"y", "06", //A two digit representation of a year   Examples: 99 or 03	// month	"m", "01", // Numeric representation of a month, with leading zeros 01 through 12	"n", "1", // Numeric representation of a month, without leading zeros   1 through 12	"M", "Jan", // A short textual representation of a month, three letters Jan through Dec	"F", "January", // A full textual representation of a month, such as January or March   January through December	// day	"d", "02", // Day of the month, 2 digits with leading zeros 01 to 31	"j", "2", // Day of the month without leading zeros 1 to 31	// week	"D", "Mon", // A textual representation of a day, three letters Mon through Sun	"l", "Monday", // A full textual representation of the day of the week  Sunday through Saturday	// time	"g", "3", // 12-hour format of an hour without leading zeros    1 through 12	"G", "15", // 24-hour format of an hour without leading zeros   0 through 23	"h", "03", // 12-hour format of an hour with leading zeros  01 through 12	"H", "15", // 24-hour format of an hour with leading zeros  00 through 23	"a", "pm", // Lowercase Ante meridiem and Post meridiem am or pm	"A", "PM", // Uppercase Ante meridiem and Post meridiem AM or PM	"i", "04", // Minutes with leading zeros    00 to 59	"s", "05", // Seconds, with leading zeros   00 through 59	// time zone	"T", "MST",	"P", "-07:00",	"O", "-0700",	// RFC 2822	"r", time.RFC1123Z,}// Parse Date use PHP time format.func DateParse(dateString, format string) (time.Time, error) {	replacer := strings.NewReplacer(datePatterns...)	format = replacer.Replace(format)	return time.ParseInLocation(format, dateString, time.Local)}// Date takes a PHP like date func to Go's time format.func DateFormat(t time.Time, format string) string {	replacer := strings.NewReplacer(datePatterns...)	format = replacer.Replace(format)	return t.Format(format)}type xssFilter struct {	reg  *regexp.Regexp	repl []byte}var (	whiteSpace = []byte(" ")	xssFilters = []xssFilter{		{regexp.MustCompile(`\ [ONon]\w*=["]*`), whiteSpace},		{regexp.MustCompile(`<[SCRIPTscript]{6}`), whiteSpace},		{regexp.MustCompile(`=[` + "`" + `'"]*[JAVASCRIPTjavascript \t\0
]*:`), whiteSpace},	})// XSS goes through all the XSS filters to make user input content as safe as possible.func XSS(in []byte) []byte {	for _, filter := range xssFilters {		in = filter.reg.ReplaceAll(in, filter.repl)	}	return in}func XSSString(in string) string {	return string(XSS([]byte(in)))}
 |