Jiri Simsa | d7616c9 | 2015-03-24 23:44:30 -0700 | [diff] [blame] | 1 | // Copyright 2015 The Vanadium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style |
| 3 | // license that can be found in the LICENSE file. |
| 4 | |
Matt Rosencrantz | 86ba1a1 | 2015-03-09 13:19:02 -0700 | [diff] [blame] | 5 | package lib |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 6 | |
| 7 | import ( |
| 8 | "errors" |
| 9 | "fmt" |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 10 | "sync" |
| 11 | |
Jiri Simsa | 6ac9522 | 2015-02-23 16:11:49 -0800 | [diff] [blame] | 12 | "v.io/v23/context" |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 13 | "v.io/v23/rpc" |
Jiri Simsa | 6ac9522 | 2015-02-23 16:11:49 -0800 | [diff] [blame] | 14 | "v.io/v23/security" |
| 15 | "v.io/v23/vdl" |
Jiri Simsa | 6ac9522 | 2015-02-23 16:11:49 -0800 | [diff] [blame] | 16 | "v.io/v23/vom" |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 17 | ) |
| 18 | |
Matt Rosencrantz | 86ba1a1 | 2015-03-09 13:19:02 -0700 | [diff] [blame] | 19 | type clientWithTimesCalled interface { |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 20 | rpc.Client |
Bogdan Caprita | e96cd04 | 2015-02-03 17:32:57 -0800 | [diff] [blame] | 21 | TimesCalled(method string) int |
| 22 | } |
| 23 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 24 | // NewSimpleClient creates a new mocked rpc client where the given map of method name |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 25 | // to outputs is used for evaluating the method calls. |
| 26 | // It also adds some testing features such as counters for number of times a method is called |
Matt Rosencrantz | 86ba1a1 | 2015-03-09 13:19:02 -0700 | [diff] [blame] | 27 | func newSimpleClient(methodsResults map[string][]interface{}) clientWithTimesCalled { |
Bogdan Caprita | e96cd04 | 2015-02-03 17:32:57 -0800 | [diff] [blame] | 28 | return &simpleMockClient{ |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 29 | results: methodsResults, |
| 30 | timesCalled: make(map[string]int), |
| 31 | } |
| 32 | } |
| 33 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 34 | // simpleMockClient implements rpc.Client |
Bogdan Caprita | e96cd04 | 2015-02-03 17:32:57 -0800 | [diff] [blame] | 35 | type simpleMockClient struct { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 36 | // Protects timesCalled |
| 37 | sync.Mutex |
| 38 | |
| 39 | // results is a map of method names to results |
| 40 | results map[string][]interface{} |
| 41 | // timesCalled is a counter for number of times StartCall is called on a specific method name |
| 42 | timesCalled map[string]int |
| 43 | } |
| 44 | |
| 45 | // TimesCalled returns number of times the given method has been called. |
Bogdan Caprita | e96cd04 | 2015-02-03 17:32:57 -0800 | [diff] [blame] | 46 | func (c *simpleMockClient) TimesCalled(method string) int { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 47 | return c.timesCalled[method] |
| 48 | } |
| 49 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 50 | // StartCall Implements rpc.Client |
| 51 | func (c *simpleMockClient) StartCall(ctx *context.T, name, method string, args []interface{}, opts ...rpc.CallOpt) (rpc.ClientCall, error) { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 52 | results, ok := c.results[method] |
| 53 | if !ok { |
Suharsh Sivakumar | 076e953 | 2015-04-09 17:36:25 -0700 | [diff] [blame] | 54 | return nil, fmt.Errorf("method %s not found", method) |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 55 | } |
| 56 | |
Benjamin Prosnitz | 0db77a2 | 2015-01-20 14:25:15 -0800 | [diff] [blame] | 57 | // Copy the results so that they can be modified without effecting the original. |
| 58 | // This must be done via vom encode and decode rather than a direct deep copy because (among other reasons) |
| 59 | // reflect-based deep copy on vdl.Type objects will fail because of their private fields. This is not a problem with vom |
| 60 | // as it manually creates the type objects. It is also more realistic to use the same mechanism as the ultimate calls. |
Todd Wang | 3425a90 | 2015-01-21 18:43:59 -0800 | [diff] [blame] | 61 | vomBytes, err := vom.Encode(results) |
Benjamin Prosnitz | 0db77a2 | 2015-01-20 14:25:15 -0800 | [diff] [blame] | 62 | if err != nil { |
| 63 | panic(fmt.Sprintf("Error copying value with vom (failed on encode): %v", err)) |
| 64 | } |
| 65 | var copiedResults []interface{} |
Todd Wang | 3425a90 | 2015-01-21 18:43:59 -0800 | [diff] [blame] | 66 | if err := vom.Decode(vomBytes, &copiedResults); err != nil { |
Benjamin Prosnitz | 0db77a2 | 2015-01-20 14:25:15 -0800 | [diff] [blame] | 67 | panic(fmt.Sprintf("Error copying value with vom (failed on decode): %v", err)) |
| 68 | } |
| 69 | |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 70 | clientCall := mockCall{ |
Benjamin Prosnitz | 0db77a2 | 2015-01-20 14:25:15 -0800 | [diff] [blame] | 71 | results: copiedResults, |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 72 | } |
| 73 | |
| 74 | c.Lock() |
| 75 | c.timesCalled[method]++ |
| 76 | c.Unlock() |
| 77 | |
| 78 | return &clientCall, nil |
| 79 | } |
| 80 | |
Suharsh Sivakumar | 076e953 | 2015-04-09 17:36:25 -0700 | [diff] [blame] | 81 | func (c *simpleMockClient) Call(ctx *context.T, name, method string, inArgs, outArgs []interface{}, callOpts ...rpc.CallOpt) error { |
| 82 | call, err := c.StartCall(ctx, name, method, inArgs, callOpts...) |
| 83 | if err != nil { |
| 84 | return err |
| 85 | } |
| 86 | return call.Finish(outArgs...) |
| 87 | } |
| 88 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 89 | // Close implements rpc.Client |
Bogdan Caprita | e96cd04 | 2015-02-03 17:32:57 -0800 | [diff] [blame] | 90 | func (*simpleMockClient) Close() { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 91 | } |
| 92 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 93 | // mockCall implements rpc.ClientCall |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 94 | type mockCall struct { |
| 95 | mockStream |
| 96 | results []interface{} |
| 97 | } |
| 98 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 99 | // Cancel implements rpc.ClientCall |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 100 | func (*mockCall) Cancel() { |
| 101 | } |
| 102 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 103 | // CloseSend implements rpc.ClientCall |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 104 | func (*mockCall) CloseSend() error { |
| 105 | return nil |
| 106 | } |
| 107 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 108 | // Finish implements rpc.ClientCall |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 109 | func (mc *mockCall) Finish(resultptrs ...interface{}) error { |
| 110 | if got, want := len(resultptrs), len(mc.results); got != want { |
| 111 | return errors.New(fmt.Sprintf("wrong number of output results; expected resultptrs of size %d but got %d", want, got)) |
| 112 | } |
| 113 | for ax, res := range resultptrs { |
| 114 | if mc.results[ax] != nil { |
Todd Wang | 52cd532 | 2015-02-13 18:01:56 -0800 | [diff] [blame] | 115 | if err := vdl.Convert(res, mc.results[ax]); err != nil { |
Benjamin Prosnitz | 0db77a2 | 2015-01-20 14:25:15 -0800 | [diff] [blame] | 116 | panic(fmt.Sprintf("Error converting out argument %#v: %v", mc.results[ax], err)) |
| 117 | } |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 118 | } |
| 119 | } |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 120 | return nil |
| 121 | } |
| 122 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 123 | // RemoteBlessings implements rpc.ClientCall |
Asim Shankar | 2d731a9 | 2014-09-29 17:46:38 -0700 | [diff] [blame] | 124 | func (*mockCall) RemoteBlessings() ([]string, security.Blessings) { |
Asim Shankar | 2bf7b1e | 2015-02-27 00:45:12 -0800 | [diff] [blame] | 125 | return []string{}, security.Blessings{} |
Asim Shankar | 2d731a9 | 2014-09-29 17:46:38 -0700 | [diff] [blame] | 126 | } |
| 127 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 128 | //mockStream implements rpc.Stream |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 129 | type mockStream struct{} |
| 130 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 131 | //Send implements rpc.Stream |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 132 | func (*mockStream) Send(interface{}) error { |
| 133 | return nil |
| 134 | } |
| 135 | |
Matt Rosencrantz | 94502cf | 2015-03-18 09:43:44 -0700 | [diff] [blame] | 136 | //Recv implements rpc.Stream |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 137 | func (*mockStream) Recv(interface{}) error { |
Jiri Simsa | 5293dcb | 2014-05-10 09:56:38 -0700 | [diff] [blame] | 138 | return nil |
| 139 | } |