-
Notifications
You must be signed in to change notification settings - Fork 9
Expand file tree
/
Copy pathshort.go
More file actions
195 lines (164 loc) · 3.9 KB
/
short.go
File metadata and controls
195 lines (164 loc) · 3.9 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
package main
/*
* Link Shortener, with a Redis backend.
*
* Released under and MIT License, please see the LICENSE.md file.
*
* John Nye
*
*/
import (
"github.com/johnnye/short/utils"
"github.com/garyburd/redigo/redis"
"encoding/json"
"io/ioutil"
"net/http"
"strings"
"flag"
"fmt"
"log"
)
var host = flag.String("h", "localhost", "Bind address to listen on")
var base = flag.String("b", "http://localhost/", "Base URL for the shortener")
var port = flag.String("p", "8080", "Port you want to listen on, defaults to 8080")
var maxConnections = flag.Int("c", 512, "The maximum number of active connections") //Currently Not Used
var redisConn = flag.String("r", "localhost:6379", "Redis Address, defaults to localhost:6379")
type Data struct {
Original string
Short string
FullShort string
HitCount int
}
var redisPool = &redis.Pool{
MaxIdle: 3,
MaxActive: 50, // max number of connections
Dial: func() (redis.Conn, error) {
c, err := redis.Dial("tcp", *redisConn)
if err != nil {
panic(err.Error())
}
return c, err
},
}
func handler(w http.ResponseWriter, r *http.Request) {
log.Println(r.UserAgent())
type NewURL struct {
URL string
}
var url NewURL
var domain Data
conn := redisPool.Get()
defer conn.Close()
if r.Method == "GET" {
domain, err := getLongURL(r.URL.Path[1:], conn)
if err !=nil {
log.Println(err)
}
if len(domain.Original) > 0 {
http.Redirect(w, r, domain.Original, http.StatusFound)
return
}
http.ServeFile(w, r, "./index.html")
log.Println("Served Homepage")
return
}
create, err := ioutil.ReadAll(r.Body)
if err != nil {
http.Error(w, err.Error(), 500)
return
}
err = json.Unmarshal(create, &url)
if err != nil {
log.Print(err)
http.Error(w, err.Error(), 500)
return
}
search := strings.Join([]string{"*||", url.URL}, "")
keys, err := redis.Strings(conn.Do("KEYS", search))
if len(keys) < 1 {
domain, err = createShortURL(url.URL, conn)
} else {
domain, err = getInfoForKey(keys[0], conn)
}
if err != nil {
log.Print(err)
http.Error(w, err.Error(), 500)
return
}
output, err := json.Marshal(domain)
if err != nil {
log.Print(err)
http.Error(w, err.Error(), 500)
return
}
fmt.Fprintf(w, "%s", output)
}
func getInfoForKey(key string, conn redis.Conn) (Data, error) {
var d Data
parts := strings.Split(key, "||")
d.Short = parts[0]
d.Original = parts[1]
d.FullShort = strings.Join([]string{*base, parts[0]}, "")
newCount, err := redis.Int(conn.Do("HGET", key, "count"))
if err != nil {
log.Print(err)
}
d.HitCount = newCount
return d, err
}
func createShortURL(url string, conn redis.Conn) (Data, error) {
var d Data
count, err := redis.Int(conn.Do("INCR", "global:size"))
if err != nil {
log.Print(err)
return d, err
}
log.Print("Total: ",count)
encodedVar := base62.EncodeInt(int64(count))
key := strings.Join([]string{encodedVar, url}, "||")
conn.Send("MULTI")
conn.Send("HSET", key, "count", 0)
_, err2 := conn.Do("EXEC")
if err2 != nil {
log.Print(err2)
return d, err2
}
d.Original = url
d.HitCount = 0
d.Short = encodedVar
d.FullShort = strings.Join([]string{*base, encodedVar}, "")
return d, err
}
func getLongURL(short string, conn redis.Conn) (Data, error) {
var d Data
search := strings.Join([]string{short, "||*"}, "")
fmt.Println(search)
n, err := redis.Strings(conn.Do("KEYS", search))
if err != nil {
log.Print(err)
return d, err
}
if len(n) < 1 {
log.Print("Nothing Found")
} else {
parts := strings.Split(n[0], "||")
d.Short = parts[0]
d.Original = parts[1]
d.FullShort = strings.Join([]string{*base, parts[0]}, "")
newCount, err := redis.Int(conn.Do("HINCRBY", n[0], "count", 1))
if err != nil {
log.Println(err)
}
d.HitCount = newCount
}
log.Println("Served: ",d.Original)
return d, nil
}
func main() {
flag.Parse()
http.HandleFunc("/", handler)
err := http.ListenAndServe(*host+":"+*port, nil)
if err != nil {
fmt.Println(err)
}
}