subnet.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. // Copyright 2015 flannel authors
  2. //
  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. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package subnet
  15. import (
  16. "encoding/json"
  17. "errors"
  18. "fmt"
  19. "net"
  20. "regexp"
  21. "strconv"
  22. "time"
  23. "github.com/coreos/flannel/pkg/ip"
  24. "golang.org/x/net/context"
  25. )
  26. var (
  27. ErrLeaseTaken = errors.New("subnet: lease already taken")
  28. ErrNoMoreTries = errors.New("subnet: no more tries")
  29. subnetRegex *regexp.Regexp = regexp.MustCompile(`(\d+\.\d+.\d+.\d+)-(\d+)`)
  30. )
  31. type LeaseAttrs struct {
  32. PublicIP ip.IP4
  33. BackendType string `json:",omitempty"`
  34. BackendData json.RawMessage `json:",omitempty"`
  35. }
  36. type Lease struct {
  37. Subnet ip.IP4Net
  38. Attrs LeaseAttrs
  39. Expiration time.Time
  40. Asof uint64
  41. }
  42. func (l *Lease) Key() string {
  43. return MakeSubnetKey(l.Subnet)
  44. }
  45. type Reservation struct {
  46. Subnet ip.IP4Net
  47. PublicIP ip.IP4
  48. }
  49. type (
  50. EventType int
  51. Event struct {
  52. Type EventType `json:"type"`
  53. Lease Lease `json:"lease,omitempty"`
  54. Network string `json:"network,omitempty"`
  55. }
  56. )
  57. const (
  58. EventAdded EventType = iota
  59. EventRemoved
  60. )
  61. type LeaseWatchResult struct {
  62. // Either Events or Snapshot will be set. If Events is empty, it means
  63. // the cursor was out of range and Snapshot contains the current list
  64. // of items, even if empty.
  65. Events []Event `json:"events"`
  66. Snapshot []Lease `json:"snapshot"`
  67. Cursor interface{} `json:"cursor"`
  68. }
  69. type NetworkWatchResult struct {
  70. // Either Events or Snapshot will be set. If Events is empty, it means
  71. // the cursor was out of range and Snapshot contains the current list
  72. // of items, even if empty.
  73. Events []Event `json:"events"`
  74. Snapshot []string `json:"snapshot"`
  75. Cursor interface{} `json:"cursor,omitempty"`
  76. }
  77. func (et EventType) MarshalJSON() ([]byte, error) {
  78. s := ""
  79. switch et {
  80. case EventAdded:
  81. s = "added"
  82. case EventRemoved:
  83. s = "removed"
  84. default:
  85. return nil, errors.New("bad event type")
  86. }
  87. return json.Marshal(s)
  88. }
  89. func (et *EventType) UnmarshalJSON(data []byte) error {
  90. switch string(data) {
  91. case "\"added\"":
  92. *et = EventAdded
  93. case "\"removed\"":
  94. *et = EventRemoved
  95. default:
  96. fmt.Println(string(data))
  97. return errors.New("bad event type")
  98. }
  99. return nil
  100. }
  101. func ParseSubnetKey(s string) *ip.IP4Net {
  102. if parts := subnetRegex.FindStringSubmatch(s); len(parts) == 3 {
  103. snIp := net.ParseIP(parts[1]).To4()
  104. prefixLen, err := strconv.ParseUint(parts[2], 10, 5)
  105. if snIp != nil && err == nil {
  106. return &ip.IP4Net{IP: ip.FromIP(snIp), PrefixLen: uint(prefixLen)}
  107. }
  108. }
  109. return nil
  110. }
  111. func MakeSubnetKey(sn ip.IP4Net) string {
  112. return sn.StringSep(".", "-")
  113. }
  114. type Manager interface {
  115. GetNetworkConfig(ctx context.Context, network string) (*Config, error)
  116. AcquireLease(ctx context.Context, network string, attrs *LeaseAttrs) (*Lease, error)
  117. RenewLease(ctx context.Context, network string, lease *Lease) error
  118. RevokeLease(ctx context.Context, network string, sn ip.IP4Net) error
  119. WatchLease(ctx context.Context, network string, sn ip.IP4Net, cursor interface{}) (LeaseWatchResult, error)
  120. WatchLeases(ctx context.Context, network string, cursor interface{}) (LeaseWatchResult, error)
  121. WatchNetworks(ctx context.Context, cursor interface{}) (NetworkWatchResult, error)
  122. AddReservation(ctx context.Context, network string, r *Reservation) error
  123. RemoveReservation(ctx context.Context, network string, subnet ip.IP4Net) error
  124. ListReservations(ctx context.Context, network string) ([]Reservation, error)
  125. }