Compare commits
27 Commits
98c09a6f9c
...
main
Author | SHA1 | Date | |
---|---|---|---|
27166314af | |||
|
d73a2da3e6 | ||
|
98137f11f2 | ||
|
992e99b25f | ||
f3cac73622 | |||
|
c36858f891 | ||
|
e1e199c53f | ||
|
2edd6fd3df | ||
|
f552c08b33 | ||
0b668f85c5 | |||
fe39119518 | |||
|
c7cd5dbd1b | ||
|
de1be5a536 | ||
|
bb26424787 | ||
|
c76bb23c6d | ||
|
2a5778bfe4 | ||
|
10eadc8a95 | ||
|
a6961cb19a | ||
|
af2805d793 | ||
|
24a409cfcf | ||
|
915be612a3 | ||
|
52deaacf3a | ||
|
b703cfbd7c | ||
|
528e76f3ee | ||
|
007ca43a43 | ||
|
7613b7da06 | ||
|
ad9621641f |
1146
data/data_asana.json
1146
data/data_asana.json
File diff suppressed because one or more lines are too long
@@ -1,5 +1,10 @@
|
|||||||
import logging
|
import logging
|
||||||
import logging.handlers
|
import logging.handlers
|
||||||
|
import sys
|
||||||
|
|
||||||
|
|
||||||
|
sys.stdin.reconfigure(encoding='utf-8')
|
||||||
|
sys.stdout.reconfigure(encoding='utf-8')
|
||||||
|
|
||||||
|
|
||||||
def setup_logging(name):
|
def setup_logging(name):
|
||||||
@@ -11,9 +16,9 @@ def setup_logging(name):
|
|||||||
|
|
||||||
file_handler = logging.FileHandler(
|
file_handler = logging.FileHandler(
|
||||||
filename=f"{name}.log",
|
filename=f"{name}.log",
|
||||||
mode='w'
|
mode='w',
|
||||||
#maxBytes=1024*1024,
|
encoding='utf-8',
|
||||||
#backupCount=5
|
delay=True
|
||||||
)
|
)
|
||||||
file_handler.setLevel(logging.DEBUG)
|
file_handler.setLevel(logging.DEBUG)
|
||||||
|
|
||||||
|
238
mapping.py
238
mapping.py
@@ -2,14 +2,13 @@ import csv
|
|||||||
import json
|
import json
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
# import logging
|
import time
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
|
|
||||||
|
|
||||||
# logging.basicConfig(level=logging.DEBUG, filename='mapping.log', filemode='w')
|
|
||||||
|
|
||||||
from logging_config import setup_logging
|
from logging_config import setup_logging
|
||||||
|
from urllib.parse import urlencode
|
||||||
|
|
||||||
logger = setup_logging(__name__)
|
logger = setup_logging(__name__)
|
||||||
|
|
||||||
@@ -21,17 +20,14 @@ sys.stdin.reconfigure(encoding='utf-8')
|
|||||||
sys.stdout.reconfigure(encoding='utf-8')
|
sys.stdout.reconfigure(encoding='utf-8')
|
||||||
|
|
||||||
|
|
||||||
|
QUEUE_NAME = 'TESTIMPORT'
|
||||||
|
|
||||||
# ниже функция для случаев, когда не будет готового файла для импорта
|
headers = {
|
||||||
# def get_data_from_asana():
|
'Host': 'api.tracker.yandex.net',
|
||||||
# """ Получение данных из Asana """
|
'Authorization': 'OAuth y0_AgAEA7qkB3KjAArkXwAAAADzMlP9oR1lwMzBS2e94jHzdnII8Laxi7k',
|
||||||
|
'X-Org-ID': '7095769',
|
||||||
# headers = {
|
'Content-Type': 'appication/json',
|
||||||
# 'Authorization': 'ASANA_TOKEN',
|
}
|
||||||
# }
|
|
||||||
# response = requests.get('https://api.asana.com/projects/PROJECT_ID/tasks', headers=headers)
|
|
||||||
# data = response.json()
|
|
||||||
# return data
|
|
||||||
|
|
||||||
|
|
||||||
def get_assignee_data():
|
def get_assignee_data():
|
||||||
@@ -63,13 +59,29 @@ def get_task_status(task):
|
|||||||
return status, completed_at
|
return status, completed_at
|
||||||
|
|
||||||
|
|
||||||
|
# здесь функция для дальнейшей работы с parent
|
||||||
|
def get_parent_task_ids(task, ya_imported_task_response):
|
||||||
|
""" Получение соответствий между идентификаторами подзадач в Асане и Яндекс Трекере """
|
||||||
|
|
||||||
|
parent_task_ids = {}
|
||||||
|
|
||||||
|
if task.get('parent') and 'gid' in task['parent']:
|
||||||
|
parent_gid = task['parent']['gid']
|
||||||
|
if parent_gid in ya_imported_task_response:
|
||||||
|
parent_id = ya_imported_task_response[parent_gid]
|
||||||
|
parent_task_ids[task['gid']] = parent_id
|
||||||
|
|
||||||
|
return parent_task_ids
|
||||||
|
|
||||||
|
|
||||||
def transform_data(data):
|
def transform_data(data):
|
||||||
""" Преобразование данных из Asana в ЯндексТрекер в совместимом формате """
|
""" Преобразование данных из Asana в ЯндексТрекер в совместимом формате """
|
||||||
|
|
||||||
transformed_data = []
|
transformed_data = []
|
||||||
for task in data["data"]:
|
for task in data:
|
||||||
status, completed_at = get_task_status(task)
|
status, completed_at = get_task_status(task)
|
||||||
transformed_task = {
|
transformed_task = {
|
||||||
|
'gid': task['gid'],
|
||||||
'summary': task['name'],
|
'summary': task['name'],
|
||||||
'description': task['notes'],
|
'description': task['notes'],
|
||||||
'createdAt': task['created_at'],
|
'createdAt': task['created_at'],
|
||||||
@@ -77,89 +89,157 @@ def transform_data(data):
|
|||||||
'assignee': task['assignee'],
|
'assignee': task['assignee'],
|
||||||
'status': status,
|
'status': status,
|
||||||
'completedAt': completed_at,
|
'completedAt': completed_at,
|
||||||
|
'parent': task['parent'],
|
||||||
|
'subtasks': transform_data(task['subtasks'])
|
||||||
}
|
}
|
||||||
transformed_data.append(transformed_task)
|
transformed_data.append(transformed_task)
|
||||||
return transformed_data
|
return transformed_data
|
||||||
|
|
||||||
|
|
||||||
def create_tasks_in_tracker(data):
|
def create_task(task):
|
||||||
|
base_url = 'https://api.tracker.yandex.net/v2/issues/_import'
|
||||||
|
print(task)
|
||||||
|
if task.get('assignee') and 'gid' in task['assignee']:
|
||||||
|
ya_assignee = assignee.get(task['assignee']['gid'], 'dr.cyrill')
|
||||||
|
else:
|
||||||
|
ya_assignee = 'dr.cyrill'
|
||||||
|
|
||||||
|
payload = {
|
||||||
|
'queue': QUEUE_NAME,
|
||||||
|
'summary': task['summary'],
|
||||||
|
'description': task['description'],
|
||||||
|
'createdAt': task['createdAt'],
|
||||||
|
'createdBy': 'dr.cyrill',
|
||||||
|
'deadline': task['deadline'],
|
||||||
|
'assignee': ya_assignee,
|
||||||
|
'status': task['status'],
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.debug('Request: %s', json.dumps(payload))
|
||||||
|
|
||||||
|
response = requests.post(
|
||||||
|
base_url,
|
||||||
|
headers=headers,
|
||||||
|
data=json.dumps(payload),
|
||||||
|
params=urlencode({'parent': task['parent']}) # добавила параметр для parent в url-запрос
|
||||||
|
)
|
||||||
|
|
||||||
|
if response.status_code == 201:
|
||||||
|
print('Задача успешно создана в ЯндексТрекер')
|
||||||
|
|
||||||
|
# здесь добавлено про родительскую задачу
|
||||||
|
return response.json()['id']
|
||||||
|
|
||||||
|
else:
|
||||||
|
print(
|
||||||
|
'Ошибка при создании задачи в ЯндексТрекер:',
|
||||||
|
response.content,
|
||||||
|
)
|
||||||
|
|
||||||
|
logger.debug('Response: %s', response.content)
|
||||||
|
|
||||||
|
|
||||||
|
def create_tasks_in_tracker(data, limit=10):
|
||||||
""" Создание задач в ЯндексТрекер """
|
""" Создание задач в ЯндексТрекер """
|
||||||
|
|
||||||
base_url = 'https://api.tracker.yandex.net/v2/issues/_import'
|
|
||||||
|
datalen = (len(data) if limit==-1 else min(limit, len(data)))
|
||||||
|
logger.info('Import started. Task count: %d', datalen)
|
||||||
|
|
||||||
|
# этот датасет тоже для parent добавлен
|
||||||
|
ya_imported_task_response = {}
|
||||||
|
|
||||||
|
for i in range(datalen):
|
||||||
|
task = data[i]
|
||||||
|
task_id = create_task(task)
|
||||||
|
logger.info('Created task id = %s',task_id)
|
||||||
|
for subtask in task['subtasks']:
|
||||||
|
logger.info('Found subtask gid = %s for id = %s',subtask['gid'], task_id)
|
||||||
|
subtask_id = create_task(subtask)
|
||||||
|
assign_parent_task(subtask_id, task_id)
|
||||||
|
|
||||||
|
|
||||||
|
# и эта функция для родительской задачи написана
|
||||||
|
def assign_parent_task(task_id, parent_id):
|
||||||
|
""" Назначение родительской задачи """
|
||||||
|
|
||||||
|
base_url = f'https://api.tracker.yandex.net/v2/issues/{task_id}'
|
||||||
headers = {
|
headers = {
|
||||||
'Host': 'api.tracker.yandex.net',
|
'Host': 'api.tracker.yandex.net',
|
||||||
'Authorization': 'OAuth y0_AgAEA7qkB3KjAArkXwAAAADzMlP9oR1lwMzBS2e94jHzdnII8Laxi7k',
|
'Authorization': 'OAuth y0_AgAEA7qkB3KjAArkXwAAAADzMlP9oR1lwMzBS2e94jHzdnII8Laxi7k',
|
||||||
'X-Org-ID': '7095769',
|
'X-Org-ID': '7095769',
|
||||||
'Content-Type': 'appication/json',
|
'Content-Type': 'application/json',
|
||||||
}
|
}
|
||||||
datalen = len(data)
|
|
||||||
logger.info('Import started. Task count: %d', datalen)
|
|
||||||
for task in data:
|
|
||||||
if task.get('assignee') and 'gid' in task['assignee']:
|
|
||||||
ya_assignee = assignee.get(task['assignee']['gid'], 'dr.cyrill')
|
|
||||||
else:
|
|
||||||
ya_assignee = 'dr.cyrill'
|
|
||||||
|
|
||||||
payload = {
|
|
||||||
'queue': 'TESTIMPORT', # Вынести в переменную
|
|
||||||
'summary': task['summary'],
|
|
||||||
'description': task['description'],
|
|
||||||
'createdAt': task['createdAt'],
|
|
||||||
'createdBy': 'dr.cyrill',
|
|
||||||
'deadline': task['deadline'],
|
|
||||||
'assignee': ya_assignee,
|
|
||||||
# 'milestone': task['milestone'],
|
|
||||||
# 'memberships': task['memberships'],
|
|
||||||
# 'parent': task['parent'],
|
|
||||||
'status': task['status'],
|
|
||||||
# 'completedAt': task['completedAt']
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
logger.debug('Request: %s', json.dumps(payload))
|
|
||||||
|
|
||||||
response = requests.post(
|
|
||||||
base_url,
|
|
||||||
headers=headers,
|
|
||||||
data=json.dumps(payload),
|
|
||||||
)
|
|
||||||
|
|
||||||
if response.status_code == 201:
|
|
||||||
print('Задача успешно создана в ЯндексТрекер')
|
|
||||||
else:
|
|
||||||
print(
|
|
||||||
'Ошибка при создании задачи в ЯндексТрекер:',
|
|
||||||
response.content,
|
|
||||||
)
|
|
||||||
|
|
||||||
logger.debug('Response: %s', response.content)
|
|
||||||
|
|
||||||
ya_imported_task_response = response.json()
|
|
||||||
if task.get('parent'):
|
|
||||||
parent_task_gid = task['parent']['gid']
|
|
||||||
if parent_task_gid in ya_imported_task_response:
|
|
||||||
parent_task_id = ya_imported_task_response[parent_task_gid]['id']
|
|
||||||
update_parent_task(
|
|
||||||
base_url,
|
|
||||||
headers,
|
|
||||||
ya_imported_task_response['id'],
|
|
||||||
parent_task_id,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def update_parent_task(base_url, headers, task_id, parent_task_id):
|
|
||||||
""" Обновление данных родительской задачи в ЯндексТрекере"""
|
|
||||||
|
|
||||||
payload = {
|
payload = {
|
||||||
'parent': parent_task_id
|
'parent': parent_id,
|
||||||
}
|
}
|
||||||
|
|
||||||
update_url = f"{base_url}/{task_id}"
|
logger.debug('Request: %s', json.dumps(payload))
|
||||||
|
|
||||||
|
response = requests.patch(
|
||||||
|
base_url,
|
||||||
|
headers=headers,
|
||||||
|
data=json.dumps(payload),
|
||||||
|
)
|
||||||
|
|
||||||
|
if response.status_code == 200:
|
||||||
|
print('Родительская задача успешно назначена')
|
||||||
|
logger.info('Task %s successfully assigned a parent',task_id)
|
||||||
|
else:
|
||||||
|
print('Ошибка при назначении родительской задачи:', response.content)
|
||||||
|
|
||||||
|
logger.debug('Response: %s', response.content)
|
||||||
|
|
||||||
|
time.sleep(1) # добавляем задержку перед следующим запросом к API
|
||||||
|
# здесь я закончила добавлять про родительскую задачу с большой надеждой
|
||||||
|
|
||||||
|
|
||||||
|
# пробую закрыть задачи уже в Трекере
|
||||||
|
def close_tasks_for_completedAt(queue_name):
|
||||||
|
base_url = 'https://api.tracker.yandex.net/v2/issues/_search'
|
||||||
|
payload = {
|
||||||
|
"filter": {
|
||||||
|
"queue": queue_name,
|
||||||
|
"status": "resolved" # Фильтр для задач cо статусом closed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
response = requests.post(
|
||||||
|
base_url,
|
||||||
|
headers=headers,
|
||||||
|
data=json.dumps(payload)
|
||||||
|
)
|
||||||
|
|
||||||
|
if response.status_code == 200:
|
||||||
|
logger.debug('Response: %s', response.content)
|
||||||
|
|
||||||
|
tasks = response.json()
|
||||||
|
|
||||||
|
for task in tasks:
|
||||||
|
task_id = task['id']
|
||||||
|
close_payload = {
|
||||||
|
'resolution': 'fixed'
|
||||||
|
}
|
||||||
|
close_url = f'https://api.tracker.yandex.net/v2/issues/{task_id}/transitions/close/_execute'
|
||||||
|
print(close_url)
|
||||||
|
print(headers)
|
||||||
|
close_response = requests.post(
|
||||||
|
close_url,
|
||||||
|
headers=headers,
|
||||||
|
data=json.dumps(close_payload)
|
||||||
|
)
|
||||||
|
if close_response.status_code == 200:
|
||||||
|
print(f'Задача {task_id} закрыта успешно')
|
||||||
|
else:
|
||||||
|
print(f'Ошибка при закрытии задачи {task_id}: {close_response.content}')
|
||||||
|
else:
|
||||||
|
print(f'Ошибка при получении задач для очереди {queue_name}: {response.content}')
|
||||||
|
|
||||||
|
|
||||||
# asana_data = get_data_from_asana()
|
|
||||||
file = open(asana_data_json, "r", encoding="utf8")
|
file = open(asana_data_json, "r", encoding="utf8")
|
||||||
json_data = json.loads(file.read())
|
json_data = json.loads(file.read())
|
||||||
|
|
||||||
yandex_tracker_data = transform_data(json_data)
|
yandex_tracker_data = transform_data(json_data['data'])
|
||||||
create_tasks_in_tracker(yandex_tracker_data)
|
create_tasks_in_tracker(yandex_tracker_data, limit=3)
|
||||||
|
close_tasks_for_completedAt('TESTIMPORT')
|
||||||
|
@@ -43,3 +43,40 @@ def logRoundtrip(response, *args, **kwargs):
|
|||||||
session = requests.Session()
|
session = requests.Session()
|
||||||
session.hooks['response'].append(logRoundtrip)
|
session.hooks['response'].append(logRoundtrip)
|
||||||
session.get('http://secariolabs.com')
|
session.get('http://secariolabs.com')
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# ya_imported_task_response = response.json()
|
||||||
|
# if task.get('parent'):
|
||||||
|
# parent_task_gid = task['parent']['gid']
|
||||||
|
# if parent_task_gid in ya_imported_task_response:
|
||||||
|
# parent_task_id = ya_imported_task_response[parent_task_gid]['id']
|
||||||
|
# logger.debug(f'В задаче {task["summary"]} есть связка "parent"', ensure_ascii=False)
|
||||||
|
# update_parent_task(
|
||||||
|
# base_url,
|
||||||
|
# headers,
|
||||||
|
# ya_imported_task_response['id'],
|
||||||
|
# parent_task_id,
|
||||||
|
# )
|
||||||
|
|
||||||
|
|
||||||
|
# def update_parent_task(base_url, headers, task_id, parent_task_id):
|
||||||
|
# """ Обновление данных родительской задачи в ЯндексТрекере"""
|
||||||
|
|
||||||
|
# data = {
|
||||||
|
# 'parent': parent_task_id
|
||||||
|
# }
|
||||||
|
# payload = json.dumps(data)
|
||||||
|
# update_url = f"{base_url}/{task_id}"
|
||||||
|
# response = requests.post(update_url, headers=headers, data=payload)
|
||||||
|
# result = json.loads(response.text)
|
||||||
|
# logger.warning(json.dumps(result, ensure_ascii=False))
|
||||||
|
|
||||||
|
# if response.status_code == 200:
|
||||||
|
# logger.info('Данные о связи с задачей "parent" успешно обновлены')
|
||||||
|
# else:
|
||||||
|
# logger.error(
|
||||||
|
# 'Ошибка при обновлении данных о связи с задачей "parent": %s',
|
||||||
|
# response.content,
|
||||||
|
# )
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
certifi==2023.11.17
|
certifi==2023.11.17
|
||||||
charset-normalizer==3.3.2
|
charset-normalizer==3.3.2
|
||||||
idna==3.4
|
idna==3.4
|
||||||
|
dotenv==1.0.1
|
||||||
requests==2.31.0
|
requests==2.31.0
|
||||||
setuptools==69.0.0
|
setuptools==69.0.0
|
||||||
six==1.16.0
|
six==1.16.0
|
||||||
|
78
resolved_task.py
Normal file
78
resolved_task.py
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
import json
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import requests
|
||||||
|
|
||||||
|
|
||||||
|
from dotenv import load_dotenv
|
||||||
|
from logging_config import setup_logging
|
||||||
|
|
||||||
|
load_dotenv()
|
||||||
|
|
||||||
|
logger = setup_logging(__name__)
|
||||||
|
|
||||||
|
base_dir = os.path.dirname(os.path.realpath(__file__))
|
||||||
|
asana_data_json = os.path.join(base_dir, './data/data_asana.json')
|
||||||
|
users_data = os.path.join(base_dir, './data/users.mapping.asana2ya.csv')
|
||||||
|
|
||||||
|
sys.stdin.reconfigure(encoding='utf-8')
|
||||||
|
sys.stdout.reconfigure(encoding='utf-8')
|
||||||
|
|
||||||
|
|
||||||
|
QUEUE_NAME = 'TESTIMPORT'
|
||||||
|
TOKEN = os.getenv('TOKEN')
|
||||||
|
|
||||||
|
def close_tasks_in_queue(queue_name):
|
||||||
|
base_url = 'https://api.tracker.yandex.net/v2/issues/_search'
|
||||||
|
# base_url = 'https://api.tracker.yandex.net/v2/issues/_search?expand=transitions'
|
||||||
|
headers = {
|
||||||
|
'Host': 'api.tracker.yandex.net',
|
||||||
|
'Authorization': TOKEN,
|
||||||
|
'X-Org-ID': '7095769',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
}
|
||||||
|
|
||||||
|
payload = {
|
||||||
|
"filter": {
|
||||||
|
"queue": queue_name,
|
||||||
|
"status": "open" # Фильтр для открытых задач в указанной очереди
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
while True:
|
||||||
|
|
||||||
|
response = requests.post(
|
||||||
|
base_url,
|
||||||
|
headers=headers,
|
||||||
|
data=json.dumps(payload)
|
||||||
|
)
|
||||||
|
|
||||||
|
if response.status_code == 200:
|
||||||
|
logger.debug('Response: %s', response.content)
|
||||||
|
|
||||||
|
tasks = response.json()
|
||||||
|
|
||||||
|
for task in tasks:
|
||||||
|
task_id = task['id']
|
||||||
|
close_payload = {
|
||||||
|
'resolution': 'fixed'
|
||||||
|
}
|
||||||
|
close_url = f'https://api.tracker.yandex.net/v2/issues/{task_id}/transitions/close/_execute'
|
||||||
|
print(close_url)
|
||||||
|
print(headers)
|
||||||
|
close_response = requests.post(
|
||||||
|
close_url,
|
||||||
|
headers=headers,
|
||||||
|
data=json.dumps(close_payload)
|
||||||
|
)
|
||||||
|
if close_response.status_code == 200:
|
||||||
|
print(f'Задача {task_id} закрыта успешно')
|
||||||
|
else:
|
||||||
|
print(f'Ошибка при закрытии задачи {task_id}: {close_response.content}')
|
||||||
|
else:
|
||||||
|
print(f'Ошибка при получении задач для очереди {queue_name}: {response.content}')
|
||||||
|
|
||||||
|
|
||||||
|
# Вызов функции для закрытия задач в очереди 'TESTIMPORT'
|
||||||
|
close_tasks_in_queue('TESTIMPORT')
|
Reference in New Issue
Block a user