package main // lingua evalia // // try it with `curl -i localhost:8000/run --data-binary @hello-world.go` import ( "crypto/rand" "flag" "fmt" "html/template" "io" "io/ioutil" "log" "math/big" "net/http" "os" "os/exec" "path" "strings" ) type Language interface { RunFile(f *os.File) (result []byte, err error) Name() string Extension() string } type LanguageGeneral struct { name string ext string command string args []string } func (l LanguageGeneral) RunFile(f *os.File) ([]byte, error) { args := append(l.args, f.Name()) cmd := exec.Command(l.command, args...) return cmd.CombinedOutput() } func (l LanguageGeneral) Name() string { return l.name } func (l LanguageGeneral) Extension() string { return l.ext } var Go = LanguageGeneral{"Go", "go", "go", []string{"run"}} var Python = LanguageGeneral{"Python", "py", "python", []string{}} var Ruby = LanguageGeneral{"Ruby", "rb", "ruby", []string{}} var JavaScript = LanguageGeneral{"JavaScript", "js", "node", []string{}} var Haskell = LanguageGeneral{"Haskell", "hs", "runhaskell", []string{}} var Rust = LanguageGeneral{"Rust", "rs", "./bin/run-rust", []string{}} var Julia = LanguageGeneral{"Julia", "jl", "julia", []string{}} var Pixie = LanguageGeneral{"Pixie", "pxi", "pixie-vm", []string{}} var C = LanguageGeneral{"C", "c", "./bin/run-c", []string{}} var Bash = LanguageGeneral{"Bash", "bash", "bash", []string{}} var Lua = LanguageGeneral{"Lua", "lua", "lua", []string{}} var languageMappings = map[string]Language{ "go": Go, "python": Python, "ruby": Ruby, "javascript": JavaScript, "haskell": Haskell, "rust": Rust, "julia": Julia, "pixie": Pixie, "c": C, "bash": Bash, "lua": Lua, } func writeCode(code string, extension string) (*os.File, error) { // create tmp file f, err := tempFile("/tmp", "linguaevalia", extension) if err != nil { return f, err } // write code to it _, err = f.Write([]byte(code)) if err != nil { return f, err } return f, nil } func tempFile(dir, prefix, suffix string) (*os.File, error) { rnd, _ := rand.Int(rand.Reader, big.NewInt(999999)) f, err := os.Create(path.Join(dir, fmt.Sprintf("%s%d.%s", prefix, rnd, suffix))) return f, err } func Eval(lang Language, code string) ([]byte, error) { // write code to temp file f, err := writeCode(code, lang.Extension()) defer f.Close() defer os.Remove(f.Name()) if err != nil { return nil, err } // `go run` it res, err := lang.RunFile(f) if err != nil { return res, err } // return output return res, nil } func runCodeHandler(w http.ResponseWriter, r *http.Request) { code, err := ioutil.ReadAll(r.Body) if err != nil { http.Error(w, err.Error(), http.StatusBadRequest) return } lang := getLanguage(r) res, err := Eval(lang, string(code)) if err != nil { http.Error(w, string(res), http.StatusNotAcceptable) return } w.Write(res) } func getLanguage(r *http.Request) Language { langName := r.URL.Query().Get("language") if langName != "" { lang, ok := languageMappings[langName] if ok { return lang } } return Go } func homePageHandler(w http.ResponseWriter, r *http.Request) { bindings := map[string]interface{}{ "languages": languageMappings, } homePageTemplate.Execute(w, bindings) } var homePageTemplate = template.Must(template.New("homepage").Parse(homePageTemplateStr)) func runServer() { addr, port := "localhost", 8000 fmt.Printf("running on %s:%d\n", addr, port) http.HandleFunc("/run", runCodeHandler) http.HandleFunc("/codemirror.js", func(w http.ResponseWriter, r *http.Request) { http.ServeFile(w, r, "lib/codemirror.js") }) http.HandleFunc("/codemirror.css", func(w http.ResponseWriter, r *http.Request) { http.ServeFile(w, r, "lib/codemirror.css") }) http.HandleFunc("/", homePageHandler) err := http.ListenAndServe(fmt.Sprintf("%s:%d", addr, port), nil) if err != nil { log.Fatal(err) } } func languageForExtension(extension string) *Language { var language *Language = nil for _, lang := range languageMappings { if "."+lang.Extension() == extension { return &lang } } return language } func runOnce(args []string) { var ( f *os.File err error langName string = *language ) if len(args) > 0 { if *language == "" { l := languageForExtension(path.Ext(args[0])) if l == nil { fmt.Printf("Error: Don't know how to handle '%s' files\n", path.Ext(args[0])) os.Exit(1) } langName = (*l).Name() } f, err = os.Open(args[0]) } else { f, err = os.Stdin, nil } defer f.Close() if err != nil { fmt.Println("Error: ", err) os.Exit(1) } langName = strings.ToLower(langName) lang, ok := languageMappings[langName] if !ok { fmt.Printf("Error: Unknown language '%s'\n", langName) os.Exit(1) } if f == os.Stdin { f, err = tempFile("/tmp", "linguaevalia", lang.Extension()) _, err = io.Copy(f, os.Stdin) if err != nil { fmt.Printf("Error: ", err) os.Exit(1) } defer os.Remove(f.Name()) } res, err := lang.RunFile(f) os.Stdout.Write(res) if err != nil { os.Exit(1) } } func parseCommand() (string, []string) { if len(os.Args) == 1 { return "server", []string{} } else { return os.Args[1], os.Args[2:] } } var language = flag.String("l", "", "The language to use for code passed via stdin.") func main() { cmd, args := parseCommand() flag.CommandLine.Parse(args) switch cmd { case "server": runServer() case "run": runOnce(flag.Args()) default: fmt.Println("Error: Unknown command:", cmd) os.Exit(1) } } const homePageTemplateStr = `