mirror of https://github.com/gogits/gogs.git
Lunny Xiao
12 years ago
5 changed files with 542 additions and 20 deletions
@ -0,0 +1,113 @@ |
|||||||
|
// 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 ( |
||||||
|
"../doc" |
||||||
|
"encoding/json" |
||||||
|
"fmt" |
||||||
|
"io/ioutil" |
||||||
|
"net/http" |
||||||
|
) |
||||||
|
|
||||||
|
var CmdSearch = &Command{ |
||||||
|
UsageLine: "search [keyword]", |
||||||
|
Short: "search for package", |
||||||
|
Long: ` |
||||||
|
search packages |
||||||
|
|
||||||
|
The search flags are: |
||||||
|
|
||||||
|
-s |
||||||
|
start a search service. This must be run before search a package |
||||||
|
|
||||||
|
-e |
||||||
|
search extactly, you should input an exactly package name as keyword |
||||||
|
`, |
||||||
|
} |
||||||
|
|
||||||
|
func init() { |
||||||
|
CmdSearch.Run = runSearch |
||||||
|
CmdSearch.Flags = map[string]bool{ |
||||||
|
"-s": false, |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func printSearchPrompt(flag string) { |
||||||
|
switch flag { |
||||||
|
case "-s": |
||||||
|
doc.ColorLog("[INFO] You enabled start a service.\n") |
||||||
|
case "-e": |
||||||
|
doc.ColorLog("[INFO] You enabled exactly search.\n") |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// search packages
|
||||||
|
func runSearch(cmd *Command, args []string) { |
||||||
|
// Check flags.
|
||||||
|
num := checkFlags(cmd.Flags, args, printSearchPrompt) |
||||||
|
if num == -1 { |
||||||
|
return |
||||||
|
} |
||||||
|
args = args[num:] |
||||||
|
|
||||||
|
// Check length of arguments.
|
||||||
|
if len(args) < 1 { |
||||||
|
doc.ColorLog("[ERROR] Please input package's keyword.\n") |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if cmd.Flags["-e"] { |
||||||
|
search(args[0], true) |
||||||
|
} else { |
||||||
|
search(args[0], false) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/* |
||||||
|
request local or remote search service to find packages according to keyword inputed |
||||||
|
*/ |
||||||
|
func search(keyword string, isExactly bool) { |
||||||
|
url := "http://localhost:8991/search?" |
||||||
|
if isExactly { |
||||||
|
url = "http://localhost:8991/searche?" |
||||||
|
} |
||||||
|
resp, err := http.Get(url + keyword) |
||||||
|
if err != nil { |
||||||
|
doc.ColorLog(err.Error()) |
||||||
|
return |
||||||
|
} |
||||||
|
defer resp.Body.Close() |
||||||
|
|
||||||
|
if resp.StatusCode == 200 { |
||||||
|
contents, err := ioutil.ReadAll(resp.Body) |
||||||
|
if err != nil { |
||||||
|
doc.ColorLog(err.Error()) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
pkgs := make([]string, 0) |
||||||
|
err = json.Unmarshal(contents, &pkgs) |
||||||
|
if err != nil { |
||||||
|
doc.ColorLog(err.Error()) |
||||||
|
return |
||||||
|
} |
||||||
|
for i, pkg := range pkgs { |
||||||
|
fmt.Println(i+1, pkg) |
||||||
|
} |
||||||
|
} else { |
||||||
|
doc.ColorLog(resp.Status) |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,394 @@ |
|||||||
|
package cmd |
||||||
|
|
||||||
|
import ( |
||||||
|
"../doc" |
||||||
|
"fmt" |
||||||
|
"github.com/syndtr/goleveldb/leveldb" |
||||||
|
"github.com/syndtr/goleveldb/leveldb/errors" |
||||||
|
"github.com/syndtr/goleveldb/leveldb/opt" |
||||||
|
"io/ioutil" |
||||||
|
"net/http" |
||||||
|
"os" |
||||||
|
"os/user" |
||||||
|
"strconv" |
||||||
|
"strings" |
||||||
|
) |
||||||
|
|
||||||
|
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": |
||||||
|
doc.ColorLog("[INFO] You enabled start a service only localhost.\n") |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Not implemented
|
||||||
|
func autoPort() string { |
||||||
|
return "8991" |
||||||
|
} |
||||||
|
|
||||||
|
// 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] |
||||||
|
} |
||||||
|
|
||||||
|
startService(listen + port) |
||||||
|
} |
||||||
|
|
||||||
|
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, 0) |
||||||
|
res[strings.Join(ps, "/")] = true |
||||||
|
for _, w := range ps { |
||||||
|
splitWord(w, &res) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
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 addPkg(pkg *Pkg) error { |
||||||
|
batch := new(leveldb.Batch) |
||||||
|
strLastId, err := dbGet("lastId") |
||||||
|
if err != nil { |
||||||
|
if err == errors.ErrNotFound { |
||||||
|
strLastId = "0" |
||||||
|
err = batchPut(batch, "lastId", strLastId) |
||||||
|
} else { |
||||||
|
return err |
||||||
|
} |
||||||
|
} |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
fmt.Println("last id is ", strLastId) |
||||||
|
|
||||||
|
lastId, err := strconv.ParseInt(strLastId, 0, 64) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
pkgKey := fmt.Sprintf("index:%v", pkg.Name) |
||||||
|
|
||||||
|
id, err := dbGet(pkgKey) |
||||||
|
if err != nil { |
||||||
|
if err == errors.ErrNotFound { |
||||||
|
id = fmt.Sprintf("%v", lastId+1) |
||||||
|
fmt.Println(id) |
||||||
|
err = batchPut(batch, "lastId", id) |
||||||
|
if err == nil { |
||||||
|
err = batchPut(batch, pkgKey, id) |
||||||
|
} |
||||||
|
if err == nil { |
||||||
|
err = batchPut(batch, "pkg:"+id, pkg.Name) |
||||||
|
} |
||||||
|
total, err := dbGet("total") |
||||||
|
if err != nil { |
||||||
|
if err == errors.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 |
||||||
|
} |
||||||
|
|
||||||
|
vers, err := dbGet("ver:" + id) |
||||||
|
needSplit := (err == errors.ErrNotFound) |
||||||
|
if err != nil { |
||||||
|
if err != errors.ErrNotFound { |
||||||
|
return err |
||||||
|
} |
||||||
|
} else { |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
if vers == "" { |
||||||
|
fmt.Println(pkg) |
||||||
|
vers = pkg.VerString() |
||||||
|
} else { |
||||||
|
if !strings.Contains(vers, pkg.VerString()) { |
||||||
|
vers = vers + "," + pkg.VerString() |
||||||
|
} else { |
||||||
|
return nil |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
err = batchPut(batch, "ver:"+id, vers) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if !needSplit { |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
keys := splitPkgName(pkg.Name) |
||||||
|
|
||||||
|
for key, _ := range keys { |
||||||
|
err = batchPut(batch, fmt.Sprintf("key:%v:%v", key, id), "") |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return db.Write(batch, wo) |
||||||
|
} |
||||||
|
|
||||||
|
func rmPkg(pkg *Pkg) { |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
var db *leveldb.DB |
||||||
|
|
||||||
|
// service should be run
|
||||||
|
func autoRun() { |
||||||
|
s, _, _ := runningStatus() |
||||||
|
if s == STOP { |
||||||
|
os.StartProcess("gopm", []string{"serve", "-l"}, nil) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func runningStatus() (int, int, int) { |
||||||
|
contentByte, err := ioutil.ReadFile("~/.gopm/var/pid") |
||||||
|
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 startService(listen string) { |
||||||
|
// check the pre serve's type
|
||||||
|
curUser, err := user.Current() |
||||||
|
if err != nil { |
||||||
|
fmt.Println(err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
dbDir = strings.Replace(dbDir, "~", curUser.HomeDir, -1) |
||||||
|
|
||||||
|
db, err = leveldb.OpenFile(dbDir, &opt.Options{Flag: opt.OFCreateIfMissing}) |
||||||
|
if err != nil { |
||||||
|
fmt.Println(err) |
||||||
|
return |
||||||
|
} |
||||||
|
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, 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:", 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 { |
||||||
|
doc.ColorLog(err.Error()) |
||||||
|
continue |
||||||
|
} |
||||||
|
pkgs = append(pkgs, pkg) |
||||||
|
} |
||||||
|
|
||||||
|
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 { |
||||||
|
_, err := dbGet("index:" + key) |
||||||
|
|
||||||
|
if err != nil { |
||||||
|
doc.ColorLog(err.Error()) |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
pkgs = append(pkgs, key) |
||||||
|
} |
||||||
|
|
||||||
|
w.Write([]byte("[\"" + strings.Join(pkgs, "\", \"") + "\"]")) |
||||||
|
//}
|
||||||
|
} |
||||||
|
|
||||||
|
func addHandler(w http.ResponseWriter, r *http.Request) { |
||||||
|
//if r.Method == "POST" {
|
||||||
|
r.ParseForm() |
||||||
|
for key, _ := range r.Form { |
||||||
|
fmt.Println(key) |
||||||
|
pkg := NewPkg(key, "") |
||||||
|
if pkg != nil { |
||||||
|
err := addPkg(pkg) |
||||||
|
if err != nil { |
||||||
|
fmt.Println(err) |
||||||
|
} |
||||||
|
} else { |
||||||
|
fmt.Println(key) |
||||||
|
} |
||||||
|
} |
||||||
|
//}
|
||||||
|
} |
||||||
|
|
||||||
|
func rmHandler(w http.ResponseWriter, r *http.Request) { |
||||||
|
|
||||||
|
} |
Loading…
Reference in new issue