云效跨账号迁移

360影视 欧美动漫 2025-04-03 21:31 1

摘要:由于业务调整或组织架构变更,原阿里云账号将不再继续使用。为了确保业务连续性和数据完整性,需要将源账号下的所有云产品及其相关数据完整、安全地迁移到目标账号。由于云效(DevOps平台)无法直接通过实例划拨至目标账号,因此需要对云效中的数据进行迁移。

由于业务调整或组织架构变更,原阿里云账号将不再继续使用。为了确保业务连续性和数据完整性,需要将源账号下的所有云产品及其相关数据完整、安全地迁移到目标账号。由于云效(DevOps平台)无法直接通过实例划拨至目标账号,因此需要对云效中的数据进行迁移。

在目标账号上点击

此链接,进入云效控制台界面,点击“新建组织”按钮。

选择“云效DevOps”点击“立即开启”按钮。

根据源账号云效,填写“组织名称”和“研发组织规模”,填写完成后点击“立即创建”按钮,完成云效企业的创建。

进入刚新建企业的控制台后,点击“项目协作 Projects”旁的“进入工作”按钮。

点击“新建”按钮,创建项目。

根据源账号项目类型,选择对应的项目模板。

根据源账号项目信息进行填写,填写完成后点击“新建”按钮。

新建完成后的项目如下图所示:

源账号导出需求

在源账号需求界面,点击“批量操作”,在下拉菜单中单击“导出全部”。

全选所有要导出的属性,完成后点击“开始导出”按钮。

在跳转后的“批量操作记录”界面,等待“进展”的进度条显示成功后,点击右边的“下载”图表按钮,下载数据文件。

需求数据表格打开后如下图所示:

目标账号导入需求

在目标账号的任务界面右上角,点击“导入数据”按钮。

选中“包含子工作项”后,点击“下载模板”按钮。

将源账号导出的数据,根据导入模板的格式,写入到该excel表格里。

注意:负责人必须为当前企业已存在的用户名称,迭代为当前项目已创建的迭代,填写不存在的信息会导致导入报错

返回到导入数据界面中,继续点击“下一步”按钮。

将刚填写完数据的导入模板上传上去,完成后点击“开始导入”按钮。

直到“导入工作项”显示为成功,则表示需求数据导入完成。

检查需求数据是否跟源账号上的数据一致。

迁移任务

源账号导出任务

在源账号任务界面,点击“批量操作”,在下拉菜单中单击“导出全部”。

查看导出的任务数据,如下图所示:

目标账号导入任务

和迁移需求一样,点击“下一步”按钮,将任务数据导入到目标账号中。

代码和迭代无法像需求和任务一样批量导出导入,为了提高效率和准确性,我们采用代码化方式批量迁移代码组、代码仓库和迭代,避免手动操作的费时费力及潜在错误。

获取组织ID

代码里需要用到“组织ID”,点击头像下拉菜单里的“管理后台”按钮,在“基本信息”页面即可看到“组织ID”。

获取企业空间ID

此链接,在“identity 企业标识”输入框内输入“组织ID”,完成后点击“发起调用”按钮。在左侧“调用结果”页面可以看到“namespaceId”,即为企业空间ID。

获取个人访问令牌

输入“令牌名称”,选择到期时间,按需选择权限。代码管理权限必选,后续要用它进行代码仓库数据导入操作。完成上述配置后,点击“新建”按钮创建个人访问令牌。

执行代码批量迁移

运行“源账号导出代码”,会print输出代码组、代码仓库和迭代的信息,运行“目标账号导入代码”时按代码提示输入代码组、代码仓库和迭代的信息,即可完成导入。执行前请在测试环境运行测试!!!

源账号导出代码

