2014-10-16 20:01:43 +02:00
|
|
|
/*
|
|
|
|
The MIT License (MIT)
|
|
|
|
|
2017-03-23 00:02:36 +01:00
|
|
|
Copyright (c) 2014-2017 DutchCoders [https://github.com/dutchcoders/]
|
2014-10-16 20:01:43 +02:00
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2017-03-22 18:09:21 +01:00
|
|
|
package server
|
2014-10-16 20:01:43 +02:00
|
|
|
|
|
|
|
import (
|
2014-11-13 21:41:43 +01:00
|
|
|
"math"
|
2014-10-16 20:01:43 +02:00
|
|
|
"net/http"
|
|
|
|
"net/mail"
|
2014-11-13 21:41:43 +01:00
|
|
|
"strconv"
|
2014-10-16 20:01:43 +02:00
|
|
|
"strings"
|
|
|
|
"time"
|
2016-05-19 23:33:44 +02:00
|
|
|
|
|
|
|
"github.com/goamz/goamz/aws"
|
|
|
|
"github.com/goamz/goamz/s3"
|
|
|
|
"github.com/golang/gddo/httputil/header"
|
2014-10-16 20:01:43 +02:00
|
|
|
)
|
|
|
|
|
2017-07-16 15:19:41 +02:00
|
|
|
func getBucket(accessKey, secretKey, bucket, endpoint string) (*s3.Bucket, error) {
|
2017-03-22 18:09:21 +01:00
|
|
|
auth, err := aws.GetAuth(accessKey, secretKey, "", time.Time{})
|
2014-10-16 20:01:43 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-05-19 23:33:44 +02:00
|
|
|
var EUWestWithoutHTTPS = aws.Region{
|
2017-03-22 21:09:40 +01:00
|
|
|
Name: "eu-west-1",
|
|
|
|
EC2Endpoint: "https://ec2.eu-west-1.amazonaws.com",
|
2017-07-16 15:19:41 +02:00
|
|
|
S3Endpoint: endpoint,
|
2017-03-22 21:09:40 +01:00
|
|
|
S3BucketEndpoint: "",
|
|
|
|
S3LocationConstraint: true,
|
|
|
|
S3LowercaseBucket: true,
|
|
|
|
SDBEndpoint: "https://sdb.eu-west-1.amazonaws.com",
|
|
|
|
SESEndpoint: "https://email.eu-west-1.amazonaws.com",
|
|
|
|
SNSEndpoint: "https://sns.eu-west-1.amazonaws.com",
|
|
|
|
SQSEndpoint: "https://sqs.eu-west-1.amazonaws.com",
|
|
|
|
IAMEndpoint: "https://iam.amazonaws.com",
|
|
|
|
ELBEndpoint: "https://elasticloadbalancing.eu-west-1.amazonaws.com",
|
|
|
|
DynamoDBEndpoint: "https://dynamodb.eu-west-1.amazonaws.com",
|
|
|
|
CloudWatchServicepoint: aws.ServiceInfo{
|
|
|
|
Endpoint: "https://monitoring.eu-west-1.amazonaws.com",
|
|
|
|
Signer: aws.V2Signature,
|
|
|
|
},
|
|
|
|
AutoScalingEndpoint: "https://autoscaling.eu-west-1.amazonaws.com",
|
|
|
|
RDSEndpoint: aws.ServiceInfo{
|
|
|
|
Endpoint: "https://rds.eu-west-1.amazonaws.com",
|
|
|
|
Signer: aws.V2Signature,
|
|
|
|
},
|
|
|
|
STSEndpoint: "https://sts.amazonaws.com",
|
|
|
|
CloudFormationEndpoint: "https://cloudformation.eu-west-1.amazonaws.com",
|
|
|
|
ECSEndpoint: "https://ecs.eu-west-1.amazonaws.com",
|
|
|
|
DynamoDBStreamsEndpoint: "https://streams.dynamodb.eu-west-1.amazonaws.com",
|
2016-05-19 23:33:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
conn := s3.New(auth, EUWestWithoutHTTPS)
|
2017-03-22 18:09:21 +01:00
|
|
|
b := conn.Bucket(bucket)
|
2014-10-16 20:01:43 +02:00
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
2014-11-13 21:41:43 +01:00
|
|
|
func formatNumber(format string, s uint64) string {
|
|
|
|
|
|
|
|
return RenderFloat(format, float64(s))
|
|
|
|
}
|
|
|
|
|
|
|
|
var renderFloatPrecisionMultipliers = [10]float64{
|
|
|
|
1,
|
|
|
|
10,
|
|
|
|
100,
|
|
|
|
1000,
|
|
|
|
10000,
|
|
|
|
100000,
|
|
|
|
1000000,
|
|
|
|
10000000,
|
|
|
|
100000000,
|
|
|
|
1000000000,
|
|
|
|
}
|
|
|
|
|
|
|
|
var renderFloatPrecisionRounders = [10]float64{
|
|
|
|
0.5,
|
|
|
|
0.05,
|
|
|
|
0.005,
|
|
|
|
0.0005,
|
|
|
|
0.00005,
|
|
|
|
0.000005,
|
|
|
|
0.0000005,
|
|
|
|
0.00000005,
|
|
|
|
0.000000005,
|
|
|
|
0.0000000005,
|
|
|
|
}
|
|
|
|
|
|
|
|
func RenderFloat(format string, n float64) string {
|
|
|
|
// Special cases:
|
|
|
|
// NaN = "NaN"
|
|
|
|
// +Inf = "+Infinity"
|
|
|
|
// -Inf = "-Infinity"
|
|
|
|
if math.IsNaN(n) {
|
|
|
|
return "NaN"
|
|
|
|
}
|
|
|
|
if n > math.MaxFloat64 {
|
|
|
|
return "Infinity"
|
|
|
|
}
|
|
|
|
if n < -math.MaxFloat64 {
|
|
|
|
return "-Infinity"
|
|
|
|
}
|
|
|
|
|
|
|
|
// default format
|
|
|
|
precision := 2
|
|
|
|
decimalStr := "."
|
|
|
|
thousandStr := ","
|
|
|
|
positiveStr := ""
|
|
|
|
negativeStr := "-"
|
|
|
|
|
|
|
|
if len(format) > 0 {
|
|
|
|
// If there is an explicit format directive,
|
|
|
|
// then default values are these:
|
|
|
|
precision = 9
|
|
|
|
thousandStr = ""
|
|
|
|
|
|
|
|
// collect indices of meaningful formatting directives
|
|
|
|
formatDirectiveChars := []rune(format)
|
|
|
|
formatDirectiveIndices := make([]int, 0)
|
|
|
|
for i, char := range formatDirectiveChars {
|
|
|
|
if char != '#' && char != '0' {
|
|
|
|
formatDirectiveIndices = append(formatDirectiveIndices, i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(formatDirectiveIndices) > 0 {
|
|
|
|
// Directive at index 0:
|
|
|
|
// Must be a '+'
|
|
|
|
// Raise an error if not the case
|
|
|
|
// index: 0123456789
|
|
|
|
// +0.000,000
|
|
|
|
// +000,000.0
|
|
|
|
// +0000.00
|
|
|
|
// +0000
|
|
|
|
if formatDirectiveIndices[0] == 0 {
|
|
|
|
if formatDirectiveChars[formatDirectiveIndices[0]] != '+' {
|
|
|
|
panic("RenderFloat(): invalid positive sign directive")
|
|
|
|
}
|
|
|
|
positiveStr = "+"
|
|
|
|
formatDirectiveIndices = formatDirectiveIndices[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
// Two directives:
|
|
|
|
// First is thousands separator
|
|
|
|
// Raise an error if not followed by 3-digit
|
|
|
|
// 0123456789
|
|
|
|
// 0.000,000
|
|
|
|
// 000,000.00
|
|
|
|
if len(formatDirectiveIndices) == 2 {
|
|
|
|
if (formatDirectiveIndices[1] - formatDirectiveIndices[0]) != 4 {
|
|
|
|
panic("RenderFloat(): thousands separator directive must be followed by 3 digit-specifiers")
|
|
|
|
}
|
|
|
|
thousandStr = string(formatDirectiveChars[formatDirectiveIndices[0]])
|
|
|
|
formatDirectiveIndices = formatDirectiveIndices[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
// One directive:
|
|
|
|
// Directive is decimal separator
|
|
|
|
// The number of digit-specifier following the separator indicates wanted precision
|
|
|
|
// 0123456789
|
|
|
|
// 0.00
|
|
|
|
// 000,0000
|
|
|
|
if len(formatDirectiveIndices) == 1 {
|
|
|
|
decimalStr = string(formatDirectiveChars[formatDirectiveIndices[0]])
|
|
|
|
precision = len(formatDirectiveChars) - formatDirectiveIndices[0] - 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// generate sign part
|
|
|
|
var signStr string
|
|
|
|
if n >= 0.000000001 {
|
|
|
|
signStr = positiveStr
|
|
|
|
} else if n <= -0.000000001 {
|
|
|
|
signStr = negativeStr
|
|
|
|
n = -n
|
|
|
|
} else {
|
|
|
|
signStr = ""
|
|
|
|
n = 0.0
|
|
|
|
}
|
|
|
|
|
|
|
|
// split number into integer and fractional parts
|
|
|
|
intf, fracf := math.Modf(n + renderFloatPrecisionRounders[precision])
|
|
|
|
|
|
|
|
// generate integer part string
|
|
|
|
intStr := strconv.Itoa(int(intf))
|
|
|
|
|
|
|
|
// add thousand separator if required
|
|
|
|
if len(thousandStr) > 0 {
|
|
|
|
for i := len(intStr); i > 3; {
|
|
|
|
i -= 3
|
|
|
|
intStr = intStr[:i] + thousandStr + intStr[i:]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// no fractional part, we can leave now
|
|
|
|
if precision == 0 {
|
|
|
|
return signStr + intStr
|
|
|
|
}
|
|
|
|
|
|
|
|
// generate fractional part
|
|
|
|
fracStr := strconv.Itoa(int(fracf * renderFloatPrecisionMultipliers[precision]))
|
|
|
|
// may need padding
|
|
|
|
if len(fracStr) < precision {
|
|
|
|
fracStr = "000000000000000"[:precision-len(fracStr)] + fracStr
|
|
|
|
}
|
|
|
|
|
|
|
|
return signStr + intStr + decimalStr + fracStr
|
|
|
|
}
|
|
|
|
|
|
|
|
func RenderInteger(format string, n int) string {
|
|
|
|
return RenderFloat(format, float64(n))
|
|
|
|
}
|
|
|
|
|
2014-10-16 20:01:43 +02:00
|
|
|
// Request.RemoteAddress contains port, which we want to remove i.e.:
|
|
|
|
// "[::1]:58292" => "[::1]"
|
|
|
|
func ipAddrFromRemoteAddr(s string) string {
|
|
|
|
idx := strings.LastIndex(s, ":")
|
|
|
|
if idx == -1 {
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
return s[:idx]
|
|
|
|
}
|
|
|
|
|
2017-03-22 21:09:40 +01:00
|
|
|
func getIPAddress(r *http.Request) string {
|
2014-10-16 20:01:43 +02:00
|
|
|
hdr := r.Header
|
2017-03-22 21:09:40 +01:00
|
|
|
hdrRealIP := hdr.Get("X-Real-Ip")
|
2014-10-16 20:01:43 +02:00
|
|
|
hdrForwardedFor := hdr.Get("X-Forwarded-For")
|
2017-03-22 21:09:40 +01:00
|
|
|
if hdrRealIP == "" && hdrForwardedFor == "" {
|
2014-10-16 20:01:43 +02:00
|
|
|
return ipAddrFromRemoteAddr(r.RemoteAddr)
|
|
|
|
}
|
|
|
|
if hdrForwardedFor != "" {
|
|
|
|
// X-Forwarded-For is potentially a list of addresses separated with ","
|
|
|
|
parts := strings.Split(hdrForwardedFor, ",")
|
|
|
|
for i, p := range parts {
|
|
|
|
parts[i] = strings.TrimSpace(p)
|
|
|
|
}
|
2017-03-22 21:09:40 +01:00
|
|
|
|
2014-10-16 20:01:43 +02:00
|
|
|
// TODO: should return first non-local address
|
|
|
|
return parts[0]
|
|
|
|
}
|
2017-03-22 21:09:40 +01:00
|
|
|
return hdrRealIP
|
2014-10-16 20:01:43 +02:00
|
|
|
}
|
|
|
|
|
2017-03-22 21:09:40 +01:00
|
|
|
func encodeRFC2047(s string) string {
|
2014-10-16 20:01:43 +02:00
|
|
|
// use mail's rfc2047 to encode any string
|
2017-03-22 21:09:40 +01:00
|
|
|
addr := mail.Address{
|
|
|
|
Name: s,
|
|
|
|
Address: "",
|
|
|
|
}
|
2014-10-16 20:01:43 +02:00
|
|
|
return strings.Trim(addr.String(), " <>")
|
|
|
|
}
|
2014-11-13 21:41:43 +01:00
|
|
|
|
2017-03-22 21:09:40 +01:00
|
|
|
func acceptsHTML(hdr http.Header) bool {
|
2014-11-13 21:41:43 +01:00
|
|
|
actual := header.ParseAccept(hdr, "Accept")
|
|
|
|
|
|
|
|
for _, s := range actual {
|
|
|
|
if s.Value == "text/html" {
|
|
|
|
return (true)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (false)
|
|
|
|
}
|