2014-04-09 18:59:44 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"crypto/md5"
|
2014-04-14 14:54:07 +00:00
|
|
|
"database/sql"
|
2014-04-09 18:59:44 +00:00
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
2014-04-14 14:54:07 +00:00
|
|
|
_ "github.com/go-sql-driver/mysql"
|
2014-04-14 18:53:10 +00:00
|
|
|
_ "github.com/mattn/go-sqlite3"
|
|
|
|
htemplate "html/template"
|
2014-04-09 18:59:44 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"path/filepath"
|
2014-04-14 18:53:10 +00:00
|
|
|
"regexp"
|
2014-04-09 18:59:44 +00:00
|
|
|
"strings"
|
|
|
|
"text/template"
|
2014-04-14 20:12:08 +00:00
|
|
|
"time"
|
2014-04-09 18:59:44 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
RESULT_COMPILE = `c++ -DSK_GAMMA_SRGB -DSK_GAMMA_APPLY_TO_A8 -DSK_SCALAR_TO_FLOAT_EXCLUDED -DSK_ALLOW_STATIC_GLOBAL_INITIALIZERS=1 -DSK_SUPPORT_GPU=0 -DSK_SUPPORT_OPENCL=0 -DSK_FORCE_DISTANCEFIELD_FONTS=0 -DSK_SCALAR_IS_FLOAT -DSK_CAN_USE_FLOAT -DSK_SAMPLES_FOR_X -DSK_BUILD_FOR_UNIX -DSK_USE_POSIX_THREADS -DSK_SYSTEM_ZLIB=1 -DSK_DEBUG -DSK_DEVELOPER=1 -I../../src/core -I../../src/images -I../../tools/flags -I../../include/config -I../../include/core -I../../include/pathops -I../../include/pipe -I../../include/effects -I../../include/ports -I../../src/sfnt -I../../include/utils -I../../src/utils -I../../include/images -g -fno-exceptions -fstrict-aliasing -Wall -Wextra -Winit-self -Wpointer-arith -Wno-unused-parameter -Wno-c++11-extensions -Werror -m64 -fno-rtti -Wnon-virtual-dtor -c ../../../cache/%s.cpp -o ../../../cache/%s.o`
|
2014-04-09 21:35:18 +00:00
|
|
|
LINK = `c++ -m64 -lstdc++ -lm -o ../../../inout/%s -Wl,--start-group ../../../cache/%s.o obj/experimental/webtry/webtry.main.o obj/gyp/libflags.a libskia_images.a libskia_core.a libskia_effects.a obj/gyp/libjpeg.a obj/gyp/libwebp_dec.a obj/gyp/libwebp_demux.a obj/gyp/libwebp_dsp.a obj/gyp/libwebp_enc.a obj/gyp/libwebp_utils.a libskia_utils.a libskia_opts.a libskia_opts_ssse3.a libskia_ports.a libskia_sfnt.a -Wl,--end-group -lpng -lz -lgif -lpthread -lfontconfig -ldl -lfreetype`
|
2014-04-14 18:53:10 +00:00
|
|
|
DEFAULT_SAMPLE = `SkPaint p;
|
|
|
|
p.setColor(SK_ColorRED);
|
|
|
|
p.setAntiAlias(true);
|
|
|
|
p.setStyle(SkPaint::kStroke_Style);
|
|
|
|
p.setStrokeWidth(10);
|
|
|
|
|
|
|
|
canvas->drawLine(20, 20, 100, 100, p);
|
|
|
|
`
|
2014-04-15 00:43:51 +00:00
|
|
|
// Don't increase above 2^16 w/o altering the db tables to accept something bigger than TEXT.
|
|
|
|
MAX_TRY_SIZE = 64000
|
2014-04-09 18:59:44 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// codeTemplate is the cpp code template the user's code is copied into.
|
|
|
|
codeTemplate *template.Template = nil
|
|
|
|
|
2014-04-14 20:12:08 +00:00
|
|
|
// indexTemplate is the main index.html page we serve.
|
|
|
|
indexTemplate *htemplate.Template = nil
|
|
|
|
|
|
|
|
// recentTemplate is a list of recent images.
|
|
|
|
recentTemplate *htemplate.Template = nil
|
2014-04-14 14:54:07 +00:00
|
|
|
|
|
|
|
// db is the database, nil if we don't have an SQL database to store data into.
|
|
|
|
db *sql.DB = nil
|
2014-04-14 18:53:10 +00:00
|
|
|
|
|
|
|
// directLink is the regex that matches URLs paths that are direct links.
|
2014-04-14 20:12:08 +00:00
|
|
|
directLink = regexp.MustCompile("^/c/([a-f0-9]+)$")
|
|
|
|
|
|
|
|
// imageLink is the regex that matches URLs paths that are direct links to PNGs.
|
|
|
|
imageLink = regexp.MustCompile("^/i/([a-f0-9]+.png)$")
|
2014-04-09 18:59:44 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// flags
|
|
|
|
var (
|
|
|
|
useChroot = flag.Bool("use_chroot", false, "Run the compiled code in the schroot jail.")
|
2014-04-14 14:54:07 +00:00
|
|
|
port = flag.String("port", ":8000", "HTTP service address (e.g., ':8000')")
|
2014-04-09 18:59:44 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// lineNumbers adds #line numbering to the user's code.
|
|
|
|
func LineNumbers(c string) string {
|
|
|
|
lines := strings.Split(c, "\n")
|
|
|
|
ret := []string{}
|
|
|
|
for i, line := range lines {
|
|
|
|
ret = append(ret, fmt.Sprintf("#line %d", i+1))
|
|
|
|
ret = append(ret, line)
|
|
|
|
}
|
|
|
|
return strings.Join(ret, "\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
2014-04-14 18:53:10 +00:00
|
|
|
|
2014-04-09 18:59:44 +00:00
|
|
|
// Change the current working directory to the directory of the executable.
|
|
|
|
var err error
|
|
|
|
cwd, err := filepath.Abs(filepath.Dir(os.Args[0]))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
os.Chdir(cwd)
|
|
|
|
|
|
|
|
codeTemplate, err = template.ParseFiles(filepath.Join(cwd, "templates/template.cpp"))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2014-04-14 18:53:10 +00:00
|
|
|
// Convert index.html into a template, which is expanded with the code.
|
2014-04-14 20:12:08 +00:00
|
|
|
indexTemplate, err = htemplate.ParseFiles(filepath.Join(cwd, "templates/index.html"))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
recentTemplate, err = htemplate.ParseFiles(filepath.Join(cwd, "templates/recent.html"))
|
2014-04-09 18:59:44 +00:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2014-04-14 14:54:07 +00:00
|
|
|
|
|
|
|
// Connect to MySQL server. First, get the password from the metadata server.
|
|
|
|
// See https://developers.google.com/compute/docs/metadata#custom.
|
|
|
|
req, err := http.NewRequest("GET", "http://metadata/computeMetadata/v1/instance/attributes/password", nil)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
client := http.Client{}
|
|
|
|
req.Header.Add("X-Google-Metadata-Request", "True")
|
|
|
|
if resp, err := client.Do(req); err == nil {
|
|
|
|
password, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("ERROR: Failed to read password from metadata server: %q\n", err)
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
// The IP address of the database is found here:
|
|
|
|
// https://console.developers.google.com/project/31977622648/sql/instances/webtry/overview
|
|
|
|
// And 3306 is the default port for MySQL.
|
2014-04-15 00:43:51 +00:00
|
|
|
db, err = sql.Open("mysql", fmt.Sprintf("webtry:%s@tcp(173.194.83.52:3306)/webtry?parseTime=true", password))
|
2014-04-14 14:54:07 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("ERROR: Failed to open connection to SQL server: %q\n", err)
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
} else {
|
2014-04-14 18:53:10 +00:00
|
|
|
// Fallback to sqlite for local use.
|
|
|
|
db, err = sql.Open("sqlite3", "./webtry.db")
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("ERROR: Failed to open: %q\n", err)
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
sql := `CREATE TABLE webtry (
|
|
|
|
code TEXT DEFAULT '' NOT NULL,
|
|
|
|
create_ts TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
|
|
|
|
hash CHAR(64) DEFAULT '' NOT NULL,
|
|
|
|
PRIMARY KEY(hash)
|
|
|
|
)`
|
|
|
|
db.Exec(sql)
|
2014-04-14 14:54:07 +00:00
|
|
|
log.Printf("INFO: Failed to find metadata, unable to connect to MySQL server (Expected when running locally): %q\n", err)
|
|
|
|
}
|
2014-04-09 18:59:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// userCode is used in template expansion.
|
|
|
|
type userCode struct {
|
|
|
|
UserCode string
|
|
|
|
}
|
|
|
|
|
|
|
|
// expandToFile expands the template and writes the result to the file.
|
|
|
|
func expandToFile(filename string, code string, t *template.Template) error {
|
|
|
|
f, err := os.Create(filename)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
2014-04-14 18:53:10 +00:00
|
|
|
return t.Execute(f, userCode{UserCode: code})
|
2014-04-09 18:59:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// expandCode expands the template into a file and calculate the MD5 hash.
|
|
|
|
func expandCode(code string) (string, error) {
|
|
|
|
h := md5.New()
|
|
|
|
h.Write([]byte(code))
|
|
|
|
hash := fmt.Sprintf("%x", h.Sum(nil))
|
|
|
|
// At this point we are running in skia/experimental/webtry, making cache a
|
|
|
|
// peer directory to skia.
|
|
|
|
// TODO(jcgregorio) Make all relative directories into flags.
|
|
|
|
err := expandToFile(fmt.Sprintf("../../../cache/%s.cpp", hash), code, codeTemplate)
|
|
|
|
return hash, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// response is serialized to JSON as a response to POSTs.
|
|
|
|
type response struct {
|
|
|
|
Message string `json:"message"`
|
|
|
|
Img string `json:"img"`
|
2014-04-14 18:53:10 +00:00
|
|
|
Hash string `json:"hash"`
|
2014-04-09 18:59:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// doCmd executes the given command line string in either the out/Debug
|
|
|
|
// directory or the inout directory. Returns the stdout, and stderr in the case
|
|
|
|
// of a non-zero exit code.
|
|
|
|
func doCmd(commandLine string, moveToDebug bool) (string, error) {
|
|
|
|
log.Printf("Command: %q\n", commandLine)
|
|
|
|
programAndArgs := strings.SplitN(commandLine, " ", 2)
|
|
|
|
program := programAndArgs[0]
|
|
|
|
args := []string{}
|
|
|
|
if len(programAndArgs) > 1 {
|
|
|
|
args = strings.Split(programAndArgs[1], " ")
|
|
|
|
}
|
|
|
|
cmd := exec.Command(program, args...)
|
|
|
|
abs, err := filepath.Abs("../../out/Debug")
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("Failed to find absolute path to Debug directory.")
|
|
|
|
}
|
|
|
|
if moveToDebug {
|
|
|
|
cmd.Dir = abs
|
|
|
|
} else if !*useChroot { // Don't set cmd.Dir when using chroot.
|
|
|
|
abs, err := filepath.Abs("../../../inout")
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("Failed to find absolute path to inout directory.")
|
|
|
|
}
|
|
|
|
cmd.Dir = abs
|
|
|
|
}
|
|
|
|
log.Printf("Run in directory: %q\n", cmd.Dir)
|
|
|
|
var stdOut bytes.Buffer
|
|
|
|
cmd.Stdout = &stdOut
|
|
|
|
var stdErr bytes.Buffer
|
|
|
|
cmd.Stderr = &stdErr
|
|
|
|
cmd.Start()
|
|
|
|
err = cmd.Wait()
|
|
|
|
message := stdOut.String()
|
|
|
|
log.Printf("StdOut: %s\n", message)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("Exit status: %s\n", err.Error())
|
|
|
|
log.Printf("StdErr: %s\n", stdErr.String())
|
|
|
|
message += stdErr.String()
|
|
|
|
return message, fmt.Errorf("Failed to run command.")
|
|
|
|
}
|
|
|
|
return message, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// reportError formats an HTTP error response and also logs the detailed error message.
|
|
|
|
func reportError(w http.ResponseWriter, r *http.Request, err error, message string) {
|
|
|
|
m := response{
|
|
|
|
Message: message,
|
|
|
|
}
|
|
|
|
log.Printf("Error: %s\n%s", message, err.Error())
|
|
|
|
resp, err := json.Marshal(m)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "Failed to serialize a response", 500)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
w.Write(resp)
|
|
|
|
}
|
|
|
|
|
2014-04-14 14:54:07 +00:00
|
|
|
func writeToDatabase(hash string, code string) {
|
|
|
|
if db == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if _, err := db.Exec("INSERT INTO webtry (code, hash) VALUES(?, ?)", code, hash); err != nil {
|
|
|
|
log.Printf("ERROR: Failed to insert code into database: %q\n", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-14 20:12:08 +00:00
|
|
|
func cssHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
http.ServeFile(w, r, "css/webtry.css")
|
|
|
|
}
|
|
|
|
|
|
|
|
// imageHandler serves up the PNG of a specific try.
|
|
|
|
func imageHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
log.Printf("Image Handler: %q\n", r.URL.Path)
|
|
|
|
if r.Method != "GET" {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
match := imageLink.FindStringSubmatch(r.URL.Path)
|
|
|
|
if len(match) != 2 {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
filename := match[1]
|
|
|
|
http.ServeFile(w, r, fmt.Sprintf("../../../inout/%s", filename))
|
|
|
|
}
|
|
|
|
|
|
|
|
type Try struct {
|
|
|
|
Hash string
|
|
|
|
CreateTS string
|
|
|
|
}
|
|
|
|
|
|
|
|
type Recent struct {
|
|
|
|
Tries []Try
|
|
|
|
}
|
|
|
|
|
|
|
|
// recentHandler shows the last 20 tries.
|
|
|
|
func recentHandler(w http.ResponseWriter, r *http.Request) {
|
|
|
|
log.Printf("Recent Handler: %q\n", r.URL.Path)
|
|
|
|
|
|
|
|
var err error
|
|
|
|
rows, err := db.Query("SELECT create_ts, hash FROM webtry ORDER BY create_ts DESC LIMIT 20")
|
|
|
|
if err != nil {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
recent := []Try{}
|
|
|
|
for rows.Next() {
|
|
|
|
var hash string
|
|
|
|
var create_ts time.Time
|
|
|
|
if err := rows.Scan(&create_ts, &hash); err != nil {
|
|
|
|
log.Printf("Error: failed to fetch from database: %q", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
recent = append(recent, Try{Hash: hash, CreateTS: create_ts.Format("2006-02-01")})
|
|
|
|
}
|
|
|
|
if err := recentTemplate.Execute(w, Recent{Tries: recent}); err != nil {
|
|
|
|
log.Printf("ERROR: Failed to expand template: %q\n", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-15 00:43:51 +00:00
|
|
|
// hasPreProcessor returns true if any line in the code begins with a # char.
|
|
|
|
func hasPreProcessor(code string) bool {
|
|
|
|
lines := strings.Split(code, "\n")
|
|
|
|
for _, s := range lines {
|
|
|
|
if strings.HasPrefix(strings.TrimSpace(s), "#") {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2014-04-09 18:59:44 +00:00
|
|
|
// mainHandler handles the GET and POST of the main page.
|
|
|
|
func mainHandler(w http.ResponseWriter, r *http.Request) {
|
2014-04-14 20:12:08 +00:00
|
|
|
log.Printf("Main Handler: %q\n", r.URL.Path)
|
2014-04-09 18:59:44 +00:00
|
|
|
if r.Method == "GET" {
|
2014-04-14 18:53:10 +00:00
|
|
|
code := DEFAULT_SAMPLE
|
|
|
|
match := directLink.FindStringSubmatch(r.URL.Path)
|
2014-04-14 20:12:08 +00:00
|
|
|
if len(match) == 2 && r.URL.Path != "/" {
|
2014-04-14 18:53:10 +00:00
|
|
|
hash := match[1]
|
|
|
|
if db == nil {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Update 'code' with the code found in the database.
|
|
|
|
if err := db.QueryRow("SELECT code FROM webtry WHERE hash=?", hash).Scan(&code); err != nil {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Expand the template.
|
2014-04-14 20:12:08 +00:00
|
|
|
if err := indexTemplate.Execute(w, userCode{UserCode: code}); err != nil {
|
2014-04-14 18:53:10 +00:00
|
|
|
log.Printf("ERROR: Failed to expand template: %q\n", err)
|
|
|
|
}
|
2014-04-09 18:59:44 +00:00
|
|
|
} else if r.Method == "POST" {
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
2014-04-15 00:43:51 +00:00
|
|
|
buf := bytes.NewBuffer(make([]byte, 0, MAX_TRY_SIZE))
|
|
|
|
n, err := buf.ReadFrom(r.Body)
|
2014-04-09 18:59:44 +00:00
|
|
|
if err != nil {
|
|
|
|
reportError(w, r, err, "Failed to read a request body.")
|
|
|
|
return
|
|
|
|
}
|
2014-04-15 00:43:51 +00:00
|
|
|
if n == MAX_TRY_SIZE {
|
|
|
|
err := fmt.Errorf("Code length equal to, or exceeded, %d", MAX_TRY_SIZE)
|
|
|
|
reportError(w, r, err, "Code too large.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
code := string(buf.Bytes())
|
|
|
|
if hasPreProcessor(code) {
|
|
|
|
err := fmt.Errorf("Found preprocessor macro in code.")
|
|
|
|
reportError(w, r, err, "Preprocessor macros aren't allowed.")
|
|
|
|
return
|
|
|
|
}
|
2014-04-14 14:54:07 +00:00
|
|
|
hash, err := expandCode(LineNumbers(code))
|
2014-04-09 18:59:44 +00:00
|
|
|
if err != nil {
|
|
|
|
reportError(w, r, err, "Failed to write the code to compile.")
|
|
|
|
return
|
|
|
|
}
|
2014-04-14 14:54:07 +00:00
|
|
|
writeToDatabase(hash, code)
|
2014-04-09 18:59:44 +00:00
|
|
|
message, err := doCmd(fmt.Sprintf(RESULT_COMPILE, hash, hash), true)
|
|
|
|
if err != nil {
|
|
|
|
reportError(w, r, err, "Failed to compile the code:\n"+message)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
linkMessage, err := doCmd(fmt.Sprintf(LINK, hash, hash), true)
|
|
|
|
if err != nil {
|
|
|
|
reportError(w, r, err, "Failed to link the code:\n"+linkMessage)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
message += linkMessage
|
|
|
|
cmd := hash + " --out " + hash + ".png"
|
|
|
|
if *useChroot {
|
|
|
|
cmd = "schroot -c webtry --directory=/inout -- /inout/" + cmd
|
|
|
|
} else {
|
|
|
|
abs, err := filepath.Abs("../../../inout")
|
|
|
|
if err != nil {
|
|
|
|
reportError(w, r, err, "Failed to find executable directory.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
cmd = abs + "/" + cmd
|
|
|
|
}
|
|
|
|
|
|
|
|
execMessage, err := doCmd(cmd, false)
|
|
|
|
if err != nil {
|
|
|
|
reportError(w, r, err, "Failed to run the code:\n"+execMessage)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
png, err := ioutil.ReadFile("../../../inout/" + hash + ".png")
|
|
|
|
if err != nil {
|
|
|
|
reportError(w, r, err, "Failed to open the generated PNG.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
m := response{
|
|
|
|
Message: message,
|
|
|
|
Img: base64.StdEncoding.EncodeToString([]byte(png)),
|
2014-04-14 18:53:10 +00:00
|
|
|
Hash: hash,
|
2014-04-09 18:59:44 +00:00
|
|
|
}
|
|
|
|
resp, err := json.Marshal(m)
|
|
|
|
if err != nil {
|
|
|
|
reportError(w, r, err, "Failed to serialize a response.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
w.Write(resp)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
flag.Parse()
|
2014-04-14 20:12:08 +00:00
|
|
|
http.HandleFunc("/i/", imageHandler)
|
|
|
|
http.HandleFunc("/recent/", recentHandler)
|
|
|
|
http.HandleFunc("/css/", cssHandler)
|
2014-04-09 18:59:44 +00:00
|
|
|
http.HandleFunc("/", mainHandler)
|
2014-04-14 14:54:07 +00:00
|
|
|
log.Fatal(http.ListenAndServe(*port, nil))
|
2014-04-09 18:59:44 +00:00
|
|
|
}
|