region.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592
  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 ucloud
  15. import (
  16. "fmt"
  17. "strings"
  18. "yunion.io/x/pkg/errors"
  19. api "yunion.io/x/cloudmux/pkg/apis/compute"
  20. "yunion.io/x/cloudmux/pkg/cloudprovider"
  21. "yunion.io/x/cloudmux/pkg/multicloud"
  22. )
  23. type SRegion struct {
  24. multicloud.SRegion
  25. multicloud.SNoLbRegion
  26. client *SUcloudClient
  27. RegionID string
  28. izones []cloudprovider.ICloudZone
  29. ivpcs []cloudprovider.ICloudVpc
  30. storageCache *SStoragecache
  31. latitude float64
  32. longitude float64
  33. fetchLocation bool
  34. }
  35. func (self *SRegion) GetId() string {
  36. return self.RegionID
  37. }
  38. func (self *SRegion) GetName() string {
  39. if name, exist := UCLOUD_REGION_NAMES[self.GetId()]; exist {
  40. return fmt.Sprintf("%s %s", CLOUD_PROVIDER_UCLOUD_CN, name)
  41. }
  42. return fmt.Sprintf("%s %s", CLOUD_PROVIDER_UCLOUD_CN, self.GetId())
  43. }
  44. func (self *SRegion) GetI18n() cloudprovider.SModelI18nTable {
  45. var en string
  46. if name, exist := UCLOUD_REGION_NAMES_EN[self.GetId()]; exist {
  47. en = fmt.Sprintf("%s %s", CLOUD_PROVIDER_UCLOUD, name)
  48. } else {
  49. en = fmt.Sprintf("%s %s", CLOUD_PROVIDER_UCLOUD, self.GetId())
  50. }
  51. table := cloudprovider.SModelI18nTable{}
  52. table["name"] = cloudprovider.NewSModelI18nEntry(self.GetName()).CN(self.GetName()).EN(en)
  53. return table
  54. }
  55. func (self *SRegion) GetGlobalId() string {
  56. return fmt.Sprintf("%s/%s", CLOUD_PROVIDER_UCLOUD, self.GetId())
  57. }
  58. func (self *SRegion) GetStatus() string {
  59. return api.CLOUD_REGION_STATUS_INSERVER
  60. }
  61. func (self *SRegion) Refresh() error {
  62. return nil
  63. }
  64. func (self *SRegion) IsEmulated() bool {
  65. return false
  66. }
  67. func (self *SRegion) GetGeographicInfo() cloudprovider.SGeographicInfo {
  68. if info, ok := LatitudeAndLongitude[self.GetId()]; ok {
  69. return info
  70. }
  71. return cloudprovider.SGeographicInfo{}
  72. }
  73. func (self *SRegion) GetIVMById(id string) (cloudprovider.ICloudVM, error) {
  74. instance, err := self.GetInstanceByID(id)
  75. if err != nil {
  76. return nil, err
  77. }
  78. return &instance, nil
  79. }
  80. func (self *SRegion) GetIDiskById(id string) (cloudprovider.ICloudDisk, error) {
  81. return self.GetDisk(id)
  82. }
  83. func (self *SRegion) GetIZones() ([]cloudprovider.ICloudZone, error) {
  84. if self.izones == nil {
  85. var err error
  86. err = self.fetchInfrastructure()
  87. if err != nil {
  88. return nil, err
  89. }
  90. }
  91. return self.izones, nil
  92. }
  93. func (self *SRegion) GetIVpcs() ([]cloudprovider.ICloudVpc, error) {
  94. if self.ivpcs == nil {
  95. err := self.fetchInfrastructure()
  96. if err != nil {
  97. return nil, err
  98. }
  99. }
  100. return self.ivpcs, nil
  101. }
  102. // https://docs.ucloud.cn/api/unet-api/describe_eip
  103. func (self *SRegion) GetIEips() ([]cloudprovider.ICloudEIP, error) {
  104. params := NewUcloudParams()
  105. eips := make([]SEip, 0)
  106. err := self.DoListAll("DescribeEIP", params, &eips)
  107. if err != nil {
  108. return nil, err
  109. }
  110. ieips := []cloudprovider.ICloudEIP{}
  111. for i := range eips {
  112. eip := eips[i]
  113. eip.region = self
  114. ieips = append(ieips, &eip)
  115. }
  116. return ieips, nil
  117. }
  118. func (self *SRegion) GetIVpcById(id string) (cloudprovider.ICloudVpc, error) {
  119. ivpcs, err := self.GetIVpcs()
  120. if err != nil {
  121. return nil, err
  122. }
  123. for i := 0; i < len(ivpcs); i += 1 {
  124. if ivpcs[i].GetGlobalId() == id {
  125. return ivpcs[i], nil
  126. }
  127. }
  128. return nil, cloudprovider.ErrNotFound
  129. }
  130. func (self *SRegion) GetIZoneById(id string) (cloudprovider.ICloudZone, error) {
  131. izones, err := self.GetIZones()
  132. if err != nil {
  133. return nil, err
  134. }
  135. for i := 0; i < len(izones); i += 1 {
  136. if izones[i].GetGlobalId() == id {
  137. return izones[i], nil
  138. }
  139. }
  140. return nil, cloudprovider.ErrNotFound
  141. }
  142. func (self *SRegion) GetEipById(eipId string) (SEip, error) {
  143. params := NewUcloudParams()
  144. params.Set("EIPIds.0", eipId)
  145. eips := make([]SEip, 0)
  146. err := self.DoListAll("DescribeEIP", params, &eips)
  147. if err != nil {
  148. return SEip{}, err
  149. }
  150. if len(eips) == 1 {
  151. eip := eips[0]
  152. eip.region = self
  153. return eip, nil
  154. } else if len(eips) == 0 {
  155. return SEip{}, cloudprovider.ErrNotFound
  156. } else {
  157. return SEip{}, fmt.Errorf("GetEipById %d eip found", len(eips))
  158. }
  159. }
  160. func (self *SRegion) GetIEipById(id string) (cloudprovider.ICloudEIP, error) {
  161. eip, err := self.GetEipById(id)
  162. return &eip, err
  163. }
  164. // https://docs.ucloud.cn/api/unet-api/delete_firewall
  165. func (self *SRegion) DeleteSecurityGroup(secgroupId string) error {
  166. params := NewUcloudParams()
  167. params.Set("FWId", secgroupId)
  168. return self.DoAction("DeleteFirewall", params, nil)
  169. }
  170. func (self *SRegion) GetISecurityGroups() ([]cloudprovider.ICloudSecurityGroup, error) {
  171. secgroups, err := self.GetSecurityGroups("", "", "")
  172. if err != nil {
  173. return nil, err
  174. }
  175. ret := []cloudprovider.ICloudSecurityGroup{}
  176. for i := 0; i < len(secgroups); i++ {
  177. secgroups[i].region = self
  178. ret = append(ret, &secgroups[i])
  179. }
  180. return ret, nil
  181. }
  182. func (self *SRegion) GetISecurityGroupById(secgroupId string) (cloudprovider.ICloudSecurityGroup, error) {
  183. return self.GetSecurityGroup(secgroupId)
  184. }
  185. func (self *SRegion) CreateISecurityGroup(opts *cloudprovider.SecurityGroupCreateInput) (cloudprovider.ICloudSecurityGroup, error) {
  186. externalId, err := self.CreateSecurityGroup(opts.Name, opts.Desc)
  187. if err != nil {
  188. return nil, err
  189. }
  190. return self.GetISecurityGroupById(externalId)
  191. }
  192. // https://docs.ucloud.cn/api/unet-api/describe_firewall
  193. // 绑定防火墙组的资源类型,默认为全部资源类型。枚举值为:"unatgw",NAT网关; "uhost",云主机; "upm",物理云主机; "hadoophost",hadoop节点; "fortresshost",堡垒机; "udhost",私有专区主机;"udockhost",容器;"dbaudit",数据库审计.
  194. // todo: 是否需要过滤出仅绑定云主机的安全组?
  195. func (self *SRegion) CreateIVpc(opts *cloudprovider.VpcCreateOptions) (cloudprovider.ICloudVpc, error) {
  196. params := NewUcloudParams()
  197. params.Set("Name", opts.NAME)
  198. params.Set("Remark", opts.Desc)
  199. for i, cidr := range strings.Split(opts.CIDR, ",") {
  200. params.Set(fmt.Sprintf("Network.%d", i), cidr)
  201. }
  202. vpcId := ""
  203. err := self.DoAction("CreateVPC", params, &vpcId)
  204. if err != nil {
  205. return nil, err
  206. }
  207. return self.GetIVpcById(vpcId)
  208. }
  209. // https://docs.ucloud.cn/api/udisk-api/describe_udisk_snapshot
  210. func (self *SRegion) GetISnapshots() ([]cloudprovider.ICloudSnapshot, error) {
  211. params := NewUcloudParams()
  212. snapshots := make([]SSnapshot, 0)
  213. err := self.DoListAll("DescribeUDiskSnapshot", params, &snapshots)
  214. if err != nil {
  215. return nil, err
  216. }
  217. isnapshots := make([]cloudprovider.ICloudSnapshot, 0)
  218. for i := range snapshots {
  219. snapshots[i].region = self
  220. isnapshots = append(isnapshots, &snapshots[i])
  221. }
  222. return isnapshots, nil
  223. }
  224. func (self *SRegion) GetISnapshotById(snapshotId string) (cloudprovider.ICloudSnapshot, error) {
  225. if len(snapshotId) == 0 {
  226. return nil, cloudprovider.ErrNotFound
  227. }
  228. params := NewUcloudParams()
  229. snapshots := make([]SSnapshot, 0)
  230. err := self.DoListAll("DescribeUDiskSnapshot", params, &snapshots)
  231. if err != nil {
  232. return nil, err
  233. }
  234. for i := range snapshots {
  235. if snapshots[i].SnapshotID == snapshotId {
  236. snapshot := snapshots[i]
  237. snapshot.region = self
  238. return &snapshot, nil
  239. }
  240. }
  241. return nil, cloudprovider.ErrNotFound
  242. }
  243. func (self *SRegion) GetIHosts() ([]cloudprovider.ICloudHost, error) {
  244. iHosts := make([]cloudprovider.ICloudHost, 0)
  245. izones, err := self.GetIZones()
  246. if err != nil {
  247. return nil, err
  248. }
  249. for i := 0; i < len(izones); i += 1 {
  250. iZoneHost, err := izones[i].GetIHosts()
  251. if err != nil {
  252. return nil, err
  253. }
  254. iHosts = append(iHosts, iZoneHost...)
  255. }
  256. return iHosts, nil
  257. }
  258. func (self *SRegion) GetIHostById(id string) (cloudprovider.ICloudHost, error) {
  259. izones, err := self.GetIZones()
  260. if err != nil {
  261. return nil, err
  262. }
  263. for i := 0; i < len(izones); i += 1 {
  264. ihost, err := izones[i].GetIHostById(id)
  265. if err == nil {
  266. return ihost, nil
  267. } else if errors.Cause(err) != cloudprovider.ErrNotFound {
  268. return nil, err
  269. }
  270. }
  271. return nil, cloudprovider.ErrNotFound
  272. }
  273. func (self *SRegion) GetIStorages() ([]cloudprovider.ICloudStorage, error) {
  274. iStores := make([]cloudprovider.ICloudStorage, 0)
  275. izones, err := self.GetIZones()
  276. if err != nil {
  277. return nil, err
  278. }
  279. for i := 0; i < len(izones); i += 1 {
  280. iZoneStores, err := izones[i].GetIStorages()
  281. if err != nil {
  282. return nil, err
  283. }
  284. iStores = append(iStores, iZoneStores...)
  285. }
  286. return iStores, nil
  287. }
  288. func (self *SRegion) GetIStorageById(id string) (cloudprovider.ICloudStorage, error) {
  289. izones, err := self.GetIZones()
  290. if err != nil {
  291. return nil, err
  292. }
  293. for i := 0; i < len(izones); i += 1 {
  294. istore, err := izones[i].GetIStorageById(id)
  295. if err == nil {
  296. return istore, nil
  297. } else if errors.Cause(err) != cloudprovider.ErrNotFound {
  298. return nil, err
  299. }
  300. }
  301. return nil, cloudprovider.ErrNotFound
  302. }
  303. func (self *SRegion) getStoragecache() *SStoragecache {
  304. if self.storageCache == nil {
  305. self.storageCache = &SStoragecache{region: self}
  306. }
  307. return self.storageCache
  308. }
  309. func (self *SRegion) GetIStoragecaches() ([]cloudprovider.ICloudStoragecache, error) {
  310. storageCache := self.getStoragecache()
  311. return []cloudprovider.ICloudStoragecache{storageCache}, nil
  312. }
  313. func (self *SRegion) GetIStoragecacheById(id string) (cloudprovider.ICloudStoragecache, error) {
  314. storageCache := self.getStoragecache()
  315. if storageCache.GetGlobalId() == id {
  316. return storageCache, nil
  317. }
  318. return nil, cloudprovider.ErrNotFound
  319. }
  320. func (self *SRegion) GetSkus(zoneId string) ([]cloudprovider.ICloudSku, error) {
  321. return nil, cloudprovider.ErrNotSupported
  322. }
  323. func (self *SRegion) GetProvider() string {
  324. return CLOUD_PROVIDER_UCLOUD
  325. }
  326. func (self *SRegion) GetCloudEnv() string {
  327. return ""
  328. }
  329. func (self *SRegion) DoListAll(action string, params SParams, result interface{}) error {
  330. params.Set("Region", self.GetId())
  331. return self.client.DoListAll(action, params, result)
  332. }
  333. // return total,lenght,error
  334. func (self *SRegion) DoListPart(action string, limit int, offset int, params SParams, result interface{}) (int, int, error) {
  335. params.Set("Region", self.GetId())
  336. return self.client.DoListPart(action, limit, offset, params, result)
  337. }
  338. func (self *SRegion) DoAction(action string, params SParams, result interface{}) error {
  339. params.Set("Region", self.GetId())
  340. return self.client.DoAction(action, params, result)
  341. }
  342. func (self *SRegion) fetchInfrastructure() error {
  343. if err := self.fetchZones(); err != nil {
  344. return err
  345. }
  346. if err := self.fetchIVpcs(); err != nil {
  347. return err
  348. }
  349. for i := 0; i < len(self.ivpcs); i += 1 {
  350. vpc := self.ivpcs[i].(*SVPC)
  351. wire := SWire{region: self, vpc: vpc}
  352. vpc.addWire(&wire)
  353. for j := 0; j < len(self.izones); j += 1 {
  354. zone := self.izones[j].(*SZone)
  355. zone.addWire(&wire)
  356. }
  357. }
  358. return nil
  359. }
  360. func (self *SRegion) fetchZones() error {
  361. type Region struct {
  362. RegionID int64 `json:"RegionId"`
  363. RegionName string `json:"RegionName"`
  364. IsDefault bool `json:"IsDefault"`
  365. BitMaps string `json:"BitMaps"`
  366. Region string `json:"Region"`
  367. Zone string `json:"Zone"`
  368. }
  369. params := NewUcloudParams()
  370. regions := make([]Region, 0)
  371. err := self.client.DoListAll("GetRegion", params, &regions)
  372. if err != nil {
  373. return err
  374. }
  375. for _, r := range regions {
  376. if r.Region != self.GetId() {
  377. continue
  378. }
  379. szone := SZone{}
  380. szone.ZoneId = r.Zone
  381. szone.RegionId = r.Region
  382. szone.region = self
  383. self.izones = append(self.izones, &szone)
  384. }
  385. return nil
  386. }
  387. func (self *SRegion) fetchIVpcs() error {
  388. vpcs := make([]SVPC, 0)
  389. params := NewUcloudParams()
  390. err := self.DoListAll("DescribeVPC", params, &vpcs)
  391. if err != nil {
  392. return err
  393. }
  394. for i := range vpcs {
  395. vpc := vpcs[i]
  396. vpc.region = self
  397. self.ivpcs = append(self.ivpcs, &vpc)
  398. }
  399. return nil
  400. }
  401. // https://docs.ucloud.cn/api/uhost-api/describe_uhost_instance
  402. func (self *SRegion) GetInstanceByID(instanceId string) (SInstance, error) {
  403. params := NewUcloudParams()
  404. params.Set("UHostIds.0", instanceId)
  405. instances := make([]SInstance, 0)
  406. err := self.DoAction("DescribeUHostInstance", params, &instances)
  407. if err != nil {
  408. return SInstance{}, err
  409. }
  410. if len(instances) == 1 {
  411. return instances[0], nil
  412. } else if len(instances) == 0 {
  413. return SInstance{}, cloudprovider.ErrNotFound
  414. } else {
  415. return SInstance{}, fmt.Errorf("GetInstanceByID %s %d found.", instanceId, len(instances))
  416. }
  417. }
  418. func (self *SRegion) GetClient() *SUcloudClient {
  419. return self.client
  420. }
  421. // https://docs.ucloud.cn/api/ufile-api/describe_bucket
  422. func (client *SUcloudClient) listBuckets(name string, offset int, limit int) ([]SBucket, error) {
  423. params := NewUcloudParams()
  424. if len(name) > 0 {
  425. params.Set("BucketName", name)
  426. } else {
  427. params.Set("Limit", limit)
  428. params.Set("Offset", offset)
  429. }
  430. buckets := make([]SBucket, 0)
  431. // request without RegionId
  432. err := client.DoAction("DescribeBucket", params, &buckets)
  433. if err != nil {
  434. return nil, errors.Wrap(err, "DoAction DescribeBucket")
  435. }
  436. return buckets, nil
  437. }
  438. // https://docs.ucloud.cn/api/ufile-api/update_bucket
  439. func (region *SRegion) updateBucket(name string, aclType string) error {
  440. params := NewUcloudParams()
  441. params.Set("BucketName", name)
  442. params.Set("ProjectId", region.client.projectId)
  443. params.Set("Type", aclType)
  444. return region.client.DoAction("UpdateBucket", params, nil)
  445. }
  446. func (region *SRegion) GetIBuckets() ([]cloudprovider.ICloudBucket, error) {
  447. iBuckets, err := region.client.getIBuckets()
  448. if err != nil {
  449. return nil, errors.Wrap(err, "getIBuckets")
  450. }
  451. ret := make([]cloudprovider.ICloudBucket, 0)
  452. for i := range iBuckets {
  453. if iBuckets[i].GetLocation() != region.GetId() {
  454. continue
  455. }
  456. ret = append(ret, iBuckets[i])
  457. }
  458. return ret, nil
  459. }
  460. func (region *SRegion) CreateIBucket(name string, storageClassStr string, aclStr string) error {
  461. if aclStr != "private" && aclStr != "public" {
  462. return errors.Error("invalid acl")
  463. }
  464. return region.CreateBucket(name, aclStr)
  465. }
  466. func (region *SRegion) DeleteIBucket(name string) error {
  467. err := region.DeleteBucket(name)
  468. if err != nil {
  469. if strings.Index(err.Error(), "bucket not found") >= 0 {
  470. return nil
  471. }
  472. return errors.Wrap(err, "region.DeleteBucket")
  473. }
  474. return nil
  475. }
  476. func (region *SRegion) IBucketExist(name string) (bool, error) {
  477. parts, err := region.client.listBuckets(name, 0, 1)
  478. if err != nil {
  479. return false, errors.Wrap(err, "region.listBuckets")
  480. }
  481. if len(parts) == 0 {
  482. return false, cloudprovider.ErrNotFound
  483. }
  484. return true, nil
  485. }
  486. func (region *SRegion) GetIBucketById(bucketId string) (cloudprovider.ICloudBucket, error) {
  487. return cloudprovider.GetIBucketById(region, bucketId)
  488. }
  489. func (region *SRegion) GetIBucketByName(name string) (cloudprovider.ICloudBucket, error) {
  490. return region.GetIBucketByName(name)
  491. }
  492. func (region *SRegion) GetCapabilities() []string {
  493. return region.client.GetCapabilities()
  494. }
  495. func (region *SRegion) GetIVMs() ([]cloudprovider.ICloudVM, error) {
  496. vms, err := region.GetInstances("", "")
  497. if err != nil {
  498. return nil, err
  499. }
  500. ret := []cloudprovider.ICloudVM{}
  501. for i := range vms {
  502. ret = append(ret, &vms[i])
  503. }
  504. return ret, nil
  505. }