# -*- coding: utf-8 -*-# This file is auto-generated, don't edit it. Thanks.import sys,timefrom typing import Listfrom alibabacloud_devops20210625.client import Client as devops20210625Clientfrom alibabacloud_tea_openapi import models as open_api_modelsfrom alibabacloud_devops20210625 import models as devops_20210625_modelsfrom alibabacloud_tea_util import models as util_modelsfrom alibabacloud_tea_util.client import Client as UtilClientak="源账号AK"sk="源账号SK"organization_id='源账号组织ID'parent_id= 源账号企业空间IDclass Sample:def __init__(self):pass@staticmethoddef create_client(access_key_id: str,access_key_secret: str,) -> devops20210625Client:config = open_api_models.Config(access_key_id=ak,access_key_secret=sk)config.endpoint = f'devops.cn-hangzhou.aliyuncs.com'return devops20210625Client(config)#导出代码组@staticmethoddef ListGroupMember(args: List[str],) -> None:organization_id = args[0]groupId = args[1]User_list=client = Sample.create_client(ak, sk)list_group_member_request = devops_20210625_models.ListGroupMemberRequest(organization_id=organization_id)runtime = util_models.RuntimeOptionsheaders = {}Users = client.list_group_member_with_options(groupId, list_group_member_request, headers,runtime).body.resultfor User in Users:User_dic={}User_dic["name"]=User.nameUser_dic["access_level"] = User.access_levelUser_list.append(User_dic)return User_list@staticmethoddef ListRepositoryGroups(args: List[str],) -> None:organization_id=args[0]parent_id=args[1]ListRepositoryGroups_list=client = Sample.create_client(ak, sk)list_repository_groups_request = devops_20210625_models.ListRepositoryGroupsRequest(organization_id=organization_id,parent_id=parent_id,page_size=100)runtime = util_models.RuntimeOptionsheaders = {}ListRepositoryGroups=client.list_repository_groups_with_options(list_repository_groups_request, headers, runtime).body.resultfor RepositoryGroups in ListRepositoryGroups:ListRepositoryGroup_dic = {}ListRepositoryGroup_dic["id"] = RepositoryGroups.idListRepositoryGroup_dic["name"] =RepositoryGroups.nameListRepositoryGroup_dic["description"] =RepositoryGroups.descriptionListRepositoryGroup_dic["path"] = RepositoryGroups.pathListRepositoryGroup_dic["visibility_level"] = RepositoryGroups.visibility_levelUser_list=Sample.ListGroupMember([organization_id, str(RepositoryGroups.id)])ListRepositoryGroup_dic["user_list"] = User_listListRepositoryGroups_list.append(ListRepositoryGroup_dic)return ListRepositoryGroups_list#获取代码仓库信息@staticmethoddef GetRepositorymain(args: List[str],) -> None:organization_id = args[0]identity = args[1]client = Sample.create_client(ak, sk)get_repository_request = devops_20210625_models.GetRepositoryRequest(organization_id=organization_id,identity=identity)runtime = util_models.RuntimeOptionsheaders = {}http_url_to_repository = client.get_repository_with_options(get_repository_request, headers,runtime).body.repository.http_url_to_repositoryreturn http_url_to_repository@staticmethoddef ListRepositoriesmain(args: List[str],) -> None:organization_id = args[0]client = Sample.create_client(ak, sk)list_repositories_request = devops_20210625_models.ListRepositoriesRequest(organization_id=organization_id,per_page=100)runtime = util_models.RuntimeOptionsheaders = {}ListRepositories = client.list_repositories_with_options(list_repositories_request, headers,runtime).body.resultrepository_list = for Repository in ListRepositories:repository_dic = {}repository_dic["name"] = Repository.namerepository_dic["path"] = Repository.pathrepository_dic["namespace_id"] = Repository.namespace_idrepository_dic["description"] = Repository.descriptionrepository_dic["visibility_level"] = Repository.visibility_levelhttp_url_to_repository = Sample.GetRepositorymain([organization_id, Repository.id])repository_dic["import_url"] = http_url_to_repositoryrepository_list.append(repository_dic)return repository_list# 导出迭代信息@staticmethoddef ListProjectsmain(args: List[str],) -> None:organization_id = args[0]client = Sample.create_client(ak, sk)list_projects_request = devops_20210625_models.ListProjectsRequest(category='Project')runtime = util_models.RuntimeOptionsheaders = {}projects = client.list_projects_with_options(organization_id, list_projects_request, headers,runtime).body.projectsprojects_info = {}for project in projects:identifier = project.identifiername = project.nameprojects_info[name] = identifierreturn projects_info@staticmethoddef ListSprintsmain(args: List[str],) -> None:organization_id = args[0]space_identifier = args[1]client = Sample.create_client(ak, sk)list_sprints_request = devops_20210625_models.ListSprintsRequest(space_identifier=space_identifier,space_type='Project',max_results=200)runtime = util_models.RuntimeOptionsheaders = {}sprints = client.list_sprints_with_options(organization_id, list_sprints_request, headers,runtime).body.sprintssprint_list = for sprint in sprints:sprint_dic = {}sprint_dic["name"] = sprint.namesprint_dic["start_date"] = time.strftime('%Y-%m-%d', time.localtime(sprint.start_date / 1000))sprint_dic["end_date"] = time.strftime('%Y-%m-%d', time.localtime(sprint.end_date / 1000))sprint_list.append(sprint_dic)return sprint_listif __name__ == '__main__':#获取代码组信息ListRepositoryGroups_list=Sample.ListRepositoryGroups([organization_id,parent_id])print("代码组信息:"+str(ListRepositoryGroups_list))#获取代码仓库信息repository_list = Sample.ListRepositoriesmain([organization_id, ])print("代码仓库信息:"+str(repository_list))#获取迭代信息while True:projects_info = Sample.ListProjectsmain([organization_id, ])print("项目名称和项目ID的对应关系"+str(projects_info))identifier = input("请输入要导出的迭代是哪个项目的ID(不需要导出请输入no):")if identifier=="no":breaksprint_list = Sample.ListSprintsmain([organization_id, identifier])print("迭代的信息:"+str(sprint_list))# -*- coding: utf-8 -*-# This file is auto-generated, don't edit it. Thanks.from typing import Listfrom alibabacloud_devops20210625.client import Client as devops20210625Clientfrom alibabacloud_tea_openapi import models as open_api_modelsfrom alibabacloud_devops20210625 import models as devops_20210625_modelsfrom alibabacloud_tea_util import models as util_modelsfrom alibabacloud_tea_util.client import Client as UtilClientimport astak="目标账号AK"sk="目标账号SK"organization_id="目标账号组织ID"namespaceId="目标账号企业空间ID"import_account="目标账号个人访问令牌名称"import_token="目标账号个人访问令牌token"class Sample:def __init__(self):pass@staticmethoddef create_client(access_key_id: str,access_key_secret: str,) -> devops20210625Client:config = open_api_models.Config(access_key_id=ak,access_key_secret=sk)config.endpoint = f'devops.cn-hangzhou.aliyuncs.com'return devops20210625Client(config)#导入代码组@staticmethoddef CreateRepositoryGroupmain(args: List[str],) -> None:organization_id = args[0]name = args[1]path = args[2]visibility_level = args[3]namespaceId = args[4]description = args[5]client = Sample.create_client(ak,sk)create_repository_group_request = devops_20210625_models.CreateRepositoryGroupRequest(organization_id=organization_id,name=name,path=path,visibility_level=visibility_level,parent_id=namespaceId,description=description)runtime = util_models.RuntimeOptionsheaders = {}print(client.create_repository_group_with_options(create_repository_group_request, headers, runtime))#获取源代码组ID与目标代码组ID的对应关系@staticmethoddef ListRepositoryGroups(args: List[str],) -> None:organization_id = args[0]parent_id = args[1]ListRepositoryGroups_list = client = Sample.create_client(ak,sk)list_repository_groups_request = devops_20210625_models.ListRepositoryGroupsRequest(organization_id=organization_id,parent_id=parent_id,page_size=100)runtime = util_models.RuntimeOptionsheaders = {}ListRepositoryGroups = client.list_repository_groups_with_options(list_repository_groups_request, headers,runtime).body.resultfor RepositoryGroups in ListRepositoryGroups:ListRepositoryGroup_dic = {}ListRepositoryGroup_dic["id"] = RepositoryGroups.idListRepositoryGroup_dic["name"] = RepositoryGroups.nameListRepositoryGroup_dic["description"] = RepositoryGroups.descriptionListRepositoryGroup_dic["path"] = RepositoryGroups.pathListRepositoryGroup_dic["visibility_level"] = RepositoryGroups.visibility_levelListRepositoryGroups_list.append(ListRepositoryGroup_dic)return ListRepositoryGroups_list#导入代码仓库@staticmethoddef CreateRepositorymain(args: List[str],) -> None:organization_id = args[0]name = args[1]path = args[2]namespace_id = args[3]description = args[4]visibility_level = args[5]import_url = args[6]import_account = args[7]import_token = args[8]client = Sample.create_client(ak,sk)create_repository_request = devops_20210625_models.CreateRepositoryRequest(organization_id=organization_id,name=name,path=path,namespace_id=namespace_id,description=description,visibility_level=visibility_level,import_url=import_url,import_account=import_account,import_token=import_token)runtime = util_models.RuntimeOptionsheaders = {}print(client.create_repository_with_options(create_repository_request, headers, runtime))# 得到项目名称和项目ID的对应关系@staticmethoddef ListProjectsmain(args: List[str],) -> None:organization_id = args[0]client = Sample.create_client(ak,sk)list_projects_request = devops_20210625_models.ListProjectsRequest(category='Project')runtime = util_models.RuntimeOptionsheaders = {}projects = client.list_projects_with_options(organization_id, list_projects_request, headers,runtime).body.projectsprojects_info = {}for project in projects:identifier = project.identifiername = project.nameprojects_info[name] = identifierreturn projects_info#创建迭代@staticmethoddef CreateSprintmain(args: List[str],) -> None:organization_id=args[0]identifier=args[1]FROM_diedai=args[2]staff_id=args[3]client = Sample.create_client(ak,sk)create_sprint_request = devops_20210625_models.CreateSprintRequest(staff_ids=[staff_id], #ram用户uidname=FROM_diedai["name"], #迭代名称space_identifier=identifier, #项目IDstart_date=FROM_diedai["start_date"], #迭代开始时间end_date=FROM_diedai["end_date"] #迭代结束时间)runtime = util_models.RuntimeOptionsheaders = {}print(client.create_sprint_with_options(organization_id, create_sprint_request, headers, runtime))if __name__ == '__main__':# 导入代码组FROMListRepositoryGroups_list=ast.literal_eval(input("请输入源账号导出的代码组列表:"))for RepositoryGroup in FROMListRepositoryGroups_list:Sample.CreateRepositoryGroupmain([organization_id,RepositoryGroup["name"],RepositoryGroup["path"],RepositoryGroup["visibility_level"],namespaceId,RepositoryGroup["description"]])# 获取源代码组ID与目标代码组ID的对应关系TOListRepositoryGroups_list = Sample.ListRepositoryGroups([organization_id, namespaceId])FROM_TOListRepositoryGroupID_dic = {}for FROMListRepositoryGroup in FROMListRepositoryGroups_list:for TOListRepositoryGroup in TOListRepositoryGroups_list:if FROMListRepositoryGroup["name"] == TOListRepositoryGroup["name"]:FROM_TOListRepositoryGroupID_dic[FROMListRepositoryGroup["id"]] = TOListRepositoryGroup["id"]FROMparent_id=ast.literal_eval(input("请输入源账号的parent_id,也就是namespace_id:"))FROM_TOListRepositoryGroupID_dic[FROMparent_id]=int(namespaceId)print("源账号代码组ID与目标账号代码组ID的对应关系:"+str(FROM_TOListRepositoryGroupID_dic))#导入代码仓库FROMrepository_list=ast.literal_eval(input("请输入源账号导出的代码仓库列表:"))for FROMrepository in FROMrepository_list:Sample.CreateRepositorymain([organization_id, FROMrepository["name"], FROMrepository["path"],FROM_TOListRepositoryGroupID_dic[FROMrepository["namespace_id"]], FROMrepository["description"],FROMrepository["visibility_level"], FROMrepository["import_url"], import_account, import_token])# 导入迭代while True:projects_info = Sample.ListProjectsmain([organization_id, ])print("项目名称和项目ID的对应关系:"+str(projects_info))identifier = input("请输入要导入的迭代是哪个项目的ID(不需要导出请输入no):")if identifier=="no":breakprint("用户ID:{用户A:用户A的RAM账号UID,用户B:用户B的RAM账号UID,用户C:用户C的RAM账号UID}")print("项目与负责人的对应关系:{'项目A':'负责人A','项目B':'负责人B','项目C':'负责人C'}")staff_id = input("请输入要导入迭代负责人的RAM账号UID:")FROM_diedai_list = ast.literal_eval(input("请输入要导入的迭代内容列表:"))for FROM_diedai in FROM_diedai_list:Sample.CreateSprintmain([organization_id, identifier, FROM_diedai, staff_id])

为了确保业务连续性和数据完整性,针对源阿里云账号即将停用的情况,我们完成了云效(DevOps平台)的数据迁移工作。由于云效无法直接通过实例迁移至目标账号,我们采取了数据导出与导入的方式,确保所有项目、流水线配置及代码仓库等数据完整无误地迁移到目标账号下。至此,平稳顺利的完成此次迁移。

如有相关问题,请在文章后面给小编留言,小编安排作者第一时间和您联系,为您答疑解惑。

来源:小丁看科技

相关推荐