1
0
Fork 0
forked from fnux/rt2zammad

Add actuall migration code

Signed-off-by: Michal Čihař <michal@cihar.com>
This commit is contained in:
Michal Čihař 2019-05-27 11:56:19 +02:00
parent 43d48c71e2
commit 88bdfc53f8
2 changed files with 161 additions and 40 deletions

View file

@ -1,2 +1,3 @@
rt rt
zammad_py zammad_py
python-dateutil

View file

@ -3,15 +3,27 @@
Quick and dirty attempt to migrate issues from Request Tracker to Zammad. Quick and dirty attempt to migrate issues from Request Tracker to Zammad.
""" """
import pickle import base64
import json import json
import os import os
import pickle
import sys import sys
from requests.exceptions import HTTPError from rt import Rt
from zammad_py import ZammadAPI from zammad_py import ZammadAPI
from zammad_py.api import TagList from zammad_py.api import Resource, TagList, TicketArticle
from rt import Rt, ALL_QUEUES
class Tag(Resource):
path_attribute = "tags"
def add(self, obj, id, item):
response = self._connection.session.get(
self.url + "/add?object=%s&o_id=%d&item=%s" % (obj, id, item)
)
return self._raise_or_return_json(response)
TEMPLATE = """{ TEMPLATE = """{
"zammad_host": "", "zammad_host": "",
@ -24,39 +36,55 @@ TEMPLATE = """{
} }
""" """
if not os.path.exists('rt2zammad.json'): COMMENT_TEMPLATE = """
print('Missing rt2zammad.json!') Ticket imported from Request Tracker
print('Create one based on following template:')
Created: {Created}
Resolved: {Resolved}
"""
if not os.path.exists("rt2zammad.json"):
print("Missing rt2zammad.json!")
print("Create one based on following template:")
print(TEMPLATE) print(TEMPLATE)
sys.exit(1) sys.exit(1)
with open('rt2zammad.json') as handle: with open("rt2zammad.json") as handle:
config = json.load(handle) config = json.load(handle)
target = ZammadAPI(
host=config['zammad_host'], def get_zammad(**kwargs):
username=config['zammad_user'], return ZammadAPI(
password=config['zammad_password'], host=config["zammad_host"],
is_secure=config['zammad_secure'], username=config["zammad_user"],
password=config["zammad_password"],
is_secure=config["zammad_secure"],
**kwargs
) )
target = get_zammad()
target.user.me() target.user.me()
source = Rt(config['rt_url'], config['rt_user'], config['rt_pass']) source = Rt(config["rt_url"], config["rt_user"], config["rt_pass"])
if not source.login(): if not source.login():
print('Failed to login to RT!') print("Failed to login to RT!")
sys.exit(2) sys.exit(2)
if os.path.exists('rt2zammad.cache'): if os.path.exists("rt2zammad.cache"):
# Load RT from cache # Load RT from cache
with open('rt2zammad.cache', 'rb') as handle: with open("rt2zammad.cache", "rb") as handle:
data = pickle.load(handle) data = pickle.load(handle)
users = data['users'] users = data["users"]
queues = data['queues'] queues = data["queues"]
tickets = data['tickets'] tickets = data["tickets"]
attachments = data["attachments"]
else: else:
# Load RT from remote # Load RT from remote
users = {} users = {}
attachments = {}
tickets = [] tickets = []
queues = set() queues = set()
@ -64,33 +92,125 @@ else:
if username not in users: if username not in users:
users[username] = source.get_user(username) users[username] = source.get_user(username)
for i in range(1, 1000): for i in range(1, 1000):
print('Loading ticket {}'.format(i)) print("Loading ticket {}".format(i))
ticket = source.get_ticket(i) ticket = source.get_ticket(i)
if ticket is None: if ticket is None:
break break
queues.add(ticket['Queue']) queues.add(ticket["Queue"])
ensure_user(ticket['Creator']) ensure_user(ticket["Creator"])
ensure_user(ticket['Owner']) ensure_user(ticket["Owner"])
history = source.get_history(i) history = source.get_history(i)
attachments = [] for item in history:
for a in source.get_attachments_ids(i): for a, title in item["Attachments"]:
attachment = source.get_attachment(i, a) attachments[a] = source.get_attachment(i, a)
attachments.append(attachments) ensure_user(item["Creator"])
ensure_user(attachment['Creator']) tickets.append({"ticket": ticket, "history": history})
tickets.append({ with open("rt2zammad.cache", "wb") as handle:
'ticket': ticket, data = pickle.dump(
'history': history, {
'attachments': attachments, "users": users,
}) "queues": queues,
with open('rt2zammad.cache', 'wb') as handle: "tickets": tickets,
data = pickle.dump({'users': users, 'queues': queues, 'tickets': tickets}, handle) "attachments": attachments,
},
handle,
)
# Create tags # Create tags
tag_list = TagList(target) tag_list = TagList(target)
tags = {tag['name'] for tag in tag_list.all()} ticket_article = TicketArticle(target)
tag_obj = Tag(target)
tags = {tag["name"] for tag in tag_list.all()}
for queue in queues: for queue in queues:
queue = queue.lower().split()[0] queue = queue.lower().split()[0]
if queue not in tags: if queue not in tags:
tag_list.create({'name': queue}) tag_list.create({"name": queue})
STATUSMAP = {"new": 1, "open": 2, "resolved": 4, "rejected": 4, "deleted": 4}
USERMAP = {}
for user in target.user.all():
USERMAP[user["email"].lower()] = user["login"]
def get_user(userdata):
email = userdata["EmailAddress"]
# Search existing users
if email not in USERMAP:
for user in target.user.search({"query": email}):
USERMAP[user["email"].lower()] = user["login"]
# Create new one
if email not in USERMAP:
kwargs = {"email": email}
if "RealName" in userdata:
realname = userdata["RealName"]
if ", " in realname:
last, first = realname.split(", ", 1)
elif " " in realname:
first, last = realname.split(None, 1)
else:
last = realname
first = ""
kwargs["lastname"] = last
kwargs["firstname"] = first
user = target.user.create(kwargs)
USERMAP[user["email"].lower()] = user["login"]
return USERMAP[email.lower()]
# Create tickets
for ticket in tickets:
label = "RT-{}".format(ticket["ticket"]["id"].split("/")[1])
print("Importing {}".format(label))
new = get_zammad(
on_behalf_of=get_user(users[ticket["ticket"]["Creator"]])
).ticket.create(
{
"title": "{} [{}]".format(ticket["ticket"]["Subject"], label),
"group": "Users",
"state_id": STATUSMAP[ticket["ticket"]["Status"]],
"customer_id": "guess:{}".format(
users[ticket["ticket"]["Creator"]]["EmailAddress"]
),
"note": "RT-import:{}".format(ticket["ticket"]["id"]),
"article": {
"subject": ticket["ticket"]["Subject"],
"body": ticket["history"][0]["Content"],
},
}
)
tag_obj.add("Ticket", new["id"], ticket["ticket"]["Queue"].lower().split()[0])
ticket_article.create(
{
"ticket_id": new["id"],
"body": COMMENT_TEMPLATE.format(**ticket["ticket"]),
"internal": True,
}
)
for item in ticket["history"]:
if item["Type"] not in ("Correspond", "Comment"):
continue
files = []
for a, title in item["Attachments"]:
data = attachments[a]
if data["Filename"] in ("", "signature.asc"):
continue
files.append(
{
"filename": data["Filename"],
"data": base64.b64encode(data["Content"]).decode("utf-8"),
"mime-type": data["ContentType"],
}
)
TicketArticle(get_zammad(on_behalf_of=get_user(users[item["Creator"]]))).create(
{
"ticket_id": new["id"],
"body": item["Content"],
"internal": item["Type"] == "Comment",
"attachments": files,
}
)