fake.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. /*
  2. Licensed under the Apache License, Version 2.0 (the "License");
  3. you may not use this file except in compliance with the License.
  4. You may obtain a copy of the License at
  5. http://www.apache.org/licenses/LICENSE-2.0
  6. Unless required by applicable law or agreed to in writing, software
  7. distributed under the License is distributed on an "AS IS" BASIS,
  8. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  9. See the License for the specific language governing permissions and
  10. limitations under the License.
  11. */
  12. package fake
  13. import (
  14. "errors"
  15. "fmt"
  16. "github.com/1Password/connect-sdk-go/onepassword"
  17. )
  18. // OnePasswordMockClient is a fake connect.Client.
  19. type OnePasswordMockClient struct {
  20. MockVaults map[string][]onepassword.Vault
  21. MockItems map[string][]onepassword.Item // ID and Title only
  22. MockItemFields map[string]map[string][]*onepassword.ItemField
  23. MockFileContents map[string][]byte
  24. UpdateItemValidateFunc func(*onepassword.Item, string) (*onepassword.Item, error)
  25. CreateItemValidateFunc func(*onepassword.Item, string) (*onepassword.Item, error)
  26. DeleteItemValidateFunc func(*onepassword.Item, string) error
  27. }
  28. // NewMockClient returns an instantiated mock client.
  29. func NewMockClient() *OnePasswordMockClient {
  30. return &OnePasswordMockClient{
  31. MockVaults: map[string][]onepassword.Vault{},
  32. MockItems: map[string][]onepassword.Item{},
  33. MockItemFields: map[string]map[string][]*onepassword.ItemField{},
  34. MockFileContents: map[string][]byte{},
  35. }
  36. }
  37. // GetVaults unused fake.
  38. func (mockClient *OnePasswordMockClient) GetVaults() ([]onepassword.Vault, error) {
  39. return []onepassword.Vault{}, nil
  40. }
  41. // GetVault unused fake.
  42. func (mockClient *OnePasswordMockClient) GetVault(uuid string) (*onepassword.Vault, error) {
  43. return mockClient.GetVaultByTitle(uuid)
  44. }
  45. // GetVaultByUUID unused fake.
  46. func (mockClient *OnePasswordMockClient) GetVaultByUUID(uuid string) (*onepassword.Vault, error) {
  47. return &mockClient.MockVaults[uuid][0], nil
  48. }
  49. // GetVaultByTitle returns a vault, you must preload, only one.
  50. func (mockClient *OnePasswordMockClient) GetVaultByTitle(uuid string) (*onepassword.Vault, error) {
  51. if len(mockClient.MockVaults[uuid]) != 0 {
  52. return &mockClient.MockVaults[uuid][0], nil
  53. }
  54. return &onepassword.Vault{}, nil
  55. }
  56. // GetVaultsByTitle returns a list of vaults, you must preload.
  57. func (mockClient *OnePasswordMockClient) GetVaultsByTitle(uuid string) ([]onepassword.Vault, error) {
  58. return mockClient.MockVaults[uuid], nil
  59. }
  60. // GetItems returns []onepassword.Item, you must preload.
  61. func (mockClient *OnePasswordMockClient) GetItems(vaultUUID string) ([]onepassword.Item, error) {
  62. return mockClient.MockItems[vaultUUID], nil
  63. }
  64. // GetItem returns a *onepassword.Item, you must preload.
  65. func (mockClient *OnePasswordMockClient) GetItem(itemUUID, vaultUUID string) (*onepassword.Item, error) {
  66. return mockClient.GetItemByUUID(itemUUID, vaultUUID)
  67. }
  68. // GetItemByUUID returns a *onepassword.Item, you must preload.
  69. func (mockClient *OnePasswordMockClient) GetItemByUUID(itemUUID, vaultUUID string) (*onepassword.Item, error) {
  70. for _, item := range mockClient.MockItems[vaultUUID] {
  71. if item.ID == itemUUID {
  72. // load the fields that GetItemsByTitle does not
  73. item.Fields = mockClient.MockItemFields[vaultUUID][itemUUID]
  74. return &item, nil
  75. }
  76. }
  77. return &onepassword.Item{}, errors.New("status 400: Invalid GetItemByUUID")
  78. }
  79. // GetItemByTitle unused fake.
  80. func (mockClient *OnePasswordMockClient) GetItemByTitle(_, _ string) (*onepassword.Item, error) {
  81. return &onepassword.Item{}, nil
  82. }
  83. // GetItemsByTitle returns a list of items, you must preload.
  84. func (mockClient *OnePasswordMockClient) GetItemsByTitle(itemUUID, vaultUUID string) ([]onepassword.Item, error) {
  85. items := []onepassword.Item{}
  86. for _, item := range mockClient.MockItems[vaultUUID] {
  87. if item.Title == itemUUID {
  88. items = append(items, item)
  89. }
  90. }
  91. return items, nil
  92. }
  93. // CreateItem will call a validation function if set.
  94. func (mockClient *OnePasswordMockClient) CreateItem(i *onepassword.Item, s string) (*onepassword.Item, error) {
  95. if mockClient.CreateItemValidateFunc != nil {
  96. item, err := mockClient.CreateItemValidateFunc(i, s)
  97. if err == nil {
  98. mockClient.MockItems[i.Vault.ID] = append(mockClient.MockItems[i.Vault.ID], *item)
  99. if mockClient.MockItemFields[i.Vault.ID] == nil {
  100. mockClient.MockItemFields[i.Vault.ID] = make(map[string][]*onepassword.ItemField)
  101. }
  102. mockClient.MockItemFields[i.Vault.ID][i.ID] = item.Fields
  103. }
  104. return item, err
  105. }
  106. return &onepassword.Item{}, nil
  107. }
  108. // UpdateItem will call a validation function if set.
  109. func (mockClient *OnePasswordMockClient) UpdateItem(i *onepassword.Item, s string) (*onepassword.Item, error) {
  110. if mockClient.UpdateItemValidateFunc != nil {
  111. updatedItem, err := mockClient.UpdateItemValidateFunc(i, s)
  112. if err == nil {
  113. for index, item := range mockClient.MockItems[i.Vault.ID] {
  114. if item.ID == updatedItem.ID {
  115. mockClient.MockItems[i.Vault.ID][index] = *updatedItem
  116. mockClient.MockItemFields[i.Vault.ID][updatedItem.ID] = updatedItem.Fields
  117. break
  118. }
  119. }
  120. }
  121. return updatedItem, err
  122. }
  123. return &onepassword.Item{}, nil
  124. }
  125. // DeleteItem will call a validation function if set.
  126. func (mockClient *OnePasswordMockClient) DeleteItem(i *onepassword.Item, s string) error {
  127. if mockClient.DeleteItemValidateFunc != nil {
  128. return mockClient.DeleteItemValidateFunc(i, s)
  129. }
  130. return nil
  131. }
  132. // DeleteItemByID unused fake.
  133. func (mockClient *OnePasswordMockClient) DeleteItemByID(_, _ string) error {
  134. return nil
  135. }
  136. // DeleteItemByTitle unused fake.
  137. func (mockClient *OnePasswordMockClient) DeleteItemByTitle(_, _ string) error {
  138. return nil
  139. }
  140. // GetFiles unused fake.
  141. func (mockClient *OnePasswordMockClient) GetFiles(_, _ string) ([]onepassword.File, error) {
  142. return []onepassword.File{}, nil
  143. }
  144. // GetFile unused fake.
  145. func (mockClient *OnePasswordMockClient) GetFile(_, _, _ string) (*onepassword.File, error) {
  146. return &onepassword.File{}, nil
  147. }
  148. // GetFileContent returns file data, you must preload.
  149. func (mockClient *OnePasswordMockClient) GetFileContent(file *onepassword.File) ([]byte, error) {
  150. value, ok := mockClient.MockFileContents[file.Name]
  151. if !ok {
  152. return []byte{}, errors.New("status 400: Invalid File Name")
  153. }
  154. return value, nil
  155. }
  156. // DownloadFile unused fake.
  157. func (mockClient *OnePasswordMockClient) DownloadFile(_ *onepassword.File, _ string, _ bool) (string, error) {
  158. return "", nil
  159. }
  160. // LoadStructFromItemByUUID unused fake.
  161. func (mockClient *OnePasswordMockClient) LoadStructFromItemByUUID(_ any, _, _ string) error {
  162. return nil
  163. }
  164. // LoadStructFromItemByTitle unused fake.
  165. func (mockClient *OnePasswordMockClient) LoadStructFromItemByTitle(_ any, _, _ string) error {
  166. return nil
  167. }
  168. // LoadStructFromItem unused fake.
  169. func (mockClient *OnePasswordMockClient) LoadStructFromItem(_ any, _, _ string) error {
  170. return nil
  171. }
  172. // LoadStructunused fake.
  173. func (mockClient *OnePasswordMockClient) LoadStruct(_ any) error {
  174. return nil
  175. }
  176. // // For rigging test cases
  177. // AddPredictableVault adds vaults to the mock client in a predictable way.
  178. func (mockClient *OnePasswordMockClient) AddPredictableVault(name string) *OnePasswordMockClient {
  179. mockClient.MockVaults[name] = append(mockClient.MockVaults[name], onepassword.Vault{
  180. ID: fmt.Sprintf("%s-id", name),
  181. Name: name,
  182. })
  183. return mockClient
  184. }
  185. // AddPredictableVaultUUID adds vaults to the mock client in a predictable way.
  186. func (mockClient *OnePasswordMockClient) AddPredictableVaultUUID(name string) *OnePasswordMockClient {
  187. mockClient.MockVaults[name] = append(mockClient.MockVaults[name], onepassword.Vault{
  188. ID: name,
  189. Name: name,
  190. })
  191. return mockClient
  192. }
  193. // AddPredictableItemWithField adds an item and it's fields to the mock client in a predictable way.
  194. func (mockClient *OnePasswordMockClient) AddPredictableItemWithField(vaultName, title, label, value string) *OnePasswordMockClient {
  195. itemID := fmt.Sprintf("%s-id", title)
  196. vaultID := fmt.Sprintf("%s-id", vaultName)
  197. mockClient.MockItems[vaultID] = append(mockClient.MockItems[vaultID], onepassword.Item{
  198. ID: itemID,
  199. Title: title,
  200. Vault: onepassword.ItemVault{ID: vaultID},
  201. })
  202. if mockClient.MockItemFields[vaultID] == nil {
  203. mockClient.MockItemFields[vaultID] = make(map[string][]*onepassword.ItemField)
  204. }
  205. mockClient.MockItemFields[vaultID][itemID] = append(mockClient.MockItemFields[vaultID][itemID], &onepassword.ItemField{
  206. Label: label,
  207. Value: value,
  208. })
  209. return mockClient
  210. }
  211. // AddPredictableItemWithFieldUUID adds an item and it's fields to the mock client in a predictable way.
  212. func (mockClient *OnePasswordMockClient) AddPredictableItemWithFieldUUID(vaultName, title, label, value string) *OnePasswordMockClient {
  213. mockClient.MockItems[vaultName] = append(mockClient.MockItems[vaultName], onepassword.Item{
  214. ID: title,
  215. Title: title,
  216. Vault: onepassword.ItemVault{ID: vaultName},
  217. })
  218. if mockClient.MockItemFields[vaultName] == nil {
  219. mockClient.MockItemFields[vaultName] = make(map[string][]*onepassword.ItemField)
  220. }
  221. mockClient.MockItemFields[vaultName][title] = append(mockClient.MockItemFields[vaultName][title], &onepassword.ItemField{
  222. Label: label,
  223. Value: value,
  224. })
  225. return mockClient
  226. }
  227. // AppendVault appends a onepassword.Vault to the mock client.
  228. func (mockClient *OnePasswordMockClient) AppendVault(name string, vault onepassword.Vault) *OnePasswordMockClient {
  229. mockClient.MockVaults[name] = append(mockClient.MockVaults[name], vault)
  230. return mockClient
  231. }
  232. // AppendItem appends a onepassword.Item to the mock client.
  233. func (mockClient *OnePasswordMockClient) AppendItem(vaultID string, item onepassword.Item) *OnePasswordMockClient {
  234. mockClient.MockItems[vaultID] = append(mockClient.MockItems[vaultID], item)
  235. return mockClient
  236. }
  237. // AppendItemField appends a onepassword.ItemField to the mock client.
  238. func (mockClient *OnePasswordMockClient) AppendItemField(vaultID, itemID string, itemField onepassword.ItemField) *OnePasswordMockClient {
  239. if mockClient.MockItemFields[vaultID] == nil {
  240. mockClient.MockItemFields[vaultID] = make(map[string][]*onepassword.ItemField)
  241. }
  242. mockClient.MockItemFields[vaultID][itemID] = append(mockClient.MockItemFields[vaultID][itemID], &itemField)
  243. return mockClient
  244. }
  245. // SetFileContents adds file contents to the mock client.
  246. func (mockClient *OnePasswordMockClient) SetFileContents(name string, contents []byte) *OnePasswordMockClient {
  247. // no need to test or mock same file names in different vaults, because we only GetFileContent after findItem, which already tests getting the right item from the right vault
  248. mockClient.MockFileContents[name] = contents
  249. return mockClient
  250. }