2017-09-13 15:09:13 +00:00
/ *
*
2022-02-21 03:32:53 +00:00
* Utility Functions And Stuff
* Copyright Azareal 2017 - 2020
2017-09-13 15:09:13 +00:00
*
* /
2017-11-10 03:33:11 +00:00
package common
2017-05-02 17:24:33 +00:00
2017-05-12 13:25:12 +00:00
import (
2022-02-21 03:32:53 +00:00
"crypto/rand"
"encoding/base32"
"encoding/base64"
"encoding/json"
"errors"
"fmt"
"html"
"io/ioutil"
"math"
"os"
"strconv"
"strings"
"time"
"unicode"
2017-05-12 13:25:12 +00:00
)
2016-12-02 07:38:54 +00:00
2017-09-03 04:50:31 +00:00
// Version stores a Gosora version
type Version struct {
2022-02-21 03:32:53 +00:00
Major int
Minor int
Patch int
Tag string
TagID int
2017-05-07 08:31:41 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2019-12-08 03:40:56 +00:00
func ( ver * Version ) String ( ) ( out string ) {
2022-02-21 03:32:53 +00:00
out = strconv . Itoa ( ver . Major ) + "." + strconv . Itoa ( ver . Minor ) + "." + strconv . Itoa ( ver . Patch )
if ver . Tag != "" {
out += "-" + ver . Tag
if ver . TagID != 0 {
out += strconv . Itoa ( ver . TagID )
}
}
return
2017-05-07 08:31:41 +00:00
}
2018-06-17 07:28:18 +00:00
// GenerateSafeString is for generating a cryptographically secure set of random bytes which is base64 encoded and safe for URLs
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2020-02-24 08:28:43 +00:00
func GenerateSafeString ( len int ) ( string , error ) {
2022-02-21 03:32:53 +00:00
rb := make ( [ ] byte , len )
_ , err := rand . Read ( rb )
if err != nil {
return "" , err
}
return base64 . URLEncoding . EncodeToString ( rb ) , nil
2016-12-03 13:45:08 +00:00
}
2018-06-17 07:28:18 +00:00
// GenerateStd32SafeString is for generating a cryptographically secure set of random bytes which is base32 encoded
// ? - Safe for URLs? Mostly likely due to the small range of characters
2020-02-24 08:28:43 +00:00
func GenerateStd32SafeString ( len int ) ( string , error ) {
2022-02-21 03:32:53 +00:00
rb := make ( [ ] byte , len )
_ , err := rand . Read ( rb )
if err != nil {
return "" , err
}
return base32 . StdEncoding . EncodeToString ( rb ) , nil
2018-06-17 07:28:18 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2017-11-11 04:06:16 +00:00
func RelativeTimeFromString ( in string ) ( string , error ) {
2022-02-21 03:32:53 +00:00
if in == "" {
return "" , nil
}
Added Quick Topic.
Added Attachments.
Added Attachment Media Embeds.
Renamed a load of *Store and *Cache methods to reduce the amount of unneccesary typing.
Added petabytes as a unit and cleaned up a few of the friendly units.
Refactored the username change logic to make it easier to maintain.
Refactored the avatar change logic to make it easier to maintain.
Shadow now uses CSS Variables for most of it's colours. We have plans to transpile this to support older browsers later on!
Snuck some CSS Variables into Tempra Conflux.
Added the GroupCache interface to MemoryGroupStore.
Added the Length method to MemoryGroupStore.
Added support for a site short name.
Added the UploadFiles permission.
Renamed more functions.
Fixed the background for the left gutter on the postbit for Tempra Simple and Shadow.
Added support for if statements operating on int8, int16, int32, int32, int64, uint, uint8, uint16, uint32, uint64, float32, and float64 for the template compiler.
Added support for if statements operating on slices and maps for the template compiler.
Fixed a security exploit in reply editing.
Fixed a bug in the URL detector in the parser where it couldn't find URLs with non-standard ports.
Fixed buttons having blue outlines on focus on Shadow.
Refactored the topic creation logic to make it easier to maintain.
Made a few responsive fixes, but there's still more to do in the following commits!
2017-10-05 10:20:28 +00:00
2022-02-21 03:32:53 +00:00
t , err := time . Parse ( "2006-01-02 15:04:05" , in )
if err != nil {
return "" , err
}
2017-06-05 11:57:27 +00:00
2022-02-21 03:32:53 +00:00
return RelativeTime ( t ) , nil
2017-10-14 07:39:22 +00:00
}
// TODO: Write a test for this
2017-11-11 04:06:16 +00:00
func RelativeTime ( t time . Time ) string {
2022-02-21 03:32:53 +00:00
diff := time . Since ( t )
hours := diff . Hours ( )
secs := diff . Seconds ( )
weeks := int ( hours / 24 / 7 )
months := int ( hours / 24 / 31 )
switch {
case months > 3 :
if t . Year ( ) != time . Now ( ) . Year ( ) {
//return t.Format("Mon Jan 2 2006")
return t . Format ( "Jan 2 2006" )
}
return t . Format ( "Jan 2" )
case months > 1 :
return fmt . Sprintf ( "%d months ago" , months )
case months == 1 :
return "a month ago"
case weeks > 1 :
return fmt . Sprintf ( "%d weeks ago" , weeks )
case int ( hours / 24 ) == 7 :
return "a week ago"
case int ( hours / 24 ) == 1 :
return "1 day ago"
case int ( hours / 24 ) > 1 :
return fmt . Sprintf ( "%d days ago" , int ( hours / 24 ) )
case secs <= 1 :
return "a moment ago"
case secs < 60 :
return fmt . Sprintf ( "%d seconds ago" , int ( secs ) )
case secs < 120 :
return "a minute ago"
case secs < 3600 :
return fmt . Sprintf ( "%d minutes ago" , int ( secs / 60 ) )
case secs < 7200 :
return "an hour ago"
}
return fmt . Sprintf ( "%d hours ago" , int ( secs / 60 / 60 ) )
2017-01-03 07:47:31 +00:00
}
2018-12-14 04:08:53 +00:00
// TODO: Finish this faster and more localised version of RelativeTime
/ *
// TODO: Write a test for this
// ! Experimental
func RelativeTimeBytes ( t time . Time , lang int ) [ ] byte {
2022-02-21 03:32:53 +00:00
diff := time . Since ( t )
hours := diff . Hours ( )
secs := diff . Seconds ( )
weeks := int ( hours / 24 / 7 )
months := int ( hours / 24 / 31 )
switch {
case months > 3 :
if t . Year ( ) != time . Now ( ) . Year ( ) {
return [ ] byte ( t . Format ( phrases . RTime . MultiYear ( lang ) ) )
}
return [ ] byte ( t . Format ( phrases . RTime . SingleYear ( lang ) ) )
case months > 1 :
return phrases . RTime . Months ( lang , months )
case months == 1 :
return phrases . RTime . Month ( lang )
case weeks > 1 :
return phrases . RTime . Weeks ( lang , weeks )
case int ( hours / 24 ) == 7 :
return phrases . RTime . Week ( lang )
case int ( hours / 24 ) == 1 :
return phrases . RTime . Day ( lang )
case int ( hours / 24 ) > 1 :
return phrases . RTime . Days ( lang , int ( hours / 24 ) )
case secs <= 1 :
return phrases . RTime . Moment ( lang )
case secs < 60 :
return phrases . RTime . Seconds ( lang , int ( secs ) )
case secs < 120 :
return phrases . RTime . Minute ( lang )
case secs < 3600 :
return phrases . RTime . Minutes ( lang , int ( secs / 60 ) )
case secs < 7200 :
return phrases . RTime . Hour ( lang )
}
return phrases . RTime . Hours ( lang , int ( secs / 60 / 60 ) )
2018-12-14 04:08:53 +00:00
}
* /
2020-06-17 22:03:36 +00:00
var pMs = 1000
var pSec = pMs * 1000
var pMin = pSec * 60
var pHour = pMin * 60
var pDay = pHour * 24
2020-02-24 08:28:43 +00:00
func ConvertPerfUnit ( quan float64 ) ( out float64 , unit string ) {
2022-02-21 03:32:53 +00:00
f := func ( ) ( float64 , string ) {
switch {
case quan >= float64 ( pDay ) :
return quan / float64 ( pDay ) , "d"
case quan >= float64 ( pHour ) :
return quan / float64 ( pHour ) , "h"
case quan >= float64 ( pMin ) :
return quan / float64 ( pMin ) , "m"
case quan >= float64 ( pSec ) :
return quan / float64 ( pSec ) , "s"
case quan >= float64 ( pMs ) :
return quan / float64 ( pMs ) , "ms"
}
return quan , "μs"
}
out , unit = f ( )
return math . Ceil ( out ) , unit
2020-02-24 08:28:43 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2017-11-11 04:06:16 +00:00
func ConvertByteUnit ( bytes float64 ) ( float64 , string ) {
2022-02-21 03:32:53 +00:00
switch {
case bytes >= float64 ( Petabyte ) :
return bytes / float64 ( Petabyte ) , "PB"
case bytes >= float64 ( Terabyte ) :
return bytes / float64 ( Terabyte ) , "TB"
case bytes >= float64 ( Gigabyte ) :
return bytes / float64 ( Gigabyte ) , "GB"
case bytes >= float64 ( Megabyte ) :
return bytes / float64 ( Megabyte ) , "MB"
case bytes >= float64 ( Kilobyte ) :
return bytes / float64 ( Kilobyte ) , "KB"
}
return bytes , " bytes"
2017-05-07 08:31:41 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2017-11-11 04:06:16 +00:00
func ConvertByteInUnit ( bytes float64 , unit string ) ( count float64 ) {
2022-02-21 03:32:53 +00:00
switch unit {
case "PB" :
count = bytes / float64 ( Petabyte )
case "TB" :
count = bytes / float64 ( Terabyte )
case "GB" :
count = bytes / float64 ( Gigabyte )
case "MB" :
count = bytes / float64 ( Megabyte )
case "KB" :
count = bytes / float64 ( Kilobyte )
default :
count = 0.1
}
if count < 0.1 {
count = 0.1
}
return
2017-05-07 08:31:41 +00:00
}
2018-06-17 07:28:18 +00:00
// TODO: Write a test for this
2018-12-27 05:42:41 +00:00
// TODO: Localise this?
2018-06-17 07:28:18 +00:00
func FriendlyUnitToBytes ( quantity int , unit string ) ( bytes int , err error ) {
2022-02-21 03:32:53 +00:00
switch unit {
case "PB" :
bytes = quantity * Petabyte
case "TB" :
bytes = quantity * Terabyte
case "GB" :
bytes = quantity * Gigabyte
case "MB" :
bytes = quantity * Megabyte
case "KB" :
bytes = quantity * Kilobyte
case "" :
// Do nothing
default :
return bytes , errors . New ( "Unknown unit" )
}
return bytes , nil
2018-06-17 07:28:18 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2018-05-14 08:56:56 +00:00
// TODO: Re-add T as int64
2017-11-11 04:06:16 +00:00
func ConvertUnit ( num int ) ( int , string ) {
2022-02-21 03:32:53 +00:00
switch {
case num >= 1000000000000 :
return num / 1000000000000 , "T"
case num >= 1000000000 :
return num / 1000000000 , "B"
case num >= 1000000 :
return num / 1000000 , "M"
case num >= 1000 :
return num / 1000 , "K"
}
return num , ""
2017-05-12 13:25:12 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2018-05-14 08:56:56 +00:00
// TODO: Re-add quadrillion as int64
// TODO: Re-add trillion as int64
2017-11-11 04:06:16 +00:00
func ConvertFriendlyUnit ( num int ) ( int , string ) {
2022-02-21 03:32:53 +00:00
switch {
case num >= 1000000000000000 :
return 0 , " quadrillion"
case num >= 1000000000000 :
return 0 , " trillion"
case num >= 1000000000 :
return num / 1000000000 , " billion"
case num >= 1000000 :
return num / 1000000 , " million"
case num >= 1000 :
return num / 1000 , " thousand"
}
return num , ""
2017-05-12 13:25:12 +00:00
}
2017-12-24 07:38:46 +00:00
// TODO: Make slugs optional for certain languages across the entirety of Gosora?
// TODO: Let plugins replace NameToSlug and the URL building logic with their own
2019-06-05 05:10:42 +00:00
/ * func NameToSlug ( name string ) ( slug string ) {
2022-02-21 03:32:53 +00:00
// TODO: Do we want this reliant on config file flags? This might complicate tests and oddball uses
if ! Config . BuildSlugs {
return ""
}
name = strings . TrimSpace ( name )
name = strings . Replace ( name , " " , " " , - 1 )
for _ , char := range name {
if unicode . IsLower ( char ) || unicode . IsNumber ( char ) {
slug += string ( char )
} else if unicode . IsUpper ( char ) {
slug += string ( unicode . ToLower ( char ) )
} else if unicode . IsSpace ( char ) {
slug += "-"
}
}
if slug == "" {
slug = "untitled"
}
return slug
2019-06-05 05:10:42 +00:00
} * /
// TODO: Make slugs optional for certain languages across the entirety of Gosora?
// TODO: Let plugins replace NameToSlug and the URL building logic with their own
func NameToSlug ( name string ) ( slug string ) {
2022-02-21 03:32:53 +00:00
// TODO: Do we want this reliant on config file flags? This might complicate tests and oddball uses
if ! Config . BuildSlugs {
return ""
}
name = strings . TrimSpace ( name )
name = strings . Replace ( name , " " , " " , - 1 )
var sb strings . Builder
for _ , char := range name {
if unicode . IsLower ( char ) || unicode . IsNumber ( char ) {
sb . WriteRune ( char )
} else if unicode . IsUpper ( char ) {
sb . WriteRune ( unicode . ToLower ( char ) )
} else if unicode . IsSpace ( char ) {
sb . WriteByte ( '-' )
}
}
if sb . Len ( ) == 0 {
return "untitled"
}
return sb . String ( )
2017-06-28 12:05:26 +00:00
}
2018-06-30 10:22:39 +00:00
// TODO: Write a test for this
func HasSuspiciousEmail ( email string ) bool {
2022-02-21 03:32:53 +00:00
if email == "" {
return false
}
lowEmail := strings . ToLower ( email )
// TODO: Use a more flexible blacklist, perhaps with a similar mechanism to the HTML tag registration system in PreparseMessage()
if ! strings . Contains ( lowEmail , "@" ) || strings . Contains ( lowEmail , "casino" ) || strings . Contains ( lowEmail , "viagra" ) || strings . Contains ( lowEmail , "pharma" ) || strings . Contains ( lowEmail , "pill" ) {
return true
}
var dotCount , shortBits , currentSegmentLength int
for _ , char := range lowEmail {
if char == '.' {
dotCount ++
if currentSegmentLength < 3 {
shortBits ++
}
currentSegmentLength = 0
} else {
currentSegmentLength ++
}
}
return dotCount > 7 || shortBits > 2
2018-06-30 10:22:39 +00:00
}
2020-06-17 22:03:36 +00:00
func unmarshalJsonFile ( name string , in interface { } ) error {
2022-02-21 03:32:53 +00:00
data , err := ioutil . ReadFile ( name )
if err != nil {
return err
}
return json . Unmarshal ( data , in )
2020-06-17 22:03:36 +00:00
}
func unmarshalJsonFileIgnore404 ( name string , in interface { } ) error {
2022-02-21 03:32:53 +00:00
data , err := ioutil . ReadFile ( name )
if err == os . ErrPermission || err == os . ErrClosed {
return err
} else if err != nil {
return nil
}
return json . Unmarshal ( data , in )
2020-06-17 22:03:36 +00:00
}
2020-06-16 03:25:38 +00:00
func CanonEmail ( email string ) string {
2022-02-21 03:32:53 +00:00
email = strings . ToLower ( email )
2020-06-17 22:03:36 +00:00
2022-02-21 03:32:53 +00:00
// Gmail emails are equivalent without the dots
espl := strings . Split ( email , "@" )
if len ( espl ) >= 2 && espl [ 1 ] == "gmail.com" {
return strings . Replace ( espl [ 0 ] , "." , "" , - 1 ) + "@" + espl [ 1 ]
}
2020-06-16 03:25:38 +00:00
2022-02-21 03:32:53 +00:00
return email
2020-06-16 03:25:38 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2017-09-03 04:50:31 +00:00
func createFile ( name string ) error {
2022-02-21 03:32:53 +00:00
f , err := os . Create ( name )
if err != nil {
return err
}
return f . Close ( )
2017-08-13 11:22:34 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2019-12-08 03:40:56 +00:00
func writeFile ( name , content string ) ( err error ) {
2022-02-21 03:32:53 +00:00
f , err := os . Create ( name )
if err != nil {
return err
}
_ , err = f . WriteString ( content )
if err != nil {
return err
}
err = f . Sync ( )
if err != nil {
return err
}
return f . Close ( )
2017-01-05 14:41:14 +00:00
}
2017-01-12 02:55:08 +00:00
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
func Stripslashes ( text string ) string {
2022-02-21 03:32:53 +00:00
text = strings . Replace ( text , "/" , "" , - 1 )
return strings . Replace ( text , "\\" , "" , - 1 )
2017-09-23 19:57:13 +00:00
}
2018-10-04 09:01:07 +00:00
// The word counter might run into problems with some languages where words aren't as obviously demarcated, I would advise turning it off in those cases, or if it becomes annoying in general, really.
2017-11-11 04:06:16 +00:00
func WordCount ( input string ) ( count int ) {
2022-02-21 03:32:53 +00:00
input = strings . TrimSpace ( input )
if input == "" {
return 0
}
var inSpace bool
for _ , value := range input {
if unicode . IsSpace ( value ) || unicode . IsPunct ( value ) {
if ! inSpace {
inSpace = true
}
} else if inSpace {
count ++
inSpace = false
}
}
return count + 1
2017-01-12 02:55:08 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2017-11-11 04:06:16 +00:00
func GetLevel ( score int ) ( level int ) {
2022-02-21 03:32:53 +00:00
var base float64 = 25
var current , prev float64
var expFactor = 2.8
for i := 1 ; ; i ++ {
_ , bit := math . Modf ( float64 ( i ) / 10 )
if bit == 0 {
expFactor += 0.1
}
current = base + math . Pow ( float64 ( i ) , expFactor ) + ( prev / 3 )
prev = current
if float64 ( score ) < current {
break
}
level ++
}
return level
2017-01-12 02:55:08 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2017-11-11 04:06:16 +00:00
func GetLevelScore ( getLevel int ) ( score int ) {
2022-02-21 03:32:53 +00:00
var base float64 = 25
var current float64
var expFactor = 2.8
for i := 1 ; i <= getLevel ; i ++ {
_ , bit := math . Modf ( float64 ( i ) / 10 )
if bit == 0 {
expFactor += 0.1
}
current = base + math . Pow ( float64 ( i ) , expFactor ) + ( current / 3 )
//fmt.Println("level: ", i)
//fmt.Println("current: ", current)
}
return int ( math . Ceil ( current ) )
2017-01-12 02:55:08 +00:00
}
2017-09-23 19:57:13 +00:00
// TODO: Write a test for this
2017-11-11 04:06:16 +00:00
func GetLevels ( maxLevel int ) [ ] float64 {
2022-02-21 03:32:53 +00:00
var base float64 = 25
var current , prev float64 // = 0
var expFactor = 2.8
var out [ ] float64
out = append ( out , 0 )
for i := 1 ; i <= maxLevel ; i ++ {
_ , bit := math . Modf ( float64 ( i ) / 10 )
if bit == 0 {
expFactor += 0.1
}
current = base + math . Pow ( float64 ( i ) , expFactor ) + ( prev / 3 )
prev = current
out = append ( out , current )
}
return out
2017-01-12 02:55:08 +00:00
}
2017-01-26 13:37:50 +00:00
2018-05-31 06:51:31 +00:00
// TODO: Write a test for this
// SanitiseSingleLine is a generic function for escaping html entities and removing silly characters from usernames and topic titles. It also strips newline characters
func SanitiseSingleLine ( in string ) string {
2022-02-21 03:32:53 +00:00
in = strings . Replace ( in , "\n" , "" , - 1 )
in = strings . Replace ( in , "\r" , "" , - 1 )
return SanitiseBody ( in )
2018-05-31 06:51:31 +00:00
}
// TODO: Write a test for this
// TODO: Add more strange characters
// TODO: Strip all sub-32s minus \r and \n?
// SanitiseBody is the same as SanitiseSingleLine, but it doesn't strip newline characters
func SanitiseBody ( in string ) string {
2022-02-21 03:32:53 +00:00
in = strings . Replace ( in , " " , "" , - 1 ) // Strip Zero length space
in = html . EscapeString ( in )
return strings . TrimSpace ( in )
2018-05-31 06:51:31 +00:00
}
2017-11-11 04:06:16 +00:00
func BuildSlug ( slug string , id int ) string {
2022-02-21 03:32:53 +00:00
if slug == "" || ! Config . BuildSlugs {
return strconv . Itoa ( id )
}
return slug + "." + strconv . Itoa ( id )
2019-10-29 22:13:45 +00:00
}