You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

pod.go 9.7 kB

3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. package server
  2. import (
  3. "context"
  4. "flag"
  5. "fmt"
  6. "sync"
  7. "code.gitlink.org.cn/JCCE/PCM.git/adaptor/pod/service"
  8. "code.gitlink.org.cn/JCCE/PCM.git/common/tenanter"
  9. "code.gitlink.org.cn/JCCE/PCM.git/lan_trans/idl/pbpod"
  10. "code.gitlink.org.cn/JCCE/PCM.git/lan_trans/idl/pbtenant"
  11. "github.com/golang/glog"
  12. "github.com/pkg/errors"
  13. )
  14. // GetPodRegion get the available region for pod
  15. func GetPodRegion(ctx context.Context, req *pbpod.GetPodRegionReq) (resp *pbpod.GetPodRegionResp, err error) {
  16. var (
  17. regionInit tenanter.Region
  18. regions []*pbtenant.Region
  19. )
  20. switch req.GetProvider() {
  21. case pbtenant.CloudProvider_ali:
  22. regionInit, _ = tenanter.NewRegion(req.GetProvider(), 2)
  23. case pbtenant.CloudProvider_tencent:
  24. regionInit, _ = tenanter.NewRegion(req.GetProvider(), 5)
  25. case pbtenant.CloudProvider_huawei:
  26. regionInit, _ = tenanter.NewRegion(req.GetProvider(), 5)
  27. }
  28. tenanters, err := tenanter.GetTenanters(req.GetProvider())
  29. if err != nil {
  30. return nil, errors.WithMessage(err, "getTenanters error")
  31. }
  32. for _, tenant := range tenanters {
  33. pod, err := poder.NewPodClient(req.GetProvider(), regionInit, tenant)
  34. if err != nil {
  35. return nil, errors.WithMessage(err, "NewPodClient error")
  36. }
  37. request := &pbpod.GetPodRegionReq{
  38. Provider: req.GetProvider(),
  39. }
  40. resp, err := pod.GetPodRegion(ctx, request)
  41. if err != nil {
  42. return nil, errors.Wrap(err, "GetPodRegion error")
  43. }
  44. for _, region := range resp.GetRegions() {
  45. regions = append(regions, region)
  46. }
  47. }
  48. return &pbpod.GetPodRegionResp{Regions: regions}, nil
  49. }
  50. func CreatePods(ctx context.Context, req *pbpod.CreatePodsReq) (*pbpod.CreatePodsResp, error) {
  51. var (
  52. wg sync.WaitGroup
  53. requestIds = make([]string, 0)
  54. )
  55. wg.Add(len(req.CreatePodReq))
  56. c := make(chan string)
  57. for k := range req.CreatePodReq {
  58. reqPod := req.CreatePodReq[k]
  59. go func() {
  60. defer wg.Done()
  61. resp, err := CreatePod(ctx, reqPod)
  62. if err != nil || resp == nil {
  63. fmt.Println(errors.Wrap(err, "Batch pod creation error"))
  64. return
  65. }
  66. c <- resp.RequestId
  67. }()
  68. }
  69. go func() {
  70. defer close(c)
  71. wg.Wait()
  72. }()
  73. isFinished := false
  74. if len(requestIds) > 0 {
  75. isFinished = true
  76. }
  77. for v := range c {
  78. requestIds = append(requestIds, v)
  79. }
  80. return &pbpod.CreatePodsResp{
  81. Finished: isFinished,
  82. RequestId: requestIds,
  83. }, nil
  84. }
  85. func CreatePod(ctx context.Context, req *pbpod.CreatePodReq) (*pbpod.CreatePodResp, error) {
  86. var (
  87. pod poder.Poder
  88. )
  89. tenanters, err := tenanter.GetTenanters(req.Provider)
  90. if err != nil {
  91. return nil, errors.WithMessage(err, "getTenanters error")
  92. }
  93. region, err := tenanter.NewRegion(req.Provider, req.RegionId)
  94. if err != nil {
  95. return nil, errors.WithMessagef(err, "provider %v regionId %v", req.Provider, req.RegionId)
  96. }
  97. for _, tenant := range tenanters {
  98. if req.AccountName == "" || tenant.AccountName() == req.AccountName {
  99. if pod, err = poder.NewPodClient(req.Provider, region, tenant); err != nil {
  100. return nil, errors.WithMessage(err, "NewPodClient error")
  101. }
  102. break
  103. }
  104. }
  105. return pod.CreatePod(ctx, req)
  106. }
  107. func DeletePod(ctx context.Context, req *pbpod.DeletePodReq) (*pbpod.DeletePodResp, error) {
  108. var (
  109. pod poder.Poder
  110. )
  111. //pcm adk过来的请求需要从用户本地读取配置文件
  112. if len(req.RequestSource) > 0 {
  113. var configFile string
  114. flag.StringVar(&configFile, "conf", "configs/config.yaml", "config.yaml")
  115. flag.Parse()
  116. defer glog.Flush()
  117. if err := tenanter.LoadCloudConfigsFromFile(configFile); err != nil {
  118. if !errors.Is(err, tenanter.ErrLoadTenanterFileEmpty) {
  119. glog.Fatalf("tenanter.LoadCloudConfigsFromFile error %+v", err)
  120. }
  121. glog.Warningf("tenanter.LoadCloudConfigsFromFile empty file path %s", configFile)
  122. }
  123. glog.Infof("load tenant from file finished")
  124. }
  125. tenanters, err := tenanter.GetTenanters(req.Provider)
  126. if err != nil {
  127. return nil, errors.WithMessage(err, "getTenanters error")
  128. }
  129. region, err := tenanter.NewRegion(req.Provider, req.RegionId)
  130. if err != nil {
  131. return nil, errors.WithMessagef(err, "provider %v regionId %v", req.Provider, req.RegionId)
  132. }
  133. for _, tenant := range tenanters {
  134. if req.AccountName == "" || tenant.AccountName() == req.AccountName {
  135. if pod, err = poder.NewPodClient(req.Provider, region, tenant); err != nil {
  136. return nil, errors.WithMessage(err, "NewPodClient error")
  137. }
  138. break
  139. }
  140. }
  141. return pod.DeletePod(ctx, req)
  142. }
  143. func UpdatePod(ctx context.Context, req *pbpod.UpdatePodReq) (*pbpod.UpdatePodResp, error) {
  144. var (
  145. pod poder.Poder
  146. )
  147. //pcm adk过来的请求需要从用户本地读取配置文件
  148. if len(req.RequestSource) > 0 {
  149. var configFile string
  150. flag.StringVar(&configFile, "conf", "configs/config.yaml", "config.yaml")
  151. flag.Parse()
  152. defer glog.Flush()
  153. if err := tenanter.LoadCloudConfigsFromFile(configFile); err != nil {
  154. if !errors.Is(err, tenanter.ErrLoadTenanterFileEmpty) {
  155. glog.Fatalf("tenanter.LoadCloudConfigsFromFile error %+v", err)
  156. }
  157. glog.Warningf("tenanter.LoadCloudConfigsFromFile empty file path %s", configFile)
  158. }
  159. glog.Infof("load tenant from file finished")
  160. }
  161. tenanters, err := tenanter.GetTenanters(req.Provider)
  162. if err != nil {
  163. return nil, errors.WithMessage(err, "getTenanters error")
  164. }
  165. region, err := tenanter.NewRegion(req.Provider, req.RegionId)
  166. if err != nil {
  167. return nil, errors.WithMessagef(err, "provider %v regionId %v", req.Provider, req.RegionId)
  168. }
  169. for _, tenant := range tenanters {
  170. if req.AccountName == "" || tenant.AccountName() == req.AccountName {
  171. if pod, err = poder.NewPodClient(req.Provider, region, tenant); err != nil {
  172. return nil, errors.WithMessage(err, "NewPodClient error")
  173. }
  174. break
  175. }
  176. }
  177. return pod.UpdatePod(ctx, req)
  178. }
  179. func ListPodDetail(ctx context.Context, req *pbpod.ListPodDetailReq) (*pbpod.ListPodDetailResp, error) {
  180. var (
  181. pod poder.Poder
  182. )
  183. tenanters, err := tenanter.GetTenanters(req.Provider)
  184. if err != nil {
  185. return nil, errors.WithMessage(err, "getTenanters error")
  186. }
  187. region, err := tenanter.NewRegion(req.Provider, req.RegionId)
  188. if err != nil {
  189. return nil, errors.WithMessagef(err, "provider %v regionId %v", req.Provider, req.RegionId)
  190. }
  191. for _, tenant := range tenanters {
  192. if req.AccountName == "" || tenant.AccountName() == req.AccountName {
  193. if pod, err = poder.NewPodClient(req.Provider, region, tenant); err != nil {
  194. return nil, errors.WithMessage(err, "NewPodClient error")
  195. }
  196. break
  197. }
  198. }
  199. return pod.ListPodDetail(ctx, req)
  200. }
  201. func ListPod(ctx context.Context, req *pbpod.ListPodReq) (*pbpod.ListPodResp, error) {
  202. var (
  203. wg sync.WaitGroup
  204. mutex sync.Mutex
  205. pods []*pbpod.PodInstance
  206. tenanters []tenanter.Tenanter
  207. )
  208. //pcm adk过来的请求需要从用户本地读取配置文件
  209. if len(req.RequestSource) > 0 {
  210. var configFile string
  211. flag.StringVar(&configFile, "conf", "configs/config.yaml", "config.yaml")
  212. flag.Parse()
  213. defer glog.Flush()
  214. if err := tenanter.LoadCloudConfigsFromFile(configFile); err != nil {
  215. if !errors.Is(err, tenanter.ErrLoadTenanterFileEmpty) {
  216. glog.Fatalf("tenanter.LoadCloudConfigsFromFile error %+v", err)
  217. }
  218. glog.Warningf("tenanter.LoadCloudConfigsFromFile empty file path %s", configFile)
  219. }
  220. glog.Infof("load tenant from file finished")
  221. }
  222. tenanters, _ = tenanter.GetTenanters(req.Provider)
  223. //get the available region for product
  224. reqPodRegion := &pbpod.GetPodRegionReq{Provider: req.GetProvider()}
  225. respPodRegion, err := GetPodRegion(ctx, reqPodRegion)
  226. if err != nil {
  227. return nil, errors.WithMessage(err, "getPodRegion error")
  228. }
  229. wg.Add(len(tenanters) * len(respPodRegion.Regions))
  230. for _, t := range tenanters {
  231. for _, region := range respPodRegion.Regions {
  232. go func(tenant tenanter.Tenanter, region tenanter.Region) {
  233. defer wg.Done()
  234. pod, err := poder.NewPodClient(req.Provider, region, tenant)
  235. if err != nil {
  236. glog.Errorf("New Pod Client error %v", err)
  237. return
  238. }
  239. request := &pbpod.ListPodDetailReq{
  240. Provider: req.Provider,
  241. AccountName: tenant.AccountName(),
  242. RegionId: region.GetId(),
  243. Namespace: req.Namespace,
  244. PageNumber: 1,
  245. PageSize: 100,
  246. NextToken: "",
  247. }
  248. for {
  249. resp, err := pod.ListPodDetail(ctx, request)
  250. if err != nil {
  251. glog.Errorf("ListDetail error %v", err)
  252. return
  253. }
  254. mutex.Lock()
  255. pods = append(pods, resp.Pods...)
  256. mutex.Unlock()
  257. if resp.Finished {
  258. break
  259. }
  260. request.PageNumber, request.PageSize, request.NextToken = resp.PageNumber, resp.PageSize, resp.NextToken
  261. }
  262. }(t, region)
  263. }
  264. }
  265. wg.Wait()
  266. return &pbpod.ListPodResp{Pods: pods}, nil
  267. }
  268. func ListPodAll(ctx context.Context) (*pbpod.ListPodResp, error) {
  269. var (
  270. wg sync.WaitGroup
  271. mutex sync.Mutex
  272. pods []*pbpod.PodInstance
  273. )
  274. wg.Add(len(pbtenant.CloudProvider_name))
  275. for k := range pbtenant.CloudProvider_name {
  276. go func(provider int32) {
  277. defer wg.Done()
  278. //针对私有K8S集群,调用listAll时默认只查询ListPodDetailReq namespace下的pod
  279. if provider == 3 {
  280. resp, err := ListPod(ctx, &pbpod.ListPodReq{Provider: pbtenant.CloudProvider(provider), Namespace: "pcm"})
  281. if err != nil {
  282. glog.Errorf("List error %v", err)
  283. return
  284. }
  285. mutex.Lock()
  286. pods = append(pods, resp.Pods...)
  287. mutex.Unlock()
  288. } else {
  289. resp, err := ListPod(ctx, &pbpod.ListPodReq{Provider: pbtenant.CloudProvider(provider)})
  290. if err != nil {
  291. glog.Errorf("List error %v", err)
  292. return
  293. }
  294. mutex.Lock()
  295. pods = append(pods, resp.Pods...)
  296. mutex.Unlock()
  297. }
  298. }(k)
  299. }
  300. wg.Wait()
  301. return &pbpod.ListPodResp{Pods: pods}, nil
  302. }

PCM is positioned as Software stack over Cloud, aiming to build the standards and ecology of heterogeneous cloud collaboration for JCC in a non intrusive and autonomous peer-to-peer manner.