iface_peers.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. // Copyright 2019 Yunion
  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 models
  15. import (
  16. "context"
  17. "fmt"
  18. "net"
  19. "strings"
  20. "yunion.io/x/log"
  21. yerrors "yunion.io/x/pkg/util/errors"
  22. "yunion.io/x/pkg/util/netutils"
  23. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  24. "yunion.io/x/onecloud/pkg/mcclient"
  25. )
  26. type SIfacePeer struct {
  27. db.SStandaloneResourceBase
  28. RouterId string
  29. IfaceId string
  30. PeerIfaceId string
  31. PeerRouterId string
  32. PublicKey string
  33. AllowedIPs string
  34. Endpoint string
  35. PersistentKeepalive int
  36. }
  37. type SIfacePeerManager struct {
  38. db.SStandaloneResourceBaseManager
  39. }
  40. var IfacePeerManager *SIfacePeerManager
  41. func init() {
  42. IfacePeerManager = &SIfacePeerManager{
  43. SStandaloneResourceBaseManager: db.NewStandaloneResourceBaseManager(
  44. SIfacePeer{},
  45. "ifacepeers_tbl",
  46. "ifacepeer",
  47. "ifacepeers",
  48. ),
  49. }
  50. IfacePeerManager.SetVirtualObject(IfacePeerManager)
  51. }
  52. func (ifacePeer *SIfacePeer) subnetsStrList() []string {
  53. return strings.Split(ifacePeer.AllowedIPs, ",")
  54. }
  55. func (ifacePeer *SIfacePeer) subnetsParsed() Subnets {
  56. parts := ifacePeer.subnetsStrList()
  57. r := make([]*netutils.IPV4Prefix, 0, len(parts))
  58. for _, part := range parts {
  59. p, err := netutils.NewIPV4Prefix(part)
  60. if err != nil {
  61. log.Errorf("%s: invalid subnet sneaked in: %s", ifacePeer.Id, part)
  62. return nil
  63. }
  64. r = append(r, &p)
  65. }
  66. return Subnets(r)
  67. }
  68. func (man *SIfacePeerManager) removeByPeerIface(ctx context.Context, userCred mcclient.TokenCredential, iface *SIface) error {
  69. peers := []SIfacePeer{}
  70. q := man.Query().Equals("peer_iface_id", iface.Id)
  71. if err := db.FetchModelObjects(IfacePeerManager, q, &peers); err != nil {
  72. return err
  73. }
  74. var errs []error
  75. for j := range peers {
  76. if err := peers[j].Delete(ctx, userCred); err != nil {
  77. errs = append(errs, err)
  78. }
  79. }
  80. return yerrors.NewAggregate(errs)
  81. }
  82. func (man *SIfacePeerManager) removeByIface(ctx context.Context, userCred mcclient.TokenCredential, iface *SIface) error {
  83. peers := []SIfacePeer{}
  84. q := man.Query().Equals("iface_id", iface.Id)
  85. if err := db.FetchModelObjects(IfacePeerManager, q, &peers); err != nil {
  86. return err
  87. }
  88. var errs []error
  89. for j := range peers {
  90. if err := peers[j].Delete(ctx, userCred); err != nil {
  91. errs = append(errs, err)
  92. }
  93. }
  94. return yerrors.NewAggregate(errs)
  95. }
  96. func (man *SIfacePeerManager) getByFilter(filter map[string]string) ([]SIfacePeer, error) {
  97. ifacePeers := []SIfacePeer{}
  98. q := man.Query()
  99. for key, val := range filter {
  100. q = q.Equals(key, val)
  101. }
  102. if err := db.FetchModelObjects(IfacePeerManager, q, &ifacePeers); err != nil {
  103. return nil, err
  104. }
  105. return ifacePeers, nil
  106. }
  107. func (man *SIfacePeerManager) getOneByFilter(filter map[string]string) (*SIfacePeer, error) {
  108. ifacePeers, err := man.getByFilter(filter)
  109. if err != nil {
  110. return nil, err
  111. }
  112. if len(ifacePeers) == 0 {
  113. return nil, errNotFound(fmt.Errorf("cannot find iface peer: %#v", filter))
  114. }
  115. if len(ifacePeers) > 1 {
  116. return nil, errMoreThanOne(fmt.Errorf("found more than 1 iface peers: %#v", filter))
  117. }
  118. return &ifacePeers[0], nil
  119. }
  120. func (man *SIfacePeerManager) getByIface(iface *SIface) ([]SIfacePeer, error) {
  121. filter := map[string]string{
  122. "router_id": iface.RouterId,
  123. "iface_id": iface.Id,
  124. }
  125. ifacePeers, err := man.getByFilter(filter)
  126. if err != nil {
  127. return nil, err
  128. }
  129. return ifacePeers, nil
  130. }
  131. func (man *SIfacePeerManager) getByIfacePublicKey(iface *SIface, pubkey string) (*SIfacePeer, error) {
  132. filter := map[string]string{
  133. "router_id": iface.RouterId,
  134. "iface_id": iface.Id,
  135. "public_key": pubkey,
  136. }
  137. ifacePeer, err := man.getOneByFilter(filter)
  138. if err != nil {
  139. return nil, err
  140. }
  141. return ifacePeer, nil
  142. }
  143. func (man *SIfacePeerManager) checkAllowedIPs(iface *SIface, oldPeer *SIfacePeer, allowedNets Subnets) error {
  144. ifacePeers, err := man.getByIface(iface)
  145. if err != nil {
  146. return err
  147. }
  148. for i := range ifacePeers {
  149. ifacePeer := &ifacePeers[i]
  150. if oldPeer != nil && oldPeer.Id == ifacePeer.Id {
  151. continue
  152. }
  153. existingNets := ifacePeer.subnetsParsed()
  154. if _, net := existingNets.ContainsAnyEx(allowedNets); net != nil {
  155. return fmt.Errorf("subnet %s is already occupied by peer %s(%s)",
  156. net, ifacePeer.Name, ifacePeer.Id)
  157. }
  158. }
  159. return nil
  160. }
  161. func (man *SIfacePeerManager) updateEndpointIPByPeerRouter(ctx context.Context, endpointIP string, router *SRouter) error {
  162. filter := map[string]string{
  163. "peer_router_id": router.Id,
  164. }
  165. ifacePeers, err := man.getByFilter(filter)
  166. if err != nil {
  167. return err
  168. }
  169. var errs []error
  170. for i := range ifacePeers {
  171. ifacePeer := &ifacePeers[i]
  172. host, port, err := net.SplitHostPort(ifacePeer.Endpoint)
  173. if err != nil {
  174. errs = append(errs, err)
  175. continue
  176. }
  177. if host == endpointIP {
  178. continue
  179. }
  180. _, err = db.Update(ifacePeer, func() error {
  181. endpoint := net.JoinHostPort(endpointIP, port)
  182. ifacePeer.Endpoint = endpoint
  183. return nil
  184. })
  185. if err != nil {
  186. errs = append(errs, err)
  187. }
  188. }
  189. return yerrors.NewAggregate(errs)
  190. }