Bogdan Caprita | 2b21936 | 2014-12-09 17:03:33 -0800 | [diff] [blame] | 1 | // Package keymgr provides a client for the Device Manager to manage keys in |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 2 | // the "Agent" process. |
| 3 | package keymgr |
| 4 | |
| 5 | import ( |
| 6 | "net" |
| 7 | "os" |
| 8 | "strconv" |
| 9 | "sync" |
| 10 | |
Jiri Simsa | 6ac9522 | 2015-02-23 16:11:49 -0800 | [diff] [blame] | 11 | "v.io/v23/context" |
| 12 | "v.io/v23/verror" |
Jiri Simsa | ffceefa | 2015-02-28 11:03:34 -0800 | [diff] [blame] | 13 | "v.io/x/ref/lib/unixfd" |
| 14 | "v.io/x/ref/security/agent/server" |
Mike Burrows | 4b9b5d5 | 2014-12-05 18:08:07 -0800 | [diff] [blame] | 15 | ) |
| 16 | |
Jiri Simsa | ffceefa | 2015-02-28 11:03:34 -0800 | [diff] [blame] | 17 | const pkgPath = "v.io/x/ref/security/agent/keymgr" |
Mike Burrows | 4b9b5d5 | 2014-12-05 18:08:07 -0800 | [diff] [blame] | 18 | |
| 19 | // Errors |
| 20 | var ( |
| 21 | errInvalidResponse = verror.Register(pkgPath+".errInvalidResponse", |
| 22 | verror.NoRetry, "{1:}{2:} invalid response from agent. (expected {3} bytes, got {4})") |
| 23 | errInvalidKeyHandle = verror.Register(pkgPath+".errInvalidKeyHandle", |
| 24 | verror.NoRetry, "{1:}{2:} Invalid key handle") |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 25 | ) |
| 26 | |
| 27 | const defaultManagerSocket = 4 |
| 28 | |
| 29 | type Agent struct { |
| 30 | conn *net.UnixConn // Guarded by mu |
| 31 | mu sync.Mutex |
| 32 | } |
| 33 | |
| 34 | // NewAgent returns a client connected to the agent on the default file descriptors. |
| 35 | func NewAgent() (*Agent, error) { |
| 36 | return newAgent(defaultManagerSocket) |
| 37 | } |
| 38 | |
Ryan Brown | a08a221 | 2015-01-15 15:40:10 -0800 | [diff] [blame] | 39 | func NewLocalAgent(ctx *context.T, path string, passphrase []byte) (*Agent, error) { |
| 40 | file, err := server.RunKeyManager(ctx, path, passphrase) |
| 41 | if err != nil { |
| 42 | return nil, err |
| 43 | } |
Ryan Brown | a08a221 | 2015-01-15 15:40:10 -0800 | [diff] [blame] | 44 | conn, err := net.FileConn(file) |
| 45 | if err != nil { |
| 46 | return nil, err |
| 47 | } |
| 48 | return &Agent{conn: conn.(*net.UnixConn)}, nil |
| 49 | } |
| 50 | |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 51 | func newAgent(fd int) (a *Agent, err error) { |
| 52 | file := os.NewFile(uintptr(fd), "fd") |
| 53 | defer file.Close() |
| 54 | conn, err := net.FileConn(file) |
| 55 | if err != nil { |
| 56 | return nil, err |
| 57 | } |
| 58 | |
| 59 | return &Agent{conn: conn.(*net.UnixConn)}, nil |
| 60 | } |
| 61 | |
Matt Rosencrantz | 4f8ac60 | 2014-12-29 14:42:48 -0800 | [diff] [blame] | 62 | // TODO(caprita): Get rid of *context.T arg. Doesn't seem to be used. |
Bogdan Caprita | 7f49167 | 2014-11-13 14:51:08 -0800 | [diff] [blame] | 63 | |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 64 | // NewPrincipal creates a new principal and returns the handle and a socket serving |
| 65 | // the principal. |
| 66 | // Typically the socket will be passed to a child process using cmd.ExtraFiles. |
Matt Rosencrantz | 4f8ac60 | 2014-12-29 14:42:48 -0800 | [diff] [blame] | 67 | func (a *Agent) NewPrincipal(ctx *context.T, inMemory bool) (handle []byte, conn *os.File, err error) { |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 68 | req := make([]byte, 1) |
Bogdan Caprita | 7f49167 | 2014-11-13 14:51:08 -0800 | [diff] [blame] | 69 | if inMemory { |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 70 | req[0] = 1 |
| 71 | } |
| 72 | a.mu.Lock() |
| 73 | defer a.mu.Unlock() |
| 74 | conn, err = a.connect(req) |
| 75 | if err != nil { |
| 76 | return nil, nil, err |
| 77 | } |
| 78 | buf := make([]byte, server.PrincipalHandleByteSize) |
| 79 | n, err := a.conn.Read(buf) |
| 80 | if err != nil { |
| 81 | conn.Close() |
| 82 | return nil, nil, err |
| 83 | } |
| 84 | if n != server.PrincipalHandleByteSize { |
| 85 | conn.Close() |
Todd Wang | ff73e1f | 2015-02-10 21:45:52 -0800 | [diff] [blame] | 86 | return nil, nil, verror.New(errInvalidResponse, ctx, server.PrincipalHandleByteSize, n) |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 87 | } |
| 88 | return buf, conn, nil |
| 89 | } |
| 90 | |
| 91 | func (a *Agent) connect(req []byte) (*os.File, error) { |
Bogdan Caprita | bb37c54 | 2015-01-22 10:21:57 -0800 | [diff] [blame] | 92 | addr, err := unixfd.SendConnection(a.conn, req) |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 93 | if err != nil { |
| 94 | return nil, err |
| 95 | } |
| 96 | fd, err := strconv.ParseInt(addr.String(), 10, 32) |
| 97 | if err != nil { |
| 98 | return nil, err |
| 99 | } |
| 100 | return os.NewFile(uintptr(fd), "client"), nil |
| 101 | } |
| 102 | |
| 103 | // NewConnection creates a connection to an agent which exports a principal |
| 104 | // previously created with NewPrincipal. |
| 105 | // Typically this will be passed to a child process using cmd.ExtraFiles. |
| 106 | func (a *Agent) NewConnection(handle []byte) (*os.File, error) { |
| 107 | if len(handle) != server.PrincipalHandleByteSize { |
Todd Wang | ff73e1f | 2015-02-10 21:45:52 -0800 | [diff] [blame] | 108 | return nil, verror.New(errInvalidKeyHandle, nil) |
Ryan Brown | 8178944 | 2014-10-30 13:23:53 -0700 | [diff] [blame] | 109 | } |
| 110 | a.mu.Lock() |
| 111 | defer a.mu.Unlock() |
| 112 | return a.connect(handle) |
| 113 | } |