mirror of https://github.com/gogits/gogs.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
552 lines
12 KiB
552 lines
12 KiB
// Copyright 2013 gopm authors. |
|
// |
|
// Licensed under the Apache License, Version 2.0 (the "License"): you may |
|
// not use this file except in compliance with the License. You may obtain |
|
// a copy of the License at |
|
// |
|
// http://www.apache.org/licenses/LICENSE-2.0 |
|
// |
|
// Unless required by applicable law or agreed to in writing, software |
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
|
// License for the specific language governing permissions and limitations |
|
// under the License. |
|
|
|
package cmd |
|
|
|
// import ( |
|
// "errors" |
|
// "fmt" |
|
// "github.com/Unknwon/com" |
|
// "github.com/gpmgo/gopm/doc" |
|
// "github.com/syndtr/goleveldb/leveldb" |
|
// "github.com/syndtr/goleveldb/leveldb/opt" |
|
// "io/ioutil" |
|
// "net/http" |
|
// "net/url" |
|
// "os" |
|
// "os/exec" |
|
// "path/filepath" |
|
// "strconv" |
|
// "strings" |
|
// "time" |
|
// ) |
|
|
|
// var ( |
|
// dbDir = "~/.gopm/db" |
|
// ) |
|
|
|
// const ( |
|
// STOP = iota |
|
// LOCALRUN |
|
// RUNNING |
|
// ) |
|
|
|
// var CmdServe = &Command{ |
|
// UsageLine: "serve [:port]", |
|
// Short: "serve for package search", |
|
// Long: ` |
|
// serve provide a web service to search packages, download packages |
|
|
|
// The serve flags are: |
|
|
|
// -l |
|
// only service for localhost ip |
|
// `, |
|
// } |
|
|
|
// func init() { |
|
// CmdServe.Run = runServe |
|
// CmdServe.Flags = map[string]bool{ |
|
// "-l": false, |
|
// } |
|
// } |
|
|
|
// func printServePrompt(flag string) { |
|
// switch flag { |
|
// case "-l": |
|
// com.ColorLog("[INFO] You enabled start a service only localhost.\n") |
|
// } |
|
// } |
|
|
|
// // Not implemented |
|
// func autoPort() string { |
|
// return "8991" |
|
// } |
|
|
|
// func exePath() (string, error) { |
|
// file, err := exec.LookPath(os.Args[0]) |
|
// if err != nil { |
|
// return "", err |
|
// } |
|
|
|
// return filepath.Abs(file) |
|
// } |
|
|
|
// // search packages |
|
// func runServe(cmd *Command, args []string) { |
|
// // Check flags. |
|
// num := checkFlags(cmd.Flags, args, printServePrompt) |
|
// if num == -1 { |
|
// return |
|
// } |
|
// args = args[num:] |
|
|
|
// var listen string |
|
// var port string |
|
// if cmd.Flags["-l"] { |
|
// listen += "127.0.0.1" |
|
// port = autoPort() |
|
// } else { |
|
// listen += "0.0.0.0" |
|
// port = "8991" |
|
// } |
|
|
|
// // Check length of arguments. |
|
// if len(args) >= 1 { |
|
// port = args[0] |
|
// } |
|
|
|
// err := startService(listen, port) |
|
// if err != nil { |
|
// com.ColorLog("[ERRO] %v\n", err) |
|
// } |
|
// } |
|
|
|
// func splitWord(word string, res *map[string]bool) { |
|
// for i, _ := range word { |
|
// for j, _ := range word[i:] { |
|
// w := word[i : i+j+1] |
|
// (*res)[w] = true |
|
// } |
|
// } |
|
// return |
|
// } |
|
|
|
// func splitPkgName(pkgName string) (res map[string]bool) { |
|
// //var src string |
|
// ps := strings.Split(pkgName, "/") |
|
// if len(ps) > 1 { |
|
// ps = ps[1:] |
|
// } |
|
|
|
// res = make(map[string]bool) |
|
// res[strings.Join(ps, "/")] = true |
|
// for _, w := range ps { |
|
// splitWord(w, &res) |
|
// } |
|
// return |
|
// } |
|
|
|
// func splitSynopsis(synopsis string) map[string]bool { |
|
// res := make(map[string]bool) |
|
// ss := strings.Fields(synopsis) |
|
// for _, s := range ss { |
|
// res[s] = true |
|
// } |
|
// return res |
|
// } |
|
|
|
// var ( |
|
// ro *opt.ReadOptions = &opt.ReadOptions{} |
|
// wo *opt.WriteOptions = &opt.WriteOptions{} |
|
// ) |
|
|
|
// func dbGet(key string) (string, error) { |
|
// v, err := db.Get([]byte(key), ro) |
|
// return string(v), err |
|
// } |
|
|
|
// func dbPut(key string, value string) error { |
|
// //fmt.Println("put ", key, ": ", value) |
|
// return db.Put([]byte(key), []byte(value), wo) |
|
// } |
|
|
|
// func batchPut(batch *leveldb.Batch, key string, value string) error { |
|
// //fmt.Println("put ", key, ": ", value) |
|
// batch.Put([]byte(key), []byte(value)) |
|
// return nil |
|
// } |
|
|
|
// func getServeHost() string { |
|
// return "localhost" |
|
// } |
|
|
|
// func getServePort() string { |
|
// return "8991" |
|
// } |
|
|
|
// // for exernal of serve to add node to db |
|
// func saveNode(nod *doc.Node) error { |
|
// urlPath := fmt.Sprintf("http://%v:%v/add", getServeHost(), getServePort()) |
|
// resp, err := http.PostForm(urlPath, |
|
// url.Values{"importPath": {nod.ImportPath}, |
|
// "synopsis": {nod.Synopsis}, |
|
// "downloadURL": {nod.DownloadURL}, |
|
// "isGetDeps": {strconv.FormatBool(nod.IsGetDeps)}, |
|
// "type": {nod.Type}, |
|
// "value": {nod.Value}}) |
|
|
|
// if err != nil { |
|
// com.ColorLog("[ERRO] Fail to save node[ %s ]\n", err) |
|
// return err |
|
// } |
|
// defer resp.Body.Close() |
|
|
|
// if resp.StatusCode == 200 { |
|
// return nil |
|
// } |
|
// return errors.New("save node failed with " + resp.Status) |
|
// } |
|
|
|
// // for inetrnal of serve to add node to db |
|
// func addNode(nod *doc.Node) error { |
|
// batch := new(leveldb.Batch) |
|
// strLastId, err := dbGet("lastId") |
|
// if err != nil { |
|
// if err == leveldb.ErrNotFound { |
|
// strLastId = "0" |
|
// err = batchPut(batch, "lastId", strLastId) |
|
// } else { |
|
// return err |
|
// } |
|
// } |
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// lastId, err := strconv.ParseInt(strLastId, 0, 64) |
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// nodKey := fmt.Sprintf("index:%v", nod.ImportPath) |
|
|
|
// id, err := dbGet(nodKey) |
|
// if err != nil { |
|
// if err == leveldb.ErrNotFound { |
|
// id = fmt.Sprintf("%v", lastId+1) |
|
// err = batchPut(batch, "lastId", id) |
|
// if err == nil { |
|
// err = batchPut(batch, nodKey, id) |
|
// } |
|
// if err == nil { |
|
// err = batchPut(batch, "pkg:"+id, nod.ImportPath) |
|
// } |
|
// if err == nil { |
|
// err = batchPut(batch, "desc:"+id, nod.Synopsis) |
|
// } |
|
// if err == nil { |
|
// err = batchPut(batch, "down:"+id, nod.DownloadURL) |
|
// } |
|
// if err == nil { |
|
// err = batchPut(batch, "deps:"+id, strconv.FormatBool(nod.IsGetDeps)) |
|
// } |
|
|
|
// // save totals |
|
// total, err := dbGet("total") |
|
// if err != nil { |
|
// if err == leveldb.ErrNotFound { |
|
// total = "1" |
|
// } else { |
|
// return err |
|
// } |
|
// } else { |
|
// totalInt, err := strconv.ParseInt(total, 0, 64) |
|
// if err != nil { |
|
// return err |
|
// } |
|
// totalInt = totalInt + 1 |
|
// total = fmt.Sprintf("%v", totalInt) |
|
// } |
|
|
|
// err = batchPut(batch, "total", total) |
|
// } else { |
|
// return err |
|
// } |
|
// } |
|
|
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// // save vers |
|
// vers, err := dbGet("ver:" + id) |
|
// needSplit := (err == leveldb.ErrNotFound) |
|
// if err != nil { |
|
// if err != leveldb.ErrNotFound { |
|
// return err |
|
// } |
|
// } else { |
|
// return nil |
|
// } |
|
|
|
// if vers == "" { |
|
// //fmt.Println(nod) |
|
// vers = nod.VerString() |
|
// } else { |
|
// if !strings.Contains(vers, nod.VerString()) { |
|
// vers = vers + "," + nod.VerString() |
|
// } else { |
|
// return nil |
|
// } |
|
// } |
|
|
|
// err = batchPut(batch, "ver:"+id, vers) |
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// if !needSplit { |
|
// return nil |
|
// } |
|
|
|
// // indexing package name |
|
// keys := splitPkgName(nod.ImportPath) |
|
// for key, _ := range keys { |
|
// err = batchPut(batch, fmt.Sprintf("key:%v:%v", strings.ToLower(key), id), "") |
|
// if err != nil { |
|
// return err |
|
// } |
|
// } |
|
|
|
// if nod.Synopsis != "" { |
|
// fields := splitSynopsis(nod.Synopsis) |
|
// for field, _ := range fields { |
|
// err = batchPut(batch, fmt.Sprintf("key:%v:%v", strings.ToLower(field), id), "") |
|
// if err != nil { |
|
// return err |
|
// } |
|
// } |
|
// } |
|
|
|
// return db.Write(batch, wo) |
|
// } |
|
|
|
// func rmPkg(nod *doc.Node) { |
|
|
|
// } |
|
|
|
// var db *leveldb.DB |
|
|
|
// // service should be run |
|
// func AutoRun() error { |
|
// s, _, _ := runningStatus() |
|
// if s == STOP { |
|
// // current path |
|
// curPath, err := os.Getwd() |
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// attr := &os.ProcAttr{ |
|
// Dir: curPath, |
|
// Env: os.Environ(), |
|
// //Files: []*os.File{nil, nil, nil}, |
|
// Files: []*os.File{os.Stdin, os.Stdout, os.Stderr}, |
|
// } |
|
|
|
// p, err := exePath() |
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// //com.ColorLog("[INFO] now is starting search daemon ...\n") |
|
// _, err = os.StartProcess(p, []string{"gopm", "serve", "-l"}, attr) |
|
// if err != nil { |
|
// return err |
|
// } |
|
// time.Sleep(time.Second) |
|
// } |
|
// return nil |
|
// } |
|
|
|
// func runningStatus() (int, int, int) { |
|
// pFile, err := getPidPath() |
|
// if err != nil { |
|
// return STOP, 0, 0 |
|
// } |
|
|
|
// contentByte, err := ioutil.ReadFile(pFile) |
|
// if err != nil { |
|
// return STOP, 0, 0 |
|
// } |
|
// content := string(contentByte) |
|
// if len(content) < 0 || !strings.Contains(content, ",") { |
|
// return STOP, 0, 0 |
|
// } |
|
// cs := strings.Split(string(content), ",") |
|
// if len(cs) != 3 { |
|
// return STOP, 0, 0 |
|
// } |
|
// status, err := strconv.Atoi(cs[0]) |
|
// if err != nil { |
|
// return STOP, 0, 0 |
|
// } |
|
// if status < STOP || status > RUNNING { |
|
// return STOP, 0, 0 |
|
// } |
|
// pid, err := strconv.Atoi(cs[1]) |
|
// if err != nil { |
|
// return STOP, 0, 0 |
|
// } |
|
|
|
// _, err = os.FindProcess(pid) |
|
// if err != nil { |
|
// return STOP, 0, 0 |
|
// } |
|
|
|
// port, err := strconv.Atoi(cs[2]) |
|
// if err != nil { |
|
// return STOP, 0, 0 |
|
// } |
|
|
|
// return status, pid, port |
|
// } |
|
|
|
// func getPidPath() (string, error) { |
|
// homeDir, err := com.HomeDir() |
|
// if err != nil { |
|
// return "", err |
|
// } |
|
|
|
// pFile := strings.Replace("~/.gopm/var/", "~", homeDir, -1) |
|
// os.MkdirAll(pFile, os.ModePerm) |
|
// return pFile + "pid", nil |
|
// } |
|
|
|
// func startService(listen, port string) error { |
|
// homeDir, err := com.HomeDir() |
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// pFile, err := getPidPath() |
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// f, err := os.OpenFile(pFile, os.O_RDWR|os.O_CREATE, 0700) |
|
// if err != nil { |
|
// return err |
|
// } |
|
// defer f.Close() |
|
// _, err = f.WriteString(fmt.Sprintf("%v,%v,%v", RUNNING, os.Getpid(), port)) |
|
// if err != nil { |
|
// return err |
|
// } |
|
|
|
// dbDir = strings.Replace(dbDir, "~", homeDir, -1) |
|
|
|
// db, err = leveldb.OpenFile(dbDir, nil) |
|
// if err != nil { |
|
// return err |
|
// } |
|
// defer db.Close() |
|
|
|
// // these handlers should only access by localhost |
|
// http.HandleFunc("/add", addHandler) |
|
// http.HandleFunc("/rm", rmHandler) |
|
|
|
// // these handlers can be accessed according listen's ip |
|
// http.HandleFunc("/search", searchHandler) |
|
// http.HandleFunc("/searche", searcheHandler) |
|
// http.ListenAndServe(listen+":"+port, nil) |
|
// return nil |
|
// } |
|
|
|
// func searchHandler(w http.ResponseWriter, r *http.Request) { |
|
// r.ParseForm() |
|
// ids := make(map[string]bool) |
|
// for key, _ := range r.Form { |
|
// iter := db.NewIterator(ro) |
|
// rkey := fmt.Sprintf("key:%v:", strings.ToLower(key)) |
|
// if iter.Seek([]byte(rkey)) { |
|
// k := iter.Key() |
|
// if !strings.HasPrefix(string(k), rkey) { |
|
// break |
|
// } else { |
|
// ids[string(k)] = true |
|
// } |
|
// } |
|
// for iter.Next() { |
|
// k := iter.Key() |
|
// if !strings.HasPrefix(string(k), rkey) { |
|
// break |
|
// } |
|
// ids[string(k)] = true |
|
// } |
|
// } |
|
|
|
// pkgs := make([]string, 0) |
|
|
|
// for id, _ := range ids { |
|
// idkeys := strings.SplitN(id, ":", -1) |
|
// rId := idkeys[len(idkeys)-1] |
|
// //fmt.Println(rId) |
|
// pkg, err := dbGet(fmt.Sprintf("pkg:%v", rId)) |
|
// if err != nil { |
|
// com.ColorLog(err.Error()) |
|
// continue |
|
// } |
|
// desc, err := dbGet(fmt.Sprintf("desc:%v", rId)) |
|
// if err != nil { |
|
// com.ColorLog(err.Error()) |
|
// continue |
|
// } |
|
// pkgs = append(pkgs, fmt.Sprintf(`{"pkg":"%v", "desc":"%v"}`, pkg, desc)) |
|
// } |
|
|
|
// w.Write([]byte("[" + strings.Join(pkgs, ", ") + "]")) |
|
// } |
|
|
|
// func searcheHandler(w http.ResponseWriter, r *http.Request) { |
|
// //if r.Method == "POST" { |
|
// r.ParseForm() |
|
// pkgs := make([]string, 0) |
|
// for key, _ := range r.Form { |
|
// rId, err := dbGet("index:" + key) |
|
// if err != nil { |
|
// com.ColorLog(err.Error()) |
|
// continue |
|
// } |
|
|
|
// desc, err := dbGet(fmt.Sprintf("desc:%v", rId)) |
|
// if err != nil { |
|
// com.ColorLog(err.Error()) |
|
// continue |
|
// } |
|
|
|
// pkgs = append(pkgs, fmt.Sprintf(`{"pkg":"%v", "desc":"%v"}`, key, desc)) |
|
// } |
|
|
|
// w.Write([]byte("[" + strings.Join(pkgs, ", ") + "]")) |
|
// //} |
|
// } |
|
|
|
// func addHandler(w http.ResponseWriter, r *http.Request) { |
|
// //if r.Method == "POST" { |
|
// r.ParseForm() |
|
|
|
// nod := new(doc.Node) |
|
// nod.ImportPath = r.FormValue("importPath") |
|
// nod.Synopsis = r.FormValue("synopsis") |
|
// nod.DownloadURL = r.FormValue("downloadURL") |
|
// isGetDeps, err := strconv.ParseBool(r.FormValue("isGetDeps")) |
|
// if err != nil { |
|
// com.ColorLog("[ERRO] SEVER: Cannot get deps") |
|
// } |
|
// nod.IsGetDeps = isGetDeps |
|
// nod.Type = r.FormValue("type") |
|
// nod.Value = r.FormValue("value") |
|
|
|
// err = addNode(nod) |
|
// if err != nil { |
|
// com.ColorLog("[ERRO] SEVER: Cannot add node[ %s ]\n", err) |
|
// } |
|
// //} |
|
// } |
|
|
|
// func rmHandler(w http.ResponseWriter, r *http.Request) { |
|
|
|
// }
|
|
|