Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 1 | package core |
| 2 | |
| 3 | import ( |
| 4 | "fmt" |
| 5 | "io" |
| 6 | "os" |
| 7 | "strings" |
| 8 | |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 9 | "veyron.io/veyron/veyron2/context" |
Jiri Simsa | 519c507 | 2014-09-17 21:37:57 -0700 | [diff] [blame] | 10 | "veyron.io/veyron/veyron2/naming" |
Asim Shankar | cc04421 | 2014-10-15 23:25:26 -0700 | [diff] [blame] | 11 | "veyron.io/veyron/veyron2/options" |
Jiri Simsa | 519c507 | 2014-09-17 21:37:57 -0700 | [diff] [blame] | 12 | "veyron.io/veyron/veyron2/rt" |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 13 | |
Jiri Simsa | 519c507 | 2014-09-17 21:37:57 -0700 | [diff] [blame] | 14 | "veyron.io/veyron/veyron/lib/modules" |
| 15 | mounttable "veyron.io/veyron/veyron/services/mounttable/lib" |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 16 | ) |
| 17 | |
| 18 | func init() { |
Cosmos Nicolaou | 1e78ccc | 2014-10-09 08:10:26 -0700 | [diff] [blame] | 19 | modules.RegisterChild(RootMTCommand, "", rootMountTable) |
| 20 | modules.RegisterChild(MTCommand, `<mount point> |
| 21 | reads NAMESPACE_ROOT from its environment and mounts a new mount table at <mount point>`, mountTable) |
| 22 | modules.RegisterChild(LSExternalCommand, `<glob>... |
| 23 | issues glob requests using the current processes namespace library`, |
| 24 | ls) |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 25 | } |
| 26 | |
| 27 | func mountTable(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error { |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 28 | return runMT(false, stdin, stdout, stderr, env, args...) |
| 29 | } |
| 30 | |
| 31 | func rootMountTable(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error { |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 32 | return runMT(true, stdin, stdout, stderr, env, args...) |
| 33 | } |
| 34 | |
| 35 | func runMT(root bool, stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error { |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 36 | r := rt.R() |
Cosmos Nicolaou | d811b07 | 2014-10-28 17:46:27 -0700 | [diff] [blame] | 37 | fl, args, err := parseListenFlags(args) |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 38 | if err != nil { |
Adam Sadovsky | d21d95a | 2014-10-29 09:56:59 -0700 | [diff] [blame] | 39 | return fmt.Errorf("failed to parse args: %s", err) |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 40 | } |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 41 | lspec := initListenSpec(fl) |
Asim Shankar | cc04421 | 2014-10-15 23:25:26 -0700 | [diff] [blame] | 42 | server, err := r.NewServer(options.ServesMountTable(true)) |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 43 | if err != nil { |
| 44 | return fmt.Errorf("root failed: %v", err) |
| 45 | } |
| 46 | mp := "" |
| 47 | if !root { |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 48 | if err := checkArgs(args, 1, "<mount point>"); err != nil { |
| 49 | return err |
| 50 | } |
| 51 | mp = args[0] |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 52 | } |
| 53 | mt, err := mounttable.NewMountTable("") |
| 54 | if err != nil { |
| 55 | return fmt.Errorf("mounttable.NewMountTable failed: %s", err) |
| 56 | } |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 57 | ep, err := server.Listen(lspec) |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 58 | if err != nil { |
| 59 | return fmt.Errorf("server.Listen failed: %s", err) |
| 60 | } |
| 61 | if err := server.Serve(mp, mt); err != nil { |
| 62 | return fmt.Errorf("root failed: %s", err) |
| 63 | } |
| 64 | name := naming.JoinAddressName(ep.String(), "") |
Cosmos Nicolaou | addf483 | 2014-09-10 21:36:54 -0700 | [diff] [blame] | 65 | fmt.Fprintf(stdout, "MT_NAME=%s\n", name) |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 66 | fmt.Fprintf(stdout, "MT_ADDR=%s\n", ep.String()) |
Cosmos Nicolaou | addf483 | 2014-09-10 21:36:54 -0700 | [diff] [blame] | 67 | fmt.Fprintf(stdout, "PID=%d\n", os.Getpid()) |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 68 | modules.WaitForEOF(stdin) |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 69 | return nil |
| 70 | } |
| 71 | |
| 72 | func ls(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error { |
| 73 | details := false |
Adam Sadovsky | d21d95a | 2014-10-29 09:56:59 -0700 | [diff] [blame] | 74 | args = args[1:] // skip over command name |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 75 | if len(args) > 0 && args[0] == "-l" { |
| 76 | details = true |
| 77 | args = args[1:] |
| 78 | } |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 79 | ns := rt.R().Namespace() |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 80 | entry := 0 |
| 81 | output := "" |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 82 | for _, pattern := range args { |
| 83 | ch, err := ns.Glob(rt.R().NewContext(), pattern) |
| 84 | if err != nil { |
| 85 | return err |
| 86 | } |
| 87 | for n := range ch { |
| 88 | if details { |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 89 | output += fmt.Sprintf("R%d=%s[", entry, n.Name) |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 90 | t := "" |
| 91 | for _, s := range n.Servers { |
David Why Use Two When One Will Do Presotto | 59a254c | 2014-10-30 13:09:29 -0700 | [diff] [blame^] | 92 | t += fmt.Sprintf("%s:%s, ", s.Server, s.Expires) |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 93 | } |
| 94 | t = strings.TrimSuffix(t, ", ") |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 95 | output += fmt.Sprintf("%s]\n", t) |
| 96 | entry += 1 |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 97 | } else { |
| 98 | if len(n.Name) > 0 { |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 99 | output += fmt.Sprintf("R%d=%s\n", entry, n.Name) |
| 100 | entry += 1 |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 101 | } |
| 102 | } |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 103 | |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 104 | } |
| 105 | } |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 106 | fmt.Fprintf(stdout, "RN=%d\n", entry) |
| 107 | fmt.Fprint(stdout, output) |
Cosmos Nicolaou | 0f0e877 | 2014-09-10 21:29:25 -0700 | [diff] [blame] | 108 | return nil |
| 109 | } |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 110 | |
| 111 | type resolver func(ctx context.T, name string) (names []string, err error) |
| 112 | |
| 113 | func resolve(fn resolver, stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error { |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 114 | if err := checkArgs(args[1:], 1, "<name>"); err != nil { |
| 115 | return err |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 116 | } |
Cosmos Nicolaou | bbae388 | 2014-10-02 22:58:19 -0700 | [diff] [blame] | 117 | name := args[1] |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 118 | servers, err := fn(rt.R().NewContext(), name) |
| 119 | if err != nil { |
Cosmos Nicolaou | 9c9918d | 2014-09-23 08:45:56 -0700 | [diff] [blame] | 120 | fmt.Fprintf(stdout, "RN=0\n") |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 121 | return err |
| 122 | } |
| 123 | fmt.Fprintf(stdout, "RN=%d\n", len(servers)) |
| 124 | for i, s := range servers { |
| 125 | fmt.Fprintf(stdout, "R%d=%s\n", i, s) |
| 126 | } |
| 127 | return nil |
| 128 | } |
| 129 | |
| 130 | func resolveObject(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error { |
| 131 | return resolve(rt.R().Namespace().Resolve, stdin, stdout, stderr, env, args...) |
| 132 | } |
| 133 | |
| 134 | func resolveMT(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error { |
| 135 | return resolve(rt.R().Namespace().ResolveToMountTable, stdin, stdout, stderr, env, args...) |
| 136 | } |
| 137 | |
| 138 | func setNamespaceRoots(stdin io.Reader, stdout, stderr io.Writer, env map[string]string, args ...string) error { |
| 139 | ns := rt.R().Namespace() |
Cosmos Nicolaou | 4e213d7 | 2014-10-26 22:21:52 -0700 | [diff] [blame] | 140 | if err := checkArgs(args, -1, "<name>..."); err != nil { |
| 141 | return err |
Cosmos Nicolaou | bbae388 | 2014-10-02 22:58:19 -0700 | [diff] [blame] | 142 | } |
| 143 | return ns.SetRoots(args[1:]...) |
Cosmos Nicolaou | 9ca249d | 2014-09-18 15:07:12 -0700 | [diff] [blame] | 144 | } |