-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.go
178 lines (163 loc) · 4.94 KB
/
main.go
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
package main
import (
"encoding/json"
"log"
"net/http"
"os"
"time"
"github.com/gorilla/mux"
"github.com/speps/go-hashids"
"gopkg.in/mgo.v2"
"gopkg.in/mgo.v2/bson"
)
// Global reference
var endpoint string = "https://go-shorty.herokuapp.com"
var db *mgo.Database
var collection string
var port string
func main() {
// Connect to MongoDB
session, err := mgo.Dial(os.Getenv("MONGODB_URI"))
if err != nil {
log.Fatal(err)
}
db = session.DB(os.Getenv("DB"))
collection = os.Getenv("COLLECTION")
port = os.Getenv("PORT")
// Instatiate the Mux router
router := mux.NewRouter()
// Create routes
router.HandleFunc("/{id}", redirectHandler).Methods("GET")
router.HandleFunc("/shorten", shortenHandler).Methods("POST")
router.HandleFunc("/expand", expandHandler).Methods("POST")
// Serve static files
router.PathPrefix("/").Handler(http.FileServer(http.Dir("./public")))
// Attatch the router and start the server
log.Fatal(http.ListenAndServe(":"+port, router))
}
type myURL struct {
ID string `bson:"id" json:"id,omitempty"`
LongURL string `bson:"longURL" json:"longURL,omitempty"`
ShortURL string `bson:"shortURL" json:"shortURL,omitempty"`
}
/** Handler to shorten a long URL **/
func shortenHandler(res http.ResponseWriter, req *http.Request) {
var url myURL
// Extract JSON from body
err := json.NewDecoder(req.Body).Decode(&url)
if err != nil {
log.Fatal(err)
// Respond with an error
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusBadRequest)
res.Write([]byte(err.Error()))
return
}
log.Println("Received request to shorten URL:", url.LongURL)
// Validate if shortURL already exist
var result []myURL
// Run query
err = db.C(collection).Find(bson.M{"longURL": url.LongURL}).All(&result)
if err != nil {
log.Println("Error running query")
log.Fatal(err)
// Respond with an error
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusInternalServerError)
res.Write([]byte(err.Error()))
return
}
if len(result) == 0 {
// No results
log.Println("ShortURL doesn't exist for", url.LongURL, "creating")
// Encode a hashID
data := hashids.NewData()
hash, _ := hashids.NewWithData(data)
id, _ := hash.Encode([]int{int(time.Now().Unix())})
// Create the short URL
url.ID = id
url.ShortURL = endpoint + "/" + id
// Insert to DB
err = db.C("shorty").Insert(url)
if err != nil {
log.Fatal(err)
// Respond with an error
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusInternalServerError)
res.Write([]byte(err.Error()))
return
}
json.NewEncoder(res).Encode(url)
} else {
log.Println("ShortURL", result[0].ID, "already exists for", url.LongURL)
// Respond with the URL
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusOK)
json.NewEncoder(res).Encode(result[0])
}
}
/** Handler to expand a short URL **/
func expandHandler(res http.ResponseWriter, req *http.Request) {
var url myURL
// Extract JSON from body
err := json.NewDecoder(req.Body).Decode(&url)
if err != nil {
log.Fatal(err)
// Respond with an error
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusBadRequest)
res.Write([]byte(err.Error()))
return
}
log.Println("Received request to expand URL", url.ShortURL)
// Find the corresponding long URL
var result []myURL
// Run query
err = db.C(collection).Find(bson.M{"shortURL": url.ShortURL}).All(&result)
if err != nil {
log.Println("Error running query")
log.Fatal(err)
// Respond with an error
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusInternalServerError)
res.Write([]byte(err.Error()))
return
}
if len(result) == 0 {
// Not a valid short URL
log.Println("No longURL found for:", url.ShortURL)
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusOK)
json.NewEncoder(res).Encode(myURL{LongURL: "Invalid short URL, shorten first."})
} else {
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusOK)
json.NewEncoder(res).Encode(result[0])
}
}
/** Handler to redirect from a short URL **/
func redirectHandler(res http.ResponseWriter, req *http.Request) {
params := mux.Vars(req)
id := params["id"]
log.Println("Redirecting from:", id)
var result []myURL
err := db.C(collection).Find(bson.M{"id": id}).All(&result)
if err != nil {
log.Println("Error running query")
log.Fatal(err)
// Respond with an error
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusInternalServerError)
res.Write([]byte(err.Error()))
return
}
if len(result) == 0 {
log.Println("ShortURL not found for ID:", id)
res.Header().Set("Content-Type", "application/json")
res.WriteHeader(http.StatusOK)
json.NewEncoder(res).Encode("Impossible to redirect. URL not found.")
} else {
log.Println("Redirecting to:", result[0].LongURL)
http.Redirect(res, req, result[0].LongURL, 301)
}
}