sslcertificate.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  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. "time"
  18. "yunion.io/x/cloudmux/pkg/cloudprovider"
  19. "yunion.io/x/jsonutils"
  20. "yunion.io/x/pkg/errors"
  21. "yunion.io/x/pkg/util/compare"
  22. "yunion.io/x/sqlchemy"
  23. "yunion.io/x/onecloud/pkg/apis"
  24. api "yunion.io/x/onecloud/pkg/apis/compute"
  25. "yunion.io/x/onecloud/pkg/cloudcommon/db"
  26. "yunion.io/x/onecloud/pkg/cloudcommon/db/lockman"
  27. "yunion.io/x/onecloud/pkg/cloudcommon/db/taskman"
  28. "yunion.io/x/onecloud/pkg/cloudcommon/notifyclient"
  29. "yunion.io/x/onecloud/pkg/cloudcommon/validators"
  30. "yunion.io/x/onecloud/pkg/httperrors"
  31. "yunion.io/x/onecloud/pkg/mcclient"
  32. "yunion.io/x/onecloud/pkg/util/stringutils2"
  33. )
  34. // +onecloud:swagger-gen-model-singular=sslcertificate
  35. // +onecloud:swagger-gen-model-plural=sslcertificates
  36. type SSSLCertificateManager struct {
  37. db.SVirtualResourceBaseManager
  38. db.SExternalizedResourceBaseManager
  39. SDeletePreventableResourceBaseManager
  40. SManagedResourceBaseManager
  41. SDnsZoneResourceBaseManager
  42. }
  43. var SSLCertificateManager *SSSLCertificateManager
  44. func init() {
  45. SSLCertificateManager = &SSSLCertificateManager{
  46. SVirtualResourceBaseManager: db.NewVirtualResourceBaseManager(
  47. SSSLCertificate{},
  48. "sslcertificates_tbl",
  49. "sslcertificate",
  50. "sslcertificates",
  51. ),
  52. }
  53. SSLCertificateManager.SetVirtualObject(SSLCertificateManager)
  54. }
  55. type SSSLCertificate struct {
  56. db.SVirtualResourceBase
  57. db.SExternalizedResourceBase
  58. SManagedResourceBase
  59. SDnsZoneResourceBase
  60. SDeletePreventableResourceBase
  61. Sans string `width:"2048" charset:"utf8" nullable:"false" list:"user" create:"required"`
  62. StartDate time.Time `list:"user"`
  63. Province string `width:"64" charset:"utf8" nullable:"false" list:"user" create:"optional"`
  64. Common string `width:"128" charset:"utf8" nullable:"false" list:"user" create:"optional"`
  65. Country string `width:"32" charset:"utf8" nullable:"false" list:"user" create:"optional"`
  66. Issuer string `width:"128" charset:"utf8" nullable:"false" list:"user" create:"required"`
  67. IsUpload bool `list:"user" create:"optional"`
  68. EndDate time.Time `list:"user" create:"optional"`
  69. Fingerprint string `width:"128" charset:"utf8" nullable:"false" list:"user" create:"optional"`
  70. City string `width:"64" charset:"utf8" nullable:"false" list:"user" create:"optional"`
  71. OrgName string `width:"256" charset:"utf8" nullable:"false" list:"user" create:"optional"`
  72. Certificate string `charset:"utf8" nullable:"true" list:"user" create:"optional"`
  73. PrivateKey string `charset:"utf8" nullable:"true" list:"user" create:"optional"`
  74. }
  75. func (s SSSLCertificate) GetExternalId() string {
  76. return s.ExternalId
  77. }
  78. func (man *SSSLCertificateManager) FetchCustomizeColumns(
  79. ctx context.Context,
  80. userCred mcclient.TokenCredential,
  81. query jsonutils.JSONObject,
  82. objs []interface{},
  83. fields stringutils2.SSortedStrings,
  84. isList bool,
  85. ) []api.SSLCertificateDetails {
  86. rows := make([]api.SSLCertificateDetails, len(objs))
  87. virtRows := man.SVirtualResourceBaseManager.FetchCustomizeColumns(ctx, userCred, query, objs, fields, isList)
  88. manRows := man.SManagedResourceBaseManager.FetchCustomizeColumns(ctx, userCred, query, objs, fields, isList)
  89. for i := range rows {
  90. rows[i] = api.SSLCertificateDetails{
  91. VirtualResourceDetails: virtRows[i],
  92. ManagedResourceInfo: manRows[i],
  93. }
  94. cert := objs[i].(*SSSLCertificate)
  95. rows[i].IsExpired = cert.EndDate.Before(time.Now())
  96. }
  97. return rows
  98. }
  99. func (man *SSSLCertificateManager) GetContextManagers() [][]db.IModelManager {
  100. return [][]db.IModelManager{
  101. {CloudproviderManager},
  102. }
  103. }
  104. // SSLCertificate实例列表
  105. func (man *SSSLCertificateManager) ListItemFilter(
  106. ctx context.Context,
  107. q *sqlchemy.SQuery,
  108. userCred mcclient.TokenCredential,
  109. query api.SSLCertificateListInput,
  110. ) (*sqlchemy.SQuery, error) {
  111. var err error
  112. q, err = man.SVirtualResourceBaseManager.ListItemFilter(ctx, q, userCred, query.VirtualResourceListInput)
  113. if err != nil {
  114. return nil, errors.Wrap(err, "SVirtualResourceBaseManager.ListItemFilter")
  115. }
  116. q, err = man.SExternalizedResourceBaseManager.ListItemFilter(ctx, q, userCred, query.ExternalizedResourceBaseListInput)
  117. if err != nil {
  118. return nil, errors.Wrap(err, "SExternalizedResourceBaseManager.ListItemFilter")
  119. }
  120. q, err = man.SDeletePreventableResourceBaseManager.ListItemFilter(ctx, q, userCred, query.DeletePreventableResourceBaseListInput)
  121. if err != nil {
  122. return nil, errors.Wrap(err, "SDeletePreventableResourceBaseManager.ListItemFilter")
  123. }
  124. q, err = man.SManagedResourceBaseManager.ListItemFilter(ctx, q, userCred, query.ManagedResourceListInput)
  125. if err != nil {
  126. return nil, errors.Wrap(err, "SManagedResourceBaseManager.ListItemFilter")
  127. }
  128. q, err = man.SDnsZoneResourceBaseManager.ListItemFilter(ctx, q, userCred, query.DnsZoneFilterListBase)
  129. if err != nil {
  130. return nil, errors.Wrap(err, "SDnsZoneResourceBaseManager.ListItemFilter")
  131. }
  132. if query.IsExpired != nil {
  133. if *query.IsExpired {
  134. q = q.LT("end_date", time.Now())
  135. } else {
  136. q = q.GE("end_date", time.Now())
  137. }
  138. }
  139. return q, nil
  140. }
  141. func (man *SSSLCertificateManager) OrderByExtraFields(
  142. ctx context.Context,
  143. q *sqlchemy.SQuery,
  144. userCred mcclient.TokenCredential,
  145. query api.SSLCertificateListInput,
  146. ) (*sqlchemy.SQuery, error) {
  147. q, err := man.SVirtualResourceBaseManager.OrderByExtraFields(ctx, q, userCred, query.VirtualResourceListInput)
  148. if err != nil {
  149. return nil, errors.Wrap(err, "SVirtualResourceBaseManager.OrderByExtraFields")
  150. }
  151. q, err = man.SManagedResourceBaseManager.OrderByExtraFields(ctx, q, userCred, query.ManagedResourceListInput)
  152. if err != nil {
  153. return nil, errors.Wrap(err, "SManagedResourceBaseManager.OrderByExtraFields")
  154. }
  155. return q, nil
  156. }
  157. func (man *SSSLCertificateManager) QueryDistinctExtraField(q *sqlchemy.SQuery, field string) (*sqlchemy.SQuery, error) {
  158. q, err := man.SVirtualResourceBaseManager.QueryDistinctExtraField(q, field)
  159. if err == nil {
  160. return q, nil
  161. }
  162. q, err = man.SManagedResourceBaseManager.QueryDistinctExtraField(q, field)
  163. if err == nil {
  164. return q, nil
  165. }
  166. q, err = man.SDnsZoneResourceBaseManager.QueryDistinctExtraField(q, field)
  167. if err == nil {
  168. return q, nil
  169. }
  170. return q, httperrors.ErrNotFound
  171. }
  172. func (manager *SSSLCertificateManager) QueryDistinctExtraFields(q *sqlchemy.SQuery, resource string, fields []string) (*sqlchemy.SQuery, error) {
  173. var err error
  174. q, err = manager.SManagedResourceBaseManager.QueryDistinctExtraFields(q, resource, fields)
  175. if err == nil {
  176. return q, nil
  177. }
  178. return q, httperrors.ErrNotFound
  179. }
  180. func (man *SSSLCertificateManager) ValidateCreateData(
  181. ctx context.Context,
  182. userCred mcclient.TokenCredential,
  183. ownerId mcclient.IIdentityProvider,
  184. query jsonutils.JSONObject,
  185. input *api.SSLCertificateCreateInput,
  186. ) (*api.SSLCertificateCreateInput, error) {
  187. if len(input.DnsZoneId) > 0 {
  188. obj, err := validators.ValidateModel(ctx, userCred, DnsZoneManager, &input.DnsZoneId)
  189. if err != nil {
  190. return nil, err
  191. }
  192. input.DnsZoneId = obj.GetId()
  193. }
  194. var err error
  195. input.VirtualResourceCreateInput, err = man.SVirtualResourceBaseManager.ValidateCreateData(ctx, userCred, ownerId, query, input.VirtualResourceCreateInput)
  196. if err != nil {
  197. return nil, err
  198. }
  199. return input, nil
  200. }
  201. func (self *SSSLCertificate) PostCreate(ctx context.Context, userCred mcclient.TokenCredential, ownerId mcclient.IIdentityProvider, query jsonutils.JSONObject, data jsonutils.JSONObject) {
  202. self.SVirtualResourceBase.PostCreate(ctx, userCred, ownerId, query, data)
  203. self.StartSSLCertificateCreateTask(ctx, userCred, "")
  204. }
  205. func (self *SSSLCertificate) StartSSLCertificateCreateTask(ctx context.Context, userCred mcclient.TokenCredential, parentTaskId string) error {
  206. params := jsonutils.NewDict()
  207. task, err := taskman.TaskManager.NewTask(ctx, "SSLCertificateCreateTask", self, userCred, params, parentTaskId, "", nil)
  208. if err != nil {
  209. return errors.Wrap(err, "NewTask")
  210. }
  211. self.SetStatus(ctx, userCred, apis.STATUS_CREATING, "")
  212. return task.ScheduleRun(nil)
  213. }
  214. func (self *SSSLCertificate) GetDnsZone() (*SDnsZone, error) {
  215. if len(self.DnsZoneId) == 0 {
  216. return nil, errors.Wrapf(cloudprovider.ErrNotFound, "DnsZoneId is empty")
  217. }
  218. zone, err := DnsZoneManager.FetchById(self.DnsZoneId)
  219. if err != nil {
  220. return nil, errors.Wrapf(err, "DnsZoneManager.FetchById")
  221. }
  222. return zone.(*SDnsZone), nil
  223. }
  224. func (r *SCloudprovider) GetSSLCertificates() ([]SSSLCertificate, error) {
  225. q := SSLCertificateManager.Query().Equals("manager_id", r.Id)
  226. ret := make([]SSSLCertificate, 0)
  227. err := db.FetchModelObjects(SSLCertificateManager, q, &ret)
  228. if err != nil {
  229. return nil, errors.Wrapf(err, "db.FetchModelObjects")
  230. }
  231. return ret, nil
  232. }
  233. func (r *SCloudprovider) SyncSSLCertificates(
  234. ctx context.Context,
  235. userCred mcclient.TokenCredential,
  236. exts []cloudprovider.ICloudSSLCertificate,
  237. ) compare.SyncResult {
  238. // 加锁防止重入
  239. lockman.LockRawObject(ctx, SSLCertificateManager.KeywordPlural(), r.Id)
  240. defer lockman.ReleaseRawObject(ctx, SSLCertificateManager.KeywordPlural(), r.Id)
  241. result := compare.SyncResult{}
  242. dbEss, err := r.GetSSLCertificates()
  243. if err != nil {
  244. result.Error(err)
  245. return result
  246. }
  247. removed := make([]SSSLCertificate, 0)
  248. commondb := make([]SSSLCertificate, 0)
  249. commonext := make([]cloudprovider.ICloudSSLCertificate, 0)
  250. added := make([]cloudprovider.ICloudSSLCertificate, 0)
  251. // 本地和云上资源列表进行比对
  252. err = compare.CompareSets(dbEss, exts, &removed, &commondb, &commonext, &added)
  253. if err != nil {
  254. result.Error(err)
  255. return result
  256. }
  257. // 删除云上没有的资源
  258. for i := 0; i < len(removed); i++ {
  259. err := removed[i].syncRemoveCloudSSLCertificate(ctx, userCred)
  260. if err != nil {
  261. result.DeleteError(err)
  262. continue
  263. }
  264. result.Delete()
  265. }
  266. // 和云上资源属性进行同步
  267. for i := 0; i < len(commondb); i++ {
  268. err := commondb[i].SyncWithCloudSSLCertificate(ctx, userCred, commonext[i])
  269. if err != nil {
  270. result.UpdateError(err)
  271. continue
  272. }
  273. result.Update()
  274. }
  275. // 创建本地没有的云上资源
  276. for i := 0; i < len(added); i++ {
  277. _, err := r.newFromCloudSSLCertificate(ctx, userCred, added[i])
  278. if err != nil {
  279. result.AddError(err)
  280. continue
  281. }
  282. result.Add()
  283. }
  284. return result
  285. }
  286. func (s *SSSLCertificate) syncRemoveCloudSSLCertificate(ctx context.Context, userCred mcclient.TokenCredential) error {
  287. err := s.RealDelete(ctx, userCred)
  288. if err != nil {
  289. return err
  290. }
  291. notifyclient.EventNotify(ctx, userCred, notifyclient.SEventNotifyParam{
  292. Obj: s,
  293. Action: notifyclient.ActionSyncDelete,
  294. })
  295. return nil
  296. }
  297. // 删除证书
  298. func (s *SSSLCertificate) CustomizeDelete(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, input jsonutils.JSONObject) error {
  299. err := s.StartSSLCertificateDeleteTask(ctx, userCred, nil)
  300. if err != nil {
  301. return errors.Wrapf(err, "StartSSLCertificateDeleteTask")
  302. }
  303. return nil
  304. }
  305. func (s *SSSLCertificate) StartSSLCertificateDeleteTask(ctx context.Context, userCred mcclient.TokenCredential, params *jsonutils.JSONDict) error {
  306. task, err := taskman.TaskManager.NewTask(ctx, "SSLCertificateDeleteTask", s, userCred, params, "", "", nil)
  307. if err != nil {
  308. return errors.Wrapf(err, "NewTask")
  309. }
  310. s.SetStatus(ctx, userCred, apis.STATUS_DELETING, "")
  311. return task.ScheduleRun(nil)
  312. }
  313. func (s *SSSLCertificate) Delete(ctx context.Context, userCred mcclient.TokenCredential) error {
  314. return nil
  315. }
  316. func (s *SSSLCertificate) RealDelete(ctx context.Context, userCred mcclient.TokenCredential) error {
  317. return s.SVirtualResourceBase.Delete(ctx, userCred)
  318. }
  319. // 同步资源属性
  320. func (s *SSSLCertificate) SyncWithCloudSSLCertificate(ctx context.Context, userCred mcclient.TokenCredential, ext cloudprovider.ICloudSSLCertificate) error {
  321. diff, err := db.UpdateWithLock(ctx, s, func() error {
  322. s.ExternalId = ext.GetGlobalId()
  323. s.Status = ext.GetStatus()
  324. s.Name = ext.GetName()
  325. s.Sans = ext.GetSans()
  326. s.StartDate = ext.GetStartDate()
  327. s.Province = ext.GetProvince()
  328. s.Common = ext.GetCommon()
  329. s.Country = ext.GetCountry()
  330. s.Issuer = ext.GetIssuer()
  331. s.IsUpload = ext.GetIsUpload()
  332. s.EndDate = ext.GetEndDate()
  333. s.Fingerprint = ext.GetFingerprint()
  334. s.City = ext.GetCity()
  335. s.OrgName = ext.GetOrgName()
  336. s.Certificate = ext.GetCert()
  337. s.PrivateKey = ext.GetKey()
  338. if zoneId := ext.GetDnsZoneId(); len(zoneId) > 0 {
  339. zone, err := db.FetchByExternalIdAndManagerId(DnsZoneManager, zoneId, func(q *sqlchemy.SQuery) *sqlchemy.SQuery {
  340. return q.Equals("manager_id", s.ManagerId)
  341. })
  342. if err == nil {
  343. s.DnsZoneId = zone.GetId()
  344. }
  345. }
  346. return nil
  347. })
  348. if err != nil {
  349. return errors.Wrapf(err, "db.Update")
  350. }
  351. if len(diff) > 0 {
  352. notifyclient.EventNotify(ctx, userCred, notifyclient.SEventNotifyParam{
  353. Obj: s,
  354. Action: notifyclient.ActionSyncUpdate,
  355. })
  356. }
  357. if account := s.GetCloudaccount(); account != nil {
  358. syncVirtualResourceMetadata(ctx, userCred, s, ext, account.ReadOnly)
  359. }
  360. if provider := s.GetCloudprovider(); provider != nil {
  361. SyncCloudProject(ctx, userCred, s, provider.GetOwnerId(), ext, provider)
  362. }
  363. db.OpsLog.LogSyncUpdate(s, diff, userCred)
  364. return nil
  365. }
  366. func (r *SCloudprovider) newFromCloudSSLCertificate(
  367. ctx context.Context,
  368. userCred mcclient.TokenCredential,
  369. ext cloudprovider.ICloudSSLCertificate,
  370. ) (*SSSLCertificate, error) {
  371. s := SSSLCertificate{}
  372. s.SetModelManager(SSLCertificateManager, &s)
  373. s.ExternalId = ext.GetGlobalId()
  374. s.ManagerId = r.Id
  375. s.IsEmulated = ext.IsEmulated()
  376. s.Status = ext.GetStatus()
  377. s.Name = ext.GetName()
  378. s.Sans = ext.GetSans()
  379. s.StartDate = ext.GetStartDate()
  380. s.Province = ext.GetProvince()
  381. s.Common = ext.GetCommon()
  382. s.Country = ext.GetCountry()
  383. s.Issuer = ext.GetIssuer()
  384. //s.Expired = ext.GetExpired()
  385. s.IsUpload = ext.GetIsUpload()
  386. s.EndDate = ext.GetEndDate()
  387. s.Fingerprint = ext.GetFingerprint()
  388. s.City = ext.GetCity()
  389. s.OrgName = ext.GetOrgName()
  390. s.Certificate = ext.GetCert()
  391. s.PrivateKey = ext.GetKey()
  392. if zoneId := ext.GetDnsZoneId(); len(zoneId) > 0 {
  393. zone, err := db.FetchByExternalIdAndManagerId(DnsZoneManager, zoneId, func(q *sqlchemy.SQuery) *sqlchemy.SQuery {
  394. return q.Equals("manager_id", r.Id)
  395. })
  396. if err == nil {
  397. s.DnsZoneId = zone.GetId()
  398. }
  399. }
  400. if createdAt := ext.GetCreatedAt(); !createdAt.IsZero() {
  401. s.CreatedAt = createdAt
  402. }
  403. var err error
  404. err = func() error {
  405. // 这里加锁是为了防止名称重复
  406. lockman.LockRawObject(ctx, SSLCertificateManager.Keyword(), "name")
  407. defer lockman.ReleaseRawObject(ctx, SSLCertificateManager.Keyword(), "name")
  408. s.Name, err = db.GenerateName(ctx, SSLCertificateManager, r.GetOwnerId(), ext.GetName())
  409. if err != nil {
  410. return errors.Wrapf(err, "db.GenerateName")
  411. }
  412. return SSLCertificateManager.TableSpec().Insert(ctx, &s)
  413. }()
  414. if err != nil {
  415. return nil, errors.Wrapf(err, "newFromCloudSSLCertificate.Insert")
  416. }
  417. notifyclient.EventNotify(ctx, userCred, notifyclient.SEventNotifyParam{
  418. Obj: &s,
  419. Action: notifyclient.ActionSyncCreate,
  420. })
  421. // 同步标签
  422. _ = syncVirtualResourceMetadata(ctx, userCred, &s, ext, false)
  423. // 同步项目归属
  424. SyncCloudProject(ctx, userCred, &s, r.GetOwnerId(), ext, r)
  425. db.OpsLog.LogEvent(&s, db.ACT_CREATE, s.GetShortDesc(ctx), userCred)
  426. return &s, nil
  427. }
  428. func (s *SSSLCertificate) GetICloudSSLCertificate(ctx context.Context) (cloudprovider.ICloudSSLCertificate, error) {
  429. if len(s.ExternalId) == 0 {
  430. return nil, errors.Wrapf(cloudprovider.ErrNotFound, "ExternalId is empty")
  431. }
  432. provider := s.GetCloudprovider()
  433. if provider == nil {
  434. return nil, errors.Wrapf(cloudprovider.ErrNotFound, "Cloudprovider is empty")
  435. }
  436. drv, err := provider.GetProvider(ctx)
  437. if err != nil {
  438. return nil, errors.Wrapf(err, "GetProvider")
  439. }
  440. certs, err := drv.GetISSLCertificates()
  441. if err != nil {
  442. return nil, errors.Wrapf(err, "GetICloudSSLCertificate")
  443. }
  444. for i := range certs {
  445. if certs[i].GetGlobalId() == s.ExternalId {
  446. return certs[i], nil
  447. }
  448. }
  449. return nil, errors.Wrapf(cloudprovider.ErrNotFound, "GetICloudSSLCertificate")
  450. }
  451. func (man *SSSLCertificateManager) ListItemExportKeys(ctx context.Context,
  452. q *sqlchemy.SQuery,
  453. userCred mcclient.TokenCredential,
  454. keys stringutils2.SSortedStrings,
  455. ) (*sqlchemy.SQuery, error) {
  456. var err error
  457. q, err = man.SVirtualResourceBaseManager.ListItemExportKeys(ctx, q, userCred, keys)
  458. if err != nil {
  459. return nil, errors.Wrap(err, "SVirtualResourceBaseManager.ListItemExportKeys")
  460. }
  461. if keys.ContainsAny(man.SManagedResourceBaseManager.GetExportKeys()...) {
  462. q, err = man.SManagedResourceBaseManager.ListItemExportKeys(ctx, q, userCred, keys)
  463. if err != nil {
  464. return nil, errors.Wrap(err, "SManagedResourceBaseManager.ListItemExportKeys")
  465. }
  466. }
  467. return q, nil
  468. }