mirror of
https://gitlab.wikimedia.org/ladsgroup/Phabricator-maintenance-bot
synced 2025-01-03 08:20:58 +01:00
254 lines
7.8 KiB
Python
Executable file
254 lines
7.8 KiB
Python
Executable file
import json
|
|
import sys
|
|
import time
|
|
import os
|
|
|
|
import requests
|
|
|
|
|
|
class Client(object):
|
|
"""Phabricator client"""
|
|
|
|
def __init__(self, url, username, key):
|
|
self.url = url
|
|
self.username = username # Unused
|
|
self.column_cache = {}
|
|
self.phid_cache = {}
|
|
self.session = {
|
|
'token': key,
|
|
}
|
|
|
|
@classmethod
|
|
def newFromCreds(cls):
|
|
dir_path = os.path.dirname(os.path.realpath(__file__))
|
|
with open(os.path.join(dir_path, 'creds.json'), 'r') as f:
|
|
creds = json.loads(f.read())
|
|
return cls(*creds)
|
|
|
|
def post(self, path, data):
|
|
data['__conduit__'] = self.session
|
|
r = requests.post('%s/api/%s' % (self.url, path), data={
|
|
'params': json.dumps(data),
|
|
'output': 'json',
|
|
})
|
|
resp = r.json()
|
|
if resp['error_code'] is not None:
|
|
raise Exception(resp['error_info'])
|
|
return resp['result']
|
|
|
|
def lookupPhid(self, label):
|
|
"""Lookup information on a Phab object by name."""
|
|
if not self.phid_cache.get(label):
|
|
r = self.post('phid.lookup', {'names': [label]})
|
|
if label in r and 'phid' in r[label]:
|
|
obj = r[label]['phid']
|
|
self.phid_cache[label] = obj
|
|
else:
|
|
raise Exception('No object found for %s' % label)
|
|
return self.phid_cache[label]
|
|
|
|
def getSubprojects(self, phid):
|
|
"""Lookup information on a Phab object by name."""
|
|
r = self.post('project.search', {'constraints': {'isMilestone': True, 'ancestors': [phid]}})
|
|
return [i['phid'] for i in r['data']]
|
|
|
|
def getColumns(self, project_phid):
|
|
if not self.column_cache.get(project_phid):
|
|
self.column_cache[project_phid] = self.post(
|
|
'project.column.search', {
|
|
"constraints": {
|
|
"projects": [project_phid]}})
|
|
return self.column_cache[project_phid]
|
|
|
|
def moveColumns(self, task_phid, to_column):
|
|
self.post('maniphest.edit', {
|
|
'objectIdentifier': task_phid,
|
|
'transactions': [{
|
|
'type': 'column',
|
|
'value': [to_column],
|
|
}]
|
|
})
|
|
|
|
def setTaskDescription(self, task_phid, new_desc):
|
|
self.post('maniphest.edit', {
|
|
'objectIdentifier': task_phid,
|
|
'transactions': [{
|
|
'type': 'description',
|
|
'value': new_desc,
|
|
}]
|
|
})
|
|
|
|
def addTaskProject(self, task_phid, project_phid):
|
|
self.post('maniphest.edit', {
|
|
'objectIdentifier': task_phid,
|
|
'transactions': [{
|
|
'type': 'projects.add',
|
|
'value': [project_phid],
|
|
}]
|
|
})
|
|
|
|
def createSubtask(self, desc, project_phids, parent_phid, title):
|
|
self.post('maniphest.edit', {
|
|
'objectIdentifier': '',
|
|
'transactions': [{
|
|
'type': 'parent',
|
|
'value': parent_phid
|
|
},
|
|
{
|
|
'type': 'title',
|
|
'value': title
|
|
},
|
|
{
|
|
'type': 'description',
|
|
'value': desc,
|
|
},
|
|
{
|
|
'type': 'projects.add',
|
|
'value': project_phids
|
|
}]
|
|
})
|
|
|
|
def createParentTask(self, desc, project_phids, subtask_phid, title):
|
|
return self.post('maniphest.edit', {
|
|
'objectIdentifier': '',
|
|
'transactions': [{
|
|
'type': 'subtasks.add',
|
|
'value': [subtask_phid]
|
|
},
|
|
{
|
|
'type': 'title',
|
|
'value': title
|
|
},
|
|
{
|
|
'type': 'description',
|
|
'value': desc,
|
|
},
|
|
{
|
|
'type': 'projects.add',
|
|
'value': project_phids
|
|
}]
|
|
})
|
|
|
|
def taskDetails(self, phid):
|
|
"""Lookup details of a Maniphest task."""
|
|
r = self.post('maniphest.query', {'phids': [phid]})
|
|
if phid in r:
|
|
return r[phid]
|
|
raise Exception('No task found for phid %s' % phid)
|
|
|
|
def getTransactions(self, phid):
|
|
r = self.post('transaction.search', {'objectIdentifier': phid})
|
|
if 'data' in r:
|
|
return r['data']
|
|
raise Exception('No transaction found for phid %s' % phid)
|
|
|
|
def removeProject(self, project_phid, task):
|
|
return self.removeProjectByPhid(project_phid, self.lookupPhid(task))
|
|
|
|
def removeProjectByPhid(self, project_phid, task_phid):
|
|
self.post('maniphest.edit', {
|
|
'objectIdentifier': task_phid,
|
|
'transactions': [{
|
|
'type': 'projects.remove',
|
|
'value': [project_phid],
|
|
}]
|
|
})
|
|
|
|
def changeProjectByPhid(self, task_phid, old_project_phid, new_project_phid):
|
|
return self.post('maniphest.edit', {
|
|
'objectIdentifier': task_phid,
|
|
'transactions': [{
|
|
'type': 'projects.remove',
|
|
'value': [old_project_phid],
|
|
},
|
|
{
|
|
'type': 'projects.add',
|
|
'value': [new_project_phid],
|
|
}
|
|
]
|
|
|
|
})
|
|
|
|
def getTasksWithProject(self, project_phid, continue_=None, statuses=None):
|
|
r = self._getTasksWithProjectContinue(
|
|
project_phid, continue_, statuses=statuses)
|
|
cursor = r['cursor']
|
|
for case in r['data']:
|
|
if case['type'] != 'TASK':
|
|
continue
|
|
yield case['phid']
|
|
if cursor.get('after'):
|
|
for case in self.getTasksWithProject(
|
|
project_phid, cursor['after'], statuses=statuses):
|
|
yield case
|
|
|
|
def getInactiveTasksWithProject(self, project_phid, inactive_for=864000, statuses=['resolved'], columns=[]):
|
|
params = {
|
|
'limit': 100,
|
|
'constraints': {
|
|
'projects': [project_phid],
|
|
'statuses': statuses,
|
|
"modifiedEnd": int(time.time() - inactive_for),
|
|
'columnPHIDs': columns,
|
|
}
|
|
}
|
|
r = self.post('maniphest.search', params)
|
|
for case in r['data']:
|
|
if case['type'] != 'TASK':
|
|
continue
|
|
yield case['phid']
|
|
|
|
def _getTasksWithProjectContinue(self, project_phid, continue_=None, statuses=None):
|
|
params = {
|
|
'limit': 100,
|
|
'constraints': {
|
|
'projects': [project_phid],
|
|
"modifiedStart": int(time.time() - int(sys.argv[2]))
|
|
}
|
|
}
|
|
if continue_:
|
|
params['after'] = continue_
|
|
if statuses:
|
|
params['constraints']['statuses'] = statuses
|
|
return self.post('maniphest.search', params)
|
|
|
|
def getTaskColumns(self, phid):
|
|
params = {
|
|
"attachments": {
|
|
"columns": {"boards": {"columns": True}}
|
|
},
|
|
"constraints": {
|
|
"phids": [phid]
|
|
}
|
|
}
|
|
return self.post('maniphest.search', params)[
|
|
'data'][0]['attachments']['columns']
|
|
|
|
def getTaskSubtasks(self, phid):
|
|
params = {
|
|
"constraints": {
|
|
"phids": [phid],
|
|
"hasSubtasks": True
|
|
}
|
|
}
|
|
return self.post('maniphest.search', params)[
|
|
'data']
|
|
|
|
def getTaskName(self, keyword):
|
|
params = {
|
|
"constraints": {
|
|
"query": keyword
|
|
}
|
|
}
|
|
return self.post('maniphest.search', params)[
|
|
'data']
|
|
|
|
def getTaskParents(self, phid):
|
|
params = {
|
|
"constraints": {
|
|
"phids": [phid],
|
|
"hasParents": True
|
|
}
|
|
}
|
|
return self.post('maniphest.search', params)[
|
|
'data']
|