summaryrefslogtreecommitdiffstats
path: root/vendor/strk.kbt.io/projects/go/libravatar
diff options
context:
space:
mode:
Diffstat (limited to 'vendor/strk.kbt.io/projects/go/libravatar')
-rw-r--r--vendor/strk.kbt.io/projects/go/libravatar/LICENSE19
-rw-r--r--vendor/strk.kbt.io/projects/go/libravatar/README.md25
-rw-r--r--vendor/strk.kbt.io/projects/go/libravatar/libravatar.go297
3 files changed, 341 insertions, 0 deletions
diff --git a/vendor/strk.kbt.io/projects/go/libravatar/LICENSE b/vendor/strk.kbt.io/projects/go/libravatar/LICENSE
new file mode 100644
index 0000000000..8c89d4e34c
--- /dev/null
+++ b/vendor/strk.kbt.io/projects/go/libravatar/LICENSE
@@ -0,0 +1,19 @@
+Copyright (c) 2016 Sandro Santilli <strk@kbt.io>
+
+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.
diff --git a/vendor/strk.kbt.io/projects/go/libravatar/README.md b/vendor/strk.kbt.io/projects/go/libravatar/README.md
new file mode 100644
index 0000000000..660cd90643
--- /dev/null
+++ b/vendor/strk.kbt.io/projects/go/libravatar/README.md
@@ -0,0 +1,25 @@
+Simple [golang](https://www.golang.org) library for serving
+[federated avatars](https://www.libravatar.org)
+
+# Use
+
+```sh
+go get strk.kbt.io/projects/go/libravatar
+cd $GOPATH/src/strk.kbt.io/projects/go/libravatar
+go doc
+```
+
+# Contribute
+
+A clone of the code repository would be downloaded by `go get`.
+You can send patches or pull requests to strk@kbt.io.
+
+If you need a place to publish your contribution branches,
+you could start from a fork of the gitlab mirror:
+https://gitlab.com/strk/go-libravatar
+
+# Contacts
+
+ * Project homepage: http://strk.kbt.io/projects/go/libravatar
+ * Maintainer: Sandro Santilli <strk@kbt.io>
+
diff --git a/vendor/strk.kbt.io/projects/go/libravatar/libravatar.go b/vendor/strk.kbt.io/projects/go/libravatar/libravatar.go
new file mode 100644
index 0000000000..4c748456f3
--- /dev/null
+++ b/vendor/strk.kbt.io/projects/go/libravatar/libravatar.go
@@ -0,0 +1,297 @@
+// Copyright 2016 by Sandro Santilli <strk@kbt.io>
+// Use of this source code is governed by a MIT
+// license that can be found in the LICENSE file.
+
+// Implements support for federated avatars lookup.
+// See https://wiki.libravatar.org/api/
+
+package libravatar
+
+import (
+ "crypto/md5"
+ "crypto/sha256"
+ "fmt"
+ "math/rand"
+ "net"
+ "net/mail"
+ "net/url"
+ "strings"
+ "time"
+)
+
+// Default images (to be used as defaultURL)
+const (
+ // Do not load any image if none is associated with the email
+ // hash, instead return an HTTP 404 (File Not Found) response
+ HTTP404 = "404"
+ // (mystery-man) a simple, cartoon-style silhouetted outline of
+ // a person (does not vary by email hash)
+ MysteryMan = "mm"
+ // a geometric pattern based on an email hash
+ IdentIcon = "identicon"
+ // a generated 'monster' with different colors, faces, etc
+ MonsterID = "monsterid"
+ // generated faces with differing features and backgrounds
+ Wavatar = "wavatar"
+ // awesome generated, 8-bit arcade-style pixelated faces
+ Retro = "retro"
+)
+
+var (
+ // Default object, enabling object-less function calls
+ DefaultLibravatar = New()
+)
+
+/* This should be moved in its own file */
+type cacheKey struct {
+ service string
+ domain string
+}
+
+type cacheValue struct {
+ target string
+ checkedAt time.Time
+}
+
+type Libravatar struct {
+ defUrl string // default url
+ picSize int // picture size
+ fallbackHost string // default fallback URL
+ secureFallbackHost string // default fallback URL for secure connections
+ useHTTPS bool
+ nameCache map[cacheKey]cacheValue
+ nameCacheDuration time.Duration
+ minSize uint // smallest image dimension allowed
+ maxSize uint // largest image dimension allowed
+ size uint // what dimension should be used
+ serviceBase string // SRV record to be queried for federation
+ secureServiceBase string // SRV record to be queried for federation with secure servers
+}
+
+// Instanciate a library handle
+func New() *Libravatar {
+ // According to https://wiki.libravatar.org/running_your_own/
+ // the time-to-live (cache expiry) should be set to at least 1 day.
+ return &Libravatar{
+ fallbackHost: `cdn.libravatar.org`,
+ secureFallbackHost: `seccdn.libravatar.org`,
+ minSize: 1,
+ maxSize: 512,
+ size: 0, // unset, defaults to 80
+ serviceBase: `avatars`,
+ secureServiceBase: `avatars-sec`,
+ nameCache: make(map[cacheKey]cacheValue),
+ nameCacheDuration: 24 * time.Hour,
+ }
+}
+
+// Set the hostname for fallbacks in case no avatar service is defined
+// for a domain
+func (v *Libravatar) SetFallbackHost(host string) {
+ v.fallbackHost = host
+}
+
+// Set the hostname for fallbacks in case no avatar service is defined
+// for a domain, when requiring secure domains
+func (v *Libravatar) SetSecureFallbackHost(host string) {
+ v.secureFallbackHost = host
+}
+
+// Set useHTTPS flag
+func (v *Libravatar) SetUseHTTPS(use bool) {
+ v.useHTTPS = use
+}
+
+// Set Avatars image dimension (0 for default)
+func (v *Libravatar) SetAvatarSize(size uint) {
+ v.size = size
+}
+
+// generate hash, either with email address or OpenID
+func (v *Libravatar) genHash(email *mail.Address, openid *url.URL) string {
+ if email != nil {
+ email.Address = strings.ToLower(strings.TrimSpace(email.Address))
+ sum := md5.Sum([]byte(email.Address))
+ return fmt.Sprintf("%x", sum)
+ } else if openid != nil {
+ openid.Scheme = strings.ToLower(openid.Scheme)
+ openid.Host = strings.ToLower(openid.Host)
+ sum := sha256.Sum256([]byte(openid.String()))
+ return fmt.Sprintf("%x", sum)
+ }
+ // panic, because this should not be reachable
+ panic("Neither Email or OpenID set")
+}
+
+// Gets domain out of email or openid (for openid to be parsed, email has to be nil)
+func (v *Libravatar) getDomain(email *mail.Address, openid *url.URL) string {
+ if email != nil {
+ u, err := url.Parse("//" + email.Address)
+ if err != nil {
+ if v.useHTTPS && v.secureFallbackHost != "" {
+ return v.secureFallbackHost
+ }
+ return v.fallbackHost
+ }
+ return u.Host
+ } else if openid != nil {
+ return openid.Host
+ }
+ // panic, because this should not be reachable
+ panic("Neither Email or OpenID set")
+}
+
+// Processes email or openid (for openid to be processed, email has to be nil)
+func (v *Libravatar) process(email *mail.Address, openid *url.URL) (string, error) {
+ URL, err := v.baseURL(email, openid)
+ if err != nil {
+ return "", err
+ }
+ res := fmt.Sprintf("%s/avatar/%s", URL, v.genHash(email, openid))
+
+ values := make(url.Values)
+ if v.defUrl != "" {
+ values.Add("d", v.defUrl)
+ }
+ if v.size > 0 {
+ values.Add("s", fmt.Sprintf("%d", v.size))
+ }
+
+ if len(values) > 0 {
+ return fmt.Sprintf("%s?%s", res, values.Encode()), nil
+ }
+ return res, nil
+}
+
+// Finds or defaults a URL for Federation (for openid to be used, email has to be nil)
+func (v *Libravatar) baseURL(email *mail.Address, openid *url.URL) (string, error) {
+ var service, protocol, domain string
+
+ if v.useHTTPS {
+ protocol = "https://"
+ service = v.secureServiceBase
+ domain = v.secureFallbackHost
+
+ } else {
+ protocol = "http://"
+ service = v.serviceBase
+ domain = v.fallbackHost
+ }
+
+ host := v.getDomain(email, openid)
+ key := cacheKey{service, host}
+ now := time.Now()
+ val, found := v.nameCache[key]
+ if found && now.Sub(val.checkedAt) <= v.nameCacheDuration {
+ return protocol + val.target, nil
+ }
+
+ _, addrs, err := net.LookupSRV(service, "tcp", host)
+ if err != nil && err.(*net.DNSError).IsTimeout {
+ return "", err
+ }
+
+ if len(addrs) == 1 {
+ // select only record, if only one is available
+ domain = strings.TrimSuffix(addrs[0].Target, ".")
+ } else if len(addrs) > 1 {
+ // Select first record according to RFC2782 weight
+ // ordering algorithm (page 3)
+
+ type record struct {
+ srv *net.SRV
+ weight uint16
+ }
+
+ var (
+ total_weight uint16
+ records []record
+ top_priority = addrs[0].Priority
+ top_record *net.SRV
+ )
+
+ for _, rr := range addrs {
+ if rr.Priority > top_priority {
+ continue
+ } else if rr.Priority < top_priority {
+ // won't happen, because net sorts
+ // by priority, but just in case
+ total_weight = 0
+ records = nil
+ top_priority = rr.Priority
+ }
+
+ total_weight += rr.Weight
+
+ if rr.Weight > 0 {
+ records = append(records, record{rr, total_weight})
+ } else if rr.Weight == 0 {
+ records = append([]record{record{srv: rr, weight: total_weight}}, records...)
+ }
+ }
+
+ if len(records) == 1 {
+ top_record = records[0].srv
+ } else {
+ randnum := uint16(rand.Intn(int(total_weight)))
+
+ for _, rr := range records {
+ if rr.weight >= randnum {
+ top_record = rr.srv
+ break
+ }
+ }
+ }
+
+ domain = fmt.Sprintf("%s:%d", top_record.Target, top_record.Port)
+ }
+
+ v.nameCache[key] = cacheValue{checkedAt: now, target: domain}
+ return protocol + domain, nil
+}
+
+// Return url of the avatar for the given email
+func (v *Libravatar) FromEmail(email string) (string, error) {
+ addr, err := mail.ParseAddress(email)
+ if err != nil {
+ return "", err
+ }
+
+ link, err := v.process(addr, nil)
+ if err != nil {
+ return "", err
+ }
+
+ return link, nil
+}
+
+// Object-less call to DefaultLibravatar for an email adders
+func FromEmail(email string) (string, error) {
+ return DefaultLibravatar.FromEmail(email)
+}
+
+// Return url of the avatar for the given url (typically for OpenID)
+func (v *Libravatar) FromURL(openid string) (string, error) {
+ ourl, err := url.Parse(openid)
+ if err != nil {
+ return "", err
+ }
+
+ if !ourl.IsAbs() {
+ return "", fmt.Errorf("Is not an absolute URL")
+ } else if ourl.Scheme != "http" && ourl.Scheme != "https" {
+ return "", fmt.Errorf("Invalid protocol: %s", ourl.Scheme)
+ }
+
+ link, err := v.process(nil, ourl)
+ if err != nil {
+ return "", err
+ }
+
+ return link, nil
+}
+
+// Object-less call to DefaultLibravatar for a URL
+func FromURL(openid string) (string, error) {
+ return DefaultLibravatar.FromURL(openid)
+}