pkg: add generated core package

and internal qgo tool to generate them

Change-Id: Ic03dc27262d8769f3dc3cd56fa11d77ee0b68dc8
diff --git a/internal/cmd/qgo/qgo.go b/internal/cmd/qgo/qgo.go
new file mode 100644
index 0000000..7c2cbdf
--- /dev/null
+++ b/internal/cmd/qgo/qgo.go
@@ -0,0 +1,327 @@
+// Copyright 2018 The CUE 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.
+
+// qgo builds CUE builtin packages from Go packages.
+package main
+
+import (
+	"bytes"
+	"flag"
+	"fmt"
+	"go/ast"
+	"go/constant"
+	"go/format"
+	"go/parser"
+	"go/printer"
+	"go/token"
+	"go/types"
+	"io"
+	"io/ioutil"
+	"log"
+	"os"
+	"path/filepath"
+	"regexp"
+	"strings"
+
+	"golang.org/x/tools/go/loader"
+)
+
+const help = `
+Commands:
+extract		Extract one-line signature of exported types of
+			the given package.
+
+			Functions that have have more than one return
+			argument or unknown types are skipped.
+`
+
+// Even though all of the code is generated, the documentation is copied as is.
+// So for proper measure, include both the CUE and Go licenses.
+const copyright = `// Copyright 2018 The CUE 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.
+
+// Copyright 2018 The Go Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style
+// license that can be found in the LICENSE file.
+`
+
+var genLine string
+
+var (
+	exclude  = flag.String("exclude", "", "comma-separated list of regexps of entries to exclude")
+	stripstr = flag.Bool("stripstr", false, "Remove String suffix from functions")
+)
+
+func init() {
+	log.SetFlags(log.Lshortfile)
+}
+
+func main() {
+	flag.Parse()
+
+	genLine = "//go:generate " + strings.Join(os.Args, " ")
+
+	args := flag.Args()
+	if len(args) == 0 {
+		fmt.Println(strings.TrimSpace(help))
+		return
+	}
+
+	command := args[0]
+	args = args[1:]
+
+	switch command {
+	case "extract":
+		extract(args)
+	}
+}
+
+var exclusions []*regexp.Regexp
+
+func initExclusions() {
+	for _, re := range strings.Split(*exclude, ",") {
+		if re != "" {
+			exclusions = append(exclusions, regexp.MustCompile(re))
+		}
+	}
+}
+
+func filter(name string) bool {
+	if !ast.IsExported(name) {
+		return true
+	}
+	for _, ex := range exclusions {
+		if ex.MatchString(name) {
+			return true
+		}
+	}
+	return false
+}
+
+func pkgName() string {
+	pkg, err := os.Getwd()
+	if err != nil {
+		log.Fatal(err)
+	}
+	return filepath.Base(pkg)
+}
+
+type extracter struct {
+	prog *loader.Program
+	pkg  *loader.PackageInfo
+}
+
+func extract(args []string) {
+
+	cfg := loader.Config{
+		ParserMode: parser.ParseComments,
+	}
+	cfg.FromArgs(args, false)
+
+	e := extracter{}
+	var err error
+	e.prog, err = cfg.Load()
+	if err != nil {
+		log.Fatal(err)
+	}
+
+	lastPkg := ""
+	var w *bytes.Buffer
+	initExclusions()
+
+	flushFile := func() {
+		if w != nil && w.Len() > 0 {
+			b, err := format.Source(w.Bytes())
+			if err != nil {
+				log.Fatal(err)
+			}
+			err = ioutil.WriteFile(lastPkg+".go", b, 0644)
+			if err != nil {
+				log.Fatal(err)
+			}
+		}
+		w = &bytes.Buffer{}
+	}
+
+	for _, p := range e.prog.InitialPackages() {
+		e.pkg = p
+		for _, f := range p.Files {
+			if lastPkg != p.Pkg.Name() {
+				flushFile()
+				lastPkg = p.Pkg.Name()
+				fmt.Fprintln(w, copyright)
+				fmt.Fprintln(w, genLine)
+				fmt.Fprintln(w)
+				fmt.Fprintf(w, "package %s\n", pkgName())
+				fmt.Fprintln(w)
+				fmt.Fprintf(w, "import %q", p.Pkg.Path())
+				fmt.Fprintln(w)
+			}
+
+			for _, d := range f.Decls {
+				switch x := d.(type) {
+				case *ast.FuncDecl:
+					e.reportFun(w, x)
+				case *ast.GenDecl:
+					e.reportDecl(w, x)
+				}
+			}
+		}
+	}
+	flushFile()
+}
+
+func (e *extracter) reportFun(w io.Writer, x *ast.FuncDecl) {
+	if filter(x.Name.Name) {
+		return
+	}
+	pkgName := e.pkg.Pkg.Name()
+	override := ""
+	params := []ast.Expr{}
+	if x.Type.Params != nil {
+		for _, f := range x.Type.Params.List {
+			tx := f.Type
+			if star, isStar := tx.(*ast.StarExpr); isStar {
+				if i, ok := star.X.(*ast.Ident); ok && ast.IsExported(i.Name) {
+					f.Type = &ast.SelectorExpr{X: ast.NewIdent(pkgName), Sel: i}
+					if isStar {
+						f.Type = &ast.StarExpr{X: f.Type}
+					}
+				}
+			}
+			for _, n := range f.Names {
+				params = append(params, n)
+				if n.Name == pkgName {
+					override = pkgName + x.Name.Name
+				}
+			}
+		}
+	}
+	var fn ast.Expr = &ast.SelectorExpr{
+		X:   ast.NewIdent(pkgName),
+		Sel: x.Name,
+	}
+	if override != "" {
+		fn = ast.NewIdent(override)
+	}
+	x.Body = &ast.BlockStmt{List: []ast.Stmt{
+		&ast.ReturnStmt{Results: []ast.Expr{&ast.CallExpr{
+			Fun:  fn,
+			Args: params,
+		}}},
+	}}
+	if name := x.Name.Name; *stripstr && strings.HasSuffix(name, "String") {
+		newName := name[:len(name)-len("String")]
+		x.Name = ast.NewIdent(newName)
+		for _, c := range x.Doc.List {
+			c.Text = strings.Replace(c.Text, name, newName, -1)
+		}
+	}
+	types := []ast.Expr{}
+	if x.Recv == nil && x.Type != nil && x.Type.Results != nil && !strings.HasPrefix(x.Name.Name, "New") {
+		for _, f := range x.Type.Results.List {
+			if len(f.Names) == 0 {
+				types = append(types, f.Type)
+			} else {
+				for range f.Names {
+					types = append(types, f.Type)
+				}
+			}
+		}
+	}
+	if len(types) != 1 {
+		switch len(types) {
+		case 2:
+			if i, ok := types[1].(*ast.Ident); ok && i.Name == "error" {
+				break
+			}
+			fallthrough
+		default:
+			fmt.Printf("Skipping ")
+			x.Doc = nil
+			printer.Fprint(os.Stdout, e.prog.Fset, x)
+			fmt.Println()
+			return
+		}
+	}
+	fmt.Fprintln(w)
+	printer.Fprint(w, e.prog.Fset, x.Doc)
+	printer.Fprint(w, e.prog.Fset, x)
+	fmt.Fprint(w, "\n")
+	if override != "" {
+		fmt.Fprintf(w, "var %s = %s.%s\n\n", override, pkgName, x.Name.Name)
+	}
+}
+
+func (e *extracter) reportDecl(w io.Writer, x *ast.GenDecl) {
+	if x.Tok != token.CONST {
+		return
+	}
+	k := 0
+	for _, s := range x.Specs {
+		if v, ok := s.(*ast.ValueSpec); ok && !filter(v.Names[0].Name) {
+			if v.Values == nil {
+				v.Values = make([]ast.Expr, len(v.Names))
+			}
+			for i, expr := range v.Names {
+				// This check can be removed if we set constants to floats.
+				if _, ok := v.Values[i].(*ast.BasicLit); ok {
+					continue
+				}
+				tv, _ := types.Eval(e.prog.Fset, e.pkg.Pkg, v.Pos(), v.Names[0].Name)
+				tok := token.ILLEGAL
+				switch tv.Value.Kind() {
+				case constant.Bool:
+					v.Values[i] = ast.NewIdent(tv.Value.ExactString())
+					continue
+				case constant.String:
+					tok = token.STRING
+				case constant.Int:
+					tok = token.INT
+				case constant.Float:
+					tok = token.FLOAT
+				default:
+					fmt.Printf("Skipping %s\n", v.Names)
+					continue
+				}
+				v.Values[i] = &ast.BasicLit{
+					ValuePos: expr.Pos(),
+					Kind:     tok,
+					Value:    tv.Value.ExactString(),
+				}
+			}
+			v.Type = nil
+			x.Specs[k] = v
+			k++
+		}
+	}
+	x.Specs = x.Specs[:k]
+	if len(x.Specs) == 0 {
+		return
+	}
+	fmt.Fprintln(w)
+	printer.Fprint(w, e.prog.Fset, x)
+	fmt.Fprintln(w)
+}