image.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613
  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 azure
  15. import (
  16. "context"
  17. "fmt"
  18. "net/url"
  19. "strings"
  20. "time"
  21. "yunion.io/x/jsonutils"
  22. "yunion.io/x/log"
  23. "yunion.io/x/pkg/errors"
  24. "yunion.io/x/pkg/utils"
  25. "yunion.io/x/cloudmux/pkg/apis"
  26. api "yunion.io/x/cloudmux/pkg/apis/compute"
  27. "yunion.io/x/cloudmux/pkg/cloudprovider"
  28. "yunion.io/x/cloudmux/pkg/multicloud"
  29. )
  30. type ImageStatusType string
  31. const (
  32. ImageStatusCreating ImageStatusType = "Creating"
  33. ImageStatusAvailable ImageStatusType = "Succeeded"
  34. ImageStatusUnAvailable ImageStatusType = "UnAvailable"
  35. ImageStatusCreateFailed ImageStatusType = "CreateFailed"
  36. )
  37. type OperatingSystemStateTypes string
  38. type SubResource struct {
  39. ID string `json:"id,omitempty"`
  40. Name string `json:"name,omitempty"`
  41. Type string `json:"type,omitempty"`
  42. }
  43. type ImageOSDisk struct {
  44. OsType string `json:"osType,omitempty"`
  45. OsState string `json:"osState,omitempty"`
  46. Snapshot *SubResource `json:"snapshot,omitempty"`
  47. ManagedDisk *SubResource
  48. BlobURI string `json:"blobUri,omitempty"`
  49. Caching string `json:"caching,omitempty"`
  50. DiskSizeGB int32 `json:"diskSizeGB,omitzero"`
  51. StorageAccountType string `json:"storageAccountType,omitempty"`
  52. OperatingSystem string `json:"operatingSystem,omitempty"`
  53. }
  54. type ImageDataDisk struct {
  55. Lun int32
  56. Snapshot SubResource
  57. ManagedDisk SubResource
  58. BlobURI string
  59. Caching string
  60. DiskSizeGB int32 `json:"diskSizeGB,omitzero"`
  61. StorageAccountType string
  62. }
  63. type ImageStorageProfile struct {
  64. OsDisk ImageOSDisk `json:"osDisk,omitempty"`
  65. DataDisks []ImageDataDisk `json:"dataDisks,omitempty"`
  66. ZoneResilient bool `json:"zoneResilient,omitfalse"`
  67. }
  68. type SAutomaticOSUpgradeProperties struct {
  69. AutomaticOSUpgradeSupported bool
  70. }
  71. type ImageProperties struct {
  72. SourceVirtualMachine *SubResource
  73. StorageProfile ImageStorageProfile `json:"storageProfile,omitempty"`
  74. ProvisioningState ImageStatusType
  75. HyperVGeneration string `json:"hyperVGeneration,omitempty"`
  76. }
  77. type SImage struct {
  78. multicloud.SImageBase
  79. AzureTags
  80. storageCache *SStoragecache
  81. Properties ImageProperties `json:"properties,omitempty"`
  82. ID string `json:"id,omitempty"`
  83. Name string
  84. Type string
  85. Location string
  86. Publisher string
  87. Offer string
  88. Sku string
  89. Version string
  90. ImageType cloudprovider.TImageType
  91. }
  92. func (self *SImage) GetMinRamSizeMb() int {
  93. return 0
  94. }
  95. func (self *SImage) GetSysTags() map[string]string {
  96. data := map[string]string{}
  97. osType := string(self.Properties.StorageProfile.OsDisk.OsType)
  98. if len(osType) > 0 {
  99. data["os_name"] = osType
  100. }
  101. return data
  102. }
  103. func (self *SImage) GetId() string {
  104. return self.ID
  105. }
  106. func (self *SImage) GetName() string {
  107. return self.Name
  108. }
  109. func (self *SImage) GetGlobalId() string {
  110. return strings.ToLower(self.ID)
  111. }
  112. func (self *SImage) GetStatus() string {
  113. switch self.Properties.ProvisioningState {
  114. case "created":
  115. return api.CACHED_IMAGE_STATUS_CACHING
  116. case "Succeeded":
  117. return api.CACHED_IMAGE_STATUS_ACTIVE
  118. default:
  119. log.Errorf("Unknow image status: %s", self.Properties.ProvisioningState)
  120. return api.CACHED_IMAGE_STATUS_CACHE_FAILED
  121. }
  122. }
  123. func (self *SImage) GetImageStatus() string {
  124. switch self.Properties.ProvisioningState {
  125. case "created":
  126. return cloudprovider.IMAGE_STATUS_QUEUED
  127. case "Succeeded":
  128. return cloudprovider.IMAGE_STATUS_ACTIVE
  129. default:
  130. log.Errorf("Unknow image status: %s", self.Properties.ProvisioningState)
  131. return cloudprovider.IMAGE_STATUS_KILLED
  132. }
  133. }
  134. func (self *SImage) Refresh() error {
  135. image, err := self.storageCache.region.GetImageById(self.ID)
  136. if err != nil {
  137. return err
  138. }
  139. return jsonutils.Update(self, image)
  140. }
  141. func (self *SImage) GetImageType() cloudprovider.TImageType {
  142. return cloudprovider.TImageType(self.ImageType)
  143. }
  144. func (self *SImage) GetSizeByte() int64 {
  145. return int64(self.Properties.StorageProfile.OsDisk.DiskSizeGB) * 1024 * 1024 * 1024
  146. }
  147. func (i *SImage) GetFullOsName() string {
  148. return ""
  149. }
  150. func (self *SImage) GetOsType() cloudprovider.TOsType {
  151. osType := self.Properties.StorageProfile.OsDisk.OsType
  152. if len(osType) == 0 {
  153. osType = publisherGetOsType(self.Publisher)
  154. }
  155. return cloudprovider.TOsType(osType)
  156. }
  157. func (self *SImage) GetOsArch() string {
  158. if self.GetImageType() == cloudprovider.ImageTypeCustomized {
  159. return apis.OS_ARCH_X86_64
  160. }
  161. return publisherGetOsArch(self.Publisher, self.Offer, self.Sku, self.Version)
  162. }
  163. func (self *SImage) GetOsDist() string {
  164. if self.GetImageType() == cloudprovider.ImageTypeCustomized {
  165. return ""
  166. }
  167. return publisherGetOsDist(self.Publisher, self.Offer, self.Sku, self.Version)
  168. }
  169. func (self *SImage) GetOsVersion() string {
  170. return publisherGetOsVersion(self.Publisher, self.Offer, self.Sku, self.Version)
  171. }
  172. func (self *SImage) GetOsLang() string {
  173. return ""
  174. }
  175. func (i *SImage) GetBios() cloudprovider.TBiosType {
  176. if i.Properties.HyperVGeneration == "V2" {
  177. return cloudprovider.UEFI
  178. } else {
  179. return cloudprovider.BIOS
  180. }
  181. }
  182. func (self *SImage) GetMinOsDiskSizeGb() int {
  183. if self.Properties.StorageProfile.OsDisk.DiskSizeGB > 0 {
  184. return int(self.Properties.StorageProfile.OsDisk.DiskSizeGB)
  185. }
  186. return 30
  187. }
  188. func (self *SImage) GetImageFormat() string {
  189. return "vhd"
  190. }
  191. func (self *SImage) GetCreatedAt() time.Time {
  192. return time.Time{}
  193. }
  194. func (self *SImage) GetIStoragecache() cloudprovider.ICloudStoragecache {
  195. return self.storageCache
  196. }
  197. func (self *SRegion) GetImageStatus(imageId string) (ImageStatusType, error) {
  198. if image, err := self.GetImageById(imageId); err != nil {
  199. return "", err
  200. } else {
  201. return image.Properties.ProvisioningState, nil
  202. }
  203. }
  204. func isPrivateImageID(imageId string) bool {
  205. return strings.HasPrefix(strings.ToLower(imageId), "/subscriptions/")
  206. }
  207. func (self *SRegion) GetImageById(imageId string) (SImage, error) {
  208. if isPrivateImageID(imageId) {
  209. return self.getPrivateImage(imageId)
  210. } else {
  211. return self.getOfferedImage(imageId)
  212. }
  213. }
  214. func (self *SRegion) getPrivateImage(imageId string) (SImage, error) {
  215. image := SImage{}
  216. err := self.get(imageId, url.Values{}, &image)
  217. if err != nil {
  218. return image, err
  219. }
  220. return image, nil
  221. }
  222. func (self *SRegion) CreateImageByBlob(imageName, osType, blobURI string, diskSizeGB int32) (*SImage, error) {
  223. if diskSizeGB < 1 || diskSizeGB > 4095 {
  224. diskSizeGB = 30
  225. }
  226. image := SImage{
  227. Name: imageName,
  228. Location: self.Name,
  229. Properties: ImageProperties{
  230. StorageProfile: ImageStorageProfile{
  231. OsDisk: ImageOSDisk{
  232. OsType: osType,
  233. OsState: "Generalized",
  234. BlobURI: blobURI,
  235. DiskSizeGB: diskSizeGB,
  236. },
  237. },
  238. },
  239. Type: "Microsoft.Compute/images",
  240. }
  241. return &image, self.create("", jsonutils.Marshal(image), &image)
  242. }
  243. func (self *SRegion) CreateImage(snapshotId, imageName, osType, imageDesc string) (*SImage, error) {
  244. image := SImage{
  245. Name: imageName,
  246. Location: self.Name,
  247. Properties: ImageProperties{
  248. StorageProfile: ImageStorageProfile{
  249. OsDisk: ImageOSDisk{
  250. OsType: osType,
  251. OsState: "Generalized",
  252. Snapshot: &SubResource{
  253. ID: snapshotId,
  254. },
  255. },
  256. },
  257. },
  258. Type: "Microsoft.Compute/images",
  259. }
  260. return &image, self.create("", jsonutils.Marshal(image), &image)
  261. }
  262. func (self *SRegion) getOfferedImages(publishersFilter []string, offersFilter []string, skusFilter []string, verFilter []string, imageType cloudprovider.TImageType, latestVer bool) ([]SImage, error) {
  263. images := make([]SImage, 0)
  264. idMap, err := self.GetOfferedImageIDs(publishersFilter, offersFilter, skusFilter, verFilter, latestVer)
  265. if err != nil {
  266. return nil, err
  267. }
  268. for id, _image := range idMap {
  269. image, err := self.getOfferedImage(id)
  270. if err == nil {
  271. image.ImageType = imageType
  272. image.Properties.StorageProfile.OsDisk.DiskSizeGB = int32(_image.Properties.OsDiskImage.SizeInGb)
  273. image.Properties.StorageProfile.OsDisk.OsType = _image.Properties.OsDiskImage.OperatingSystem
  274. image.Properties.HyperVGeneration = _image.Properties.HyperVGeneration
  275. images = append(images, image)
  276. }
  277. }
  278. return images, nil
  279. }
  280. func (self *SRegion) GetOfferedImageIDs(publishersFilter []string, offersFilter []string, skusFilter []string, verFilter []string, latestVer bool) (map[string]SAzureImageResource, error) {
  281. idMap := map[string]SAzureImageResource{}
  282. publishers, err := self.GetImagePublishers(toLowerStringArray(publishersFilter))
  283. if err != nil {
  284. return nil, err
  285. }
  286. for _, publisher := range publishers {
  287. offers, err := self.getImageOffers(publisher, toLowerStringArray(offersFilter))
  288. if err != nil {
  289. log.Errorf("failed to found offers for publisher %s error: %v", publisher, err)
  290. if errors.Cause(err) != cloudprovider.ErrNotFound {
  291. return nil, errors.Wrap(err, "getImageOffers")
  292. }
  293. continue
  294. }
  295. for _, offer := range offers {
  296. skus, err := self.getImageSkus(publisher, offer, toLowerStringArray(skusFilter))
  297. if err != nil {
  298. if errors.Cause(err) != cloudprovider.ErrNotFound {
  299. return nil, errors.Wrap(err, "getImageSkus")
  300. }
  301. log.Errorf("failed to found skus for publisher %s offer %s error: %v", publisher, offer, err)
  302. continue
  303. }
  304. for _, sku := range skus {
  305. verFilter = toLowerStringArray(verFilter)
  306. vers, err := self.getImageVersions(publisher, offer, sku, verFilter, latestVer)
  307. if err != nil {
  308. if errors.Cause(err) != cloudprovider.ErrNotFound {
  309. return nil, errors.Wrap(err, "getImageVersions")
  310. }
  311. log.Errorf("failed to found publisher %s offer %s sku %s version error: %v", publisher, offer, sku, err)
  312. continue
  313. }
  314. for _, ver := range vers {
  315. idStr := strings.Join([]string{publisher, offer, sku, ver}, "/")
  316. image, err := self.getImageDetail(publisher, offer, sku, ver)
  317. if err != nil {
  318. return nil, err
  319. }
  320. idMap[idStr] = image
  321. }
  322. }
  323. }
  324. }
  325. return idMap, nil
  326. }
  327. func (self *SRegion) getPrivateImages() ([]SImage, error) {
  328. result := []SImage{}
  329. images := []SImage{}
  330. err := self.client.list("Microsoft.Compute/images", url.Values{}, &images)
  331. if err != nil {
  332. return nil, err
  333. }
  334. for i := 0; i < len(images); i++ {
  335. if images[i].Location == self.Name {
  336. images[i].ImageType = cloudprovider.ImageTypeCustomized
  337. result = append(result, images[i])
  338. }
  339. }
  340. return result, nil
  341. }
  342. func toLowerStringArray(input []string) []string {
  343. output := make([]string, len(input))
  344. for i := range input {
  345. output[i] = strings.ToLower(input[i])
  346. }
  347. return output
  348. }
  349. func (self *SRegion) GetImages(imageType cloudprovider.TImageType) ([]SImage, error) {
  350. images := make([]SImage, 0)
  351. if len(imageType) == 0 {
  352. ret, _ := self.getPrivateImages()
  353. if len(ret) > 0 {
  354. images = append(images, ret...)
  355. }
  356. ret, _ = self.getOfferedImages(knownPublishers, nil, nil, nil, cloudprovider.ImageTypeSystem, true)
  357. if len(ret) > 0 {
  358. images = append(images, ret...)
  359. }
  360. return images, nil
  361. }
  362. switch imageType {
  363. case cloudprovider.ImageTypeCustomized:
  364. return self.getPrivateImages()
  365. case cloudprovider.ImageTypeSystem:
  366. return self.getOfferedImages(knownPublishers, nil, nil, nil, cloudprovider.ImageTypeSystem, true)
  367. default:
  368. return self.getOfferedImages(nil, nil, nil, nil, cloudprovider.ImageTypeMarket, true)
  369. }
  370. }
  371. func (self *SRegion) DeleteImage(imageId string) error {
  372. return self.del(imageId)
  373. }
  374. func (self *SImage) GetBlobUri() string {
  375. return self.Properties.StorageProfile.OsDisk.BlobURI
  376. }
  377. func (self *SImage) Delete(ctx context.Context) error {
  378. return self.storageCache.region.DeleteImage(self.ID)
  379. }
  380. type SOsDiskImage struct {
  381. OperatingSystem string `json:"operatingSystem"`
  382. SizeInGb int `json:"sizeInGb"`
  383. }
  384. type SAzureImageResourceProperties struct {
  385. ReplicaType string `json:"replicaType"`
  386. OsDiskImage SOsDiskImage `json:"osDiskImage"`
  387. HyperVGeneration string `json:"hyperVGeneration,omitempty"`
  388. }
  389. type SAzureImageResource struct {
  390. Id string
  391. Name string
  392. Location string
  393. Properties SAzureImageResourceProperties
  394. }
  395. func (region *SRegion) GetImagePublishers(filter []string) ([]string, error) {
  396. publishers := make([]SAzureImageResource, 0)
  397. // TODO
  398. err := region.client.list(fmt.Sprintf("Microsoft.Compute/locations/%s/publishers", region.Name), url.Values{}, &publishers)
  399. if err != nil {
  400. return nil, err
  401. }
  402. ret := make([]string, 0)
  403. for i := range publishers {
  404. if len(filter) == 0 || utils.IsInStringArray(strings.ToLower(publishers[i].Name), filter) {
  405. ret = append(ret, publishers[i].Name)
  406. }
  407. }
  408. return ret, nil
  409. }
  410. func (region *SRegion) getImageOffers(publisher string, filter []string) ([]string, error) {
  411. ret := make([]string, 0)
  412. if driver, ok := publisherDrivers[strings.ToLower(publisher)]; ok {
  413. offers := driver.GetOffers()
  414. if len(offers) > 0 {
  415. for _, offer := range offers {
  416. if len(filter) == 0 || utils.IsInStringArray(strings.ToLower(offer), filter) {
  417. ret = append(ret, offer)
  418. }
  419. }
  420. return offers, nil
  421. }
  422. } else {
  423. log.Warningf("failed to get publisher %s driver", publisher)
  424. }
  425. offers := make([]SAzureImageResource, 0)
  426. err := region.client.list(fmt.Sprintf("Microsoft.Compute/locations/%s/publishers/%s/artifacttypes/vmimage/offers", region.Name, publisher), url.Values{}, &offers)
  427. if err != nil {
  428. return nil, err
  429. }
  430. for i := range offers {
  431. if len(filter) == 0 || utils.IsInStringArray(strings.ToLower(offers[i].Name), filter) {
  432. ret = append(ret, offers[i].Name)
  433. }
  434. }
  435. return ret, nil
  436. }
  437. func (region *SRegion) getImageSkus(publisher string, offer string, filter []string) ([]string, error) {
  438. ret := make([]string, 0)
  439. if driver, ok := publisherDrivers[strings.ToLower(publisher)]; ok {
  440. skus := driver.GetSkus(offer)
  441. if len(skus) > 0 {
  442. for _, sku := range skus {
  443. if len(filter) == 0 || utils.IsInStringArray(strings.ToLower(sku), filter) {
  444. ret = append(ret, sku)
  445. }
  446. }
  447. return ret, nil
  448. }
  449. }
  450. skus := make([]SAzureImageResource, 0)
  451. err := region.client.list(fmt.Sprintf("Microsoft.Compute/locations/%s/publishers/%s/artifacttypes/vmimage/offers/%s/skus", region.Name, publisher, offer), url.Values{}, &skus)
  452. if err != nil {
  453. return nil, err
  454. }
  455. for i := range skus {
  456. if len(filter) == 0 || utils.IsInStringArray(strings.ToLower(skus[i].Name), filter) {
  457. ret = append(ret, skus[i].Name)
  458. }
  459. }
  460. return ret, nil
  461. }
  462. func (region *SRegion) getImageVersions(publisher string, offer string, sku string, filter []string, latestVer bool) ([]string, error) {
  463. vers := make([]SAzureImageResource, 0)
  464. resource := fmt.Sprintf("Microsoft.Compute/locations/%s/publishers/%s/artifacttypes/vmimage/offers/%s/skus/%s/versions", region.Name, publisher, offer, sku)
  465. params := url.Values{}
  466. if latestVer {
  467. params.Set("$top", "1")
  468. params.Set("orderby", "name desc")
  469. }
  470. err := region.client.list(resource, params, &vers)
  471. if err != nil {
  472. return nil, err
  473. }
  474. ret := make([]string, 0)
  475. for i := range vers {
  476. if len(filter) == 0 || utils.IsInStringArray(strings.ToLower(vers[i].Name), filter) {
  477. ret = append(ret, vers[i].Name)
  478. }
  479. }
  480. return ret, nil
  481. }
  482. func (region *SRegion) getImageDetail(publisher string, offer string, sku string, version string) (SAzureImageResource, error) {
  483. image := SAzureImageResource{}
  484. id := "/Subscriptions/" + region.client.subscriptionId +
  485. "/Providers/Microsoft.Compute/locations/" + region.Name +
  486. "/publishers/" + publisher +
  487. "/artifacttypes/vmimage/offers/" + offer +
  488. "/skus/" + sku +
  489. "/versions/" + version
  490. return image, region.get(id, url.Values{}, &image)
  491. }
  492. func (region *SRegion) getOfferedImage(offerId string) (SImage, error) {
  493. image := SImage{}
  494. parts := strings.Split(offerId, "/")
  495. if len(parts) < 4 {
  496. return image, fmt.Errorf("invalid image ID %s", offerId)
  497. }
  498. publisher := parts[0]
  499. offer := parts[1]
  500. sku := parts[2]
  501. version := parts[3]
  502. for _publish := range publisherDrivers {
  503. if strings.ToLower(_publish) == publisher {
  504. publisher = _publish
  505. break
  506. }
  507. }
  508. image.ID = offerId
  509. image.Location = region.Name
  510. image.Type = "Microsoft.Compute/vmimage"
  511. image.Name = publisherGetName(publisher, offer, sku, version)
  512. image.Publisher = publisher
  513. image.Offer = offer
  514. image.Sku = sku
  515. image.Version = version
  516. image.Properties.ProvisioningState = ImageStatusAvailable
  517. _image, err := region.getImageDetail(publisher, offer, sku, version)
  518. if err == nil {
  519. image.Properties.StorageProfile.OsDisk.DiskSizeGB = int32(_image.Properties.OsDiskImage.SizeInGb)
  520. image.Properties.StorageProfile.OsDisk.OperatingSystem = _image.Properties.OsDiskImage.OperatingSystem
  521. image.Properties.HyperVGeneration = _image.Properties.HyperVGeneration
  522. }
  523. return image, nil
  524. }
  525. func (region *SRegion) getOfferedImageId(image *SImage) (string, error) {
  526. if isPrivateImageID(image.ID) {
  527. return image.ID, nil
  528. }
  529. _image, err := region.getImageDetail(image.Publisher, image.Offer, image.Sku, image.Version)
  530. if err != nil {
  531. log.Errorf("failed to get offered image ID from %s error: %v", jsonutils.Marshal(image).PrettyString(), err)
  532. return "", err
  533. }
  534. return _image.Id, nil
  535. }
  536. func (image *SImage) getImageReference() ImageReference {
  537. if isPrivateImageID(image.ID) {
  538. return ImageReference{
  539. ID: image.ID,
  540. }
  541. } else {
  542. return ImageReference{
  543. Sku: image.Sku,
  544. Publisher: image.Publisher,
  545. Version: image.Version,
  546. Offer: image.Offer,
  547. }
  548. }
  549. }