// Package main shows how you can create a simple URL SHortener using only Iris and BoltDB. // // $ go get github.com/boltdb/bolt/... // $ go run main.go // $ start http://localhost:8080 package main import ( "bytes" "html/template" "math/rand" "net/url" "time" "github.com/boltdb/bolt" "gopkg.in/kataras/iris.v6" "gopkg.in/kataras/iris.v6/adaptors/httprouter" "gopkg.in/kataras/iris.v6/adaptors/view" ) // a custom Iris event policy, which will run when server interruped (i.e control+C) // receives a func() error, most of packages are compatible with that on their Close/Shutdown/Cancel funcs. func releaser(r func() error) iris.EventPolicy { return iris.EventPolicy{ Interrupted: func(app *iris.Framework) { if err := r(); err != nil { app.Log(iris.ProdMode, "error while releasing resources: "+err.Error()) } }} } func main() { app := iris.New() // assign a variable to the DB so we can use its features later db := NewDB("shortener.db") factory := NewFactory(DefaultGenerator, db) app.Adapt( // print all kind of errors and logs at os.Stdout iris.DevLogger(), // use the httprouter, you can use adpaotrs/gorillamux if you want httprouter.New(), // serve the "./templates" directory's "*.html" files with the HTML std view engine. view.HTML("./templates", ".html").Reload(true), // `db.Close` is a `func() error` so it can be a `releaser` too. // Wrap the db.Close with the releaser in order to be released when app exits or control+C // You probably never saw that before, clever pattern which I am able to use only with Iris :) releaser(db.Close), ) // template funcs // // look ./templates/index.html#L16 app.Adapt(iris.TemplateFuncsPolicy{"isPositive": func(n int) bool { if n > 0 { return true } return false }}) // Serve static files (css) app.StaticWeb("/static", "./resources") app.Get("/", func(ctx *iris.Context) { ctx.MustRender("index.html", iris.Map{"url_count": db.Len()}) }) // find and execute a short url by its key // used on http://localhost:8080/u/dsaoj41u321dsa execShortURL := func(ctx *iris.Context, key string) { if key == "" { ctx.EmitError(iris.StatusBadRequest) return } value := db.Get(key) if value == "" { ctx.SetStatusCode(iris.StatusNotFound) ctx.Writef("Short URL for key: '%s' not found", key) return } ctx.Redirect(value, iris.StatusTemporaryRedirect) } app.Get("/u/:shortkey", func(ctx *iris.Context) { execShortURL(ctx, ctx.Param("shortkey")) }) app.Post("/shorten", func(ctx *iris.Context) { data := make(map[string]interface{}, 0) formValue := ctx.FormValue("url") if formValue == "" { data["form_result"] = "You need to a enter a URL." } else { key, err := factory.Gen(formValue) if err != nil { data["form_result"] = "Invalid URL." } else { if err = db.Set(key, formValue); err != nil { data["form_result"] = "Internal error while saving the url" app.Log(iris.DevMode, "while saving url: "+err.Error()) } else { ctx.SetStatusCode(iris.StatusOK) shortenURL := "http://" + app.Config.VHost + "/u/" + key data["form_result"] = template.HTML("
" + shortenURL + "") } } } data["url_count"] = db.Len() ctx.Render("index.html", data) }) app.Listen("localhost:8080") } // +------------------------------------------------------------+ // | | // | Store | // | | // +------------------------------------------------------------+ // Panic panics, change it if you don't want to panic on critical INITIALIZE-ONLY-ERRORS var Panic = func(v interface{}) { panic(v) } // Store is the store interface for urls. // Note: no Del functionality. type Store interface { Set(key string, value string) error // error if something went wrong Get(key string) string // empty value if not found Len() int // should return the number of all the records/tables/buckets Close() error // release the store or ignore } var ( tableURLs = []byte("urls") ) // DB representation of a Store. // Only one table/bucket which contains the urls, so it's not a fully Database, // it works only with single bucket because that all we need. type DB struct { db *bolt.DB } var _ Store = &DB{} // openDatabase open a new database connection // and returns its instance. func openDatabase(stumb string) *bolt.DB { // Open the data(base) file in the current working directory. // It will be created if it doesn't exist. db, err := bolt.Open(stumb, 0600, nil) if err != nil { Panic(err) } // create the buckets here var tables = [...][]byte{ tableURLs, } db.Update(func(tx *bolt.Tx) (err error) { for _, table := range tables { _, err = tx.CreateBucketIfNotExists(table) if err != nil { Panic(err) } } return }) return db } // NewDB returns a new DB instance, its connection is opened. // DB implements the Store. func NewDB(stumb string) *DB { return &DB{ db: openDatabase(stumb), } } // Set sets a shorten url and its key // Note: Caller is responsible to generate a key. func (d *DB) Set(key string, value string) error { d.db.Update(func(tx *bolt.Tx) error { b := tx.Bucket(tableURLs) // Generate ID for the url // Note: we could use that instead of a random string key // but we want to simulate a real-world url shortener // so we skip that. // id, _ := b.NextSequence() return b.Put([]byte(key), []byte(value)) }) return nil } // Get returns a url by its key. // // Returns an empty string if not found. func (d *DB) Get(key string) (value string) { keyb := []byte(key) d.db.View(func(tx *bolt.Tx) error { b := tx.Bucket(tableURLs) c := b.Cursor() for k, v := c.First(); k != nil; k, v = c.Next() { if bytes.Equal(keyb, k) { value = string(v) break } } return nil }) return } // Len returns all the "shorted" urls length func (d *DB) Len() (num int) { d.db.View(func(tx *bolt.Tx) error { // Assume bucket exists and has keys b := tx.Bucket(tableURLs) b.ForEach(func([]byte, []byte) error { num++ return nil }) return nil }) return } // Close the data(base) connection func (d *DB) Close() error { return d.db.Close() } // +------------------------------------------------------------+ // | | // | Factory | // | | // +------------------------------------------------------------+ // Generator the type to generate keys(short urls) based on 'n' type Generator func(n int) string // DefaultGenerator is the defautl url generator (the simple randomString) var DefaultGenerator = randomString // Factory is responsible to generate keys(short urls) type Factory struct { store Store generator Generator } // NewFactory receives a generator and a store and returns a new url Factory. func NewFactory(generator Generator, store Store) *Factory { return &Factory{ store: store, generator: generator, } } // Gen generates the key. func (f *Factory) Gen(uri string) (key string, err error) { // we don't return the parsed url because #hash are converted to uri-compatible // and we don't want to encode/decode all the time, there is no need for that, // we save the url as the user expects if the uri validation passed. _, err = url.ParseRequestURI(uri) if err != nil { return "", err } key = f.generator(len(uri)) // Make sure that the key is unique for { if v := f.store.Get(key); v == "" { break } key = f.generator((len(uri) / 2) + 1) } return key, nil } const ( letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" letterIdxBits = 6 // 6 bits to represent a letter index letterIdxMask = 1<