verify.go 3.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. /*
  2. Copyright 2018 The Knative Authors
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package duck
  14. import (
  15. "encoding/json"
  16. "fmt"
  17. "github.com/google/go-cmp/cmp"
  18. )
  19. // Implementable is implemented by the Fooable duck type that consumers
  20. // are expected to embed as a `.status.fooable` field.
  21. type Implementable interface {
  22. // GetFullType returns an instance of a full resource wrapping
  23. // an instance of this Implementable that can populate its fields
  24. // to verify json roundtripping.
  25. GetFullType() Populatable
  26. }
  27. // Populatable is implemented by a skeleton resource wrapping an Implementable
  28. // duck type. It will generally have TypeMeta, ObjectMeta, and a Status field
  29. // wrapping a Fooable field.
  30. type Populatable interface {
  31. // Populate fills in all possible fields, so that we can verify that
  32. // they roundtrip properly through JSON.
  33. Populate()
  34. }
  35. // VerifyType verifies that a particular concrete resource properly implements
  36. // the provided Implementable duck type. It is expected that under the resource
  37. // definition implementing a particular "Fooable" that one would write:
  38. //
  39. // type ConcreteResource struct { ... }
  40. //
  41. // // Check that ConcreteResource properly implement Fooable.
  42. // err := duck.VerifyType(&ConcreteResource{}, &something.Fooable{})
  43. //
  44. // This will return an error if the duck typing is not satisfied.
  45. func VerifyType(instance interface{}, iface Implementable) error {
  46. // Create instances of the full resource for our input and ultimate result
  47. // that we will compare at the end.
  48. input, output := iface.GetFullType(), iface.GetFullType()
  49. // Populate our input resource with values we will roundtrip.
  50. input.Populate()
  51. // Serialize the input to JSON and deserialize that into the provided instance
  52. // of the type that we are checking.
  53. if before, err := json.Marshal(input); err != nil {
  54. return fmt.Errorf("error serializing duck type %T", input)
  55. } else if err := json.Unmarshal(before, instance); err != nil {
  56. return fmt.Errorf("error deserializing duck type %T into %T", input, instance)
  57. }
  58. // Serialize the instance we are checking to JSON and deserialize that into the
  59. // output resource.
  60. if after, err := json.Marshal(instance); err != nil {
  61. return fmt.Errorf("error serializing %T", instance)
  62. } else if err := json.Unmarshal(after, output); err != nil {
  63. return fmt.Errorf("error deserializing %T into dock type %T", instance, output)
  64. }
  65. // Now verify that we were able to roundtrip all of our fields through the type
  66. // we are checking.
  67. if diff := cmp.Diff(input, output); diff != "" {
  68. return fmt.Errorf("%T does not implement the duck type %T, the following fields were lost: %s",
  69. instance, iface, diff)
  70. }
  71. return nil
  72. }