comparison libervia/backend/plugins/plugin_tickets_import.py @ 4071:4b842c1fb686

refactoring: renamed `sat` package to `libervia.backend`
author Goffi <goffi@goffi.org>
date Fri, 02 Jun 2023 11:49:51 +0200
parents sat/plugins/plugin_tickets_import.py@524856bd7b19
children e9971a4b0627
comparison
equal deleted inserted replaced
4070:d10748475025 4071:4b842c1fb686
1 #!/usr/bin/env python3
2
3
4 # SàT plugin for import external ticketss
5 # Copyright (C) 2009-2021 Jérôme Poisson (goffi@goffi.org)
6
7 # This program is free software: you can redistribute it and/or modify
8 # it under the terms of the GNU Affero General Public License as published by
9 # the Free Software Foundation, either version 3 of the License, or
10 # (at your option) any later version.
11
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU Affero General Public License for more details.
16
17 # You should have received a copy of the GNU Affero General Public License
18 # along with this program. If not, see <http://www.gnu.org/licenses/>.
19
20 from libervia.backend.core.i18n import _
21 from libervia.backend.core.constants import Const as C
22 from libervia.backend.core import exceptions
23 from libervia.backend.core.log import getLogger
24
25 log = getLogger(__name__)
26 from twisted.internet import defer
27 from libervia.backend.tools.common import uri
28 from libervia.backend.tools import utils
29
30
31 PLUGIN_INFO = {
32 C.PI_NAME: "tickets import",
33 C.PI_IMPORT_NAME: "TICKETS_IMPORT",
34 C.PI_TYPE: C.PLUG_TYPE_IMPORT,
35 C.PI_DEPENDENCIES: ["IMPORT", "XEP-0060", "XEP-0277", "XEP-0346"],
36 C.PI_MAIN: "TicketsImportPlugin",
37 C.PI_HANDLER: "no",
38 C.PI_DESCRIPTION: _(
39 """Tickets import management:
40 This plugin manage the different tickets importers which can register to it, and handle generic importing tasks."""
41 ),
42 }
43
44 OPT_MAPPING = "mapping"
45 FIELDS_LIST = ("labels", "cc_emails") # fields which must have a list as value
46 FIELDS_DATE = ("created", "updated")
47
48 NS_TICKETS = "fdp/submitted/org.salut-a-toi.tickets:0"
49
50
51 class TicketsImportPlugin(object):
52 BOOL_OPTIONS = ()
53 JSON_OPTIONS = (OPT_MAPPING,)
54 OPT_DEFAULTS = {}
55
56 def __init__(self, host):
57 log.info(_("plugin Tickets import initialization"))
58 self.host = host
59 self._importers = {}
60 self._p = host.plugins["XEP-0060"]
61 self._m = host.plugins["XEP-0277"]
62 self._s = host.plugins["XEP-0346"]
63 host.plugins["IMPORT"].initialize(self, "tickets")
64
65 @defer.inlineCallbacks
66 def import_item(
67 self, client, item_import_data, session, options, return_data, service, node
68 ):
69 """
70
71 @param item_import_data(dict): no key is mandatory, but if a key doesn't exists in dest form, it will be ignored.
72 Following names are recommendations which should be used where suitable in importers.
73 except if specified in description, values are unicode
74 'id': unique id (must be unique in the node) of the ticket
75 'title': title (or short description/summary) of the ticket
76 'body': main description of the ticket
77 'created': date of creation (unix time)
78 'updated': date of last update (unix time)
79 'author': full name of reporter
80 'author_jid': jid of reporter
81 'author_email': email of reporter
82 'assigned_to_name': full name of person working on it
83 'assigned_to_email': email of person working on it
84 'cc_emails': list of emails subscribed to the ticket
85 'priority': priority of the ticket
86 'severity': severity of the ticket
87 'labels': list of unicode values to use as label
88 'product': product concerned by this ticket
89 'component': part of the product concerned by this ticket
90 'version': version of the product/component concerned by this ticket
91 'platform': platform converned by this ticket
92 'os': operating system concerned by this ticket
93 'status': current status of the ticket, values:
94 - "queued": ticket is waiting
95 - "started": progress is ongoing
96 - "review": ticket is fixed and waiting for review
97 - "closed": ticket is finished or invalid
98 'milestone': target milestone for this ticket
99 'comments': list of microblog data (comment metadata, check [XEP_0277.send] data argument)
100 @param options(dict, None): Below are the generic options,
101 tickets importer can have specific ones. All options are serialized unicode values
102 generic options:
103 - OPT_MAPPING (json): dict of imported ticket key => exported ticket key
104 e.g.: if you want to map "component" to "labels", you can specify:
105 {'component': 'labels'}
106 If you specify several import ticket key to the same dest key,
107 the values will be joined with line feeds
108 """
109 if "comments_uri" in item_import_data:
110 raise exceptions.DataError(
111 _("comments_uri key will be generated and must not be used by importer")
112 )
113 for key in FIELDS_LIST:
114 if not isinstance(item_import_data.get(key, []), list):
115 raise exceptions.DataError(_("{key} must be a list").format(key=key))
116 for key in FIELDS_DATE:
117 try:
118 item_import_data[key] = utils.xmpp_date(item_import_data[key])
119 except KeyError:
120 continue
121 if session["root_node"] is None:
122 session["root_node"] = NS_TICKETS
123 if not "schema" in session:
124 session["schema"] = yield self._s.get_schema_form(
125 client, service, node or session["root_node"]
126 )
127 defer.returnValue(item_import_data)
128
129 @defer.inlineCallbacks
130 def import_sub_items(self, client, item_import_data, ticket_data, session, options):
131 # TODO: force "open" permission (except if private, check below)
132 # TODO: handle "private" metadata, to have non public access for node
133 # TODO: node access/publish model should be customisable
134 comments = ticket_data.get("comments", [])
135 service = yield self._m.get_comments_service(client)
136 node = self._m.get_comments_node(session["root_node"] + "_" + ticket_data["id"])
137 node_options = {
138 self._p.OPT_ACCESS_MODEL: self._p.ACCESS_OPEN,
139 self._p.OPT_PERSIST_ITEMS: 1,
140 self._p.OPT_DELIVER_PAYLOADS: 1,
141 self._p.OPT_SEND_ITEM_SUBSCRIBE: 1,
142 self._p.OPT_PUBLISH_MODEL: self._p.ACCESS_OPEN,
143 }
144 yield self._p.create_if_new_node(client, service, node, options=node_options)
145 ticket_data["comments_uri"] = uri.build_xmpp_uri(
146 "pubsub", subtype="microblog", path=service.full(), node=node
147 )
148 for comment in comments:
149 if "updated" not in comment and "published" in comment:
150 # we don't want an automatic update date
151 comment["updated"] = comment["published"]
152 yield self._m.send(client, comment, service, node)
153
154 def publish_item(self, client, ticket_data, service, node, session):
155 if node is None:
156 node = NS_TICKETS
157 id_ = ticket_data.pop("id", None)
158 log.debug(
159 "uploading item [{id}]: {title}".format(
160 id=id_, title=ticket_data.get("title", "")
161 )
162 )
163 return defer.ensureDeferred(
164 self._s.send_data_form_item(
165 client, service, node, ticket_data, session["schema"], id_
166 )
167 )
168
169 def item_filters(self, client, ticket_data, session, options):
170 mapping = options.get(OPT_MAPPING)
171 if mapping is not None:
172 if not isinstance(mapping, dict):
173 raise exceptions.DataError(_("mapping option must be a dictionary"))
174
175 for source, dest in mapping.items():
176 if not isinstance(source, str) or not isinstance(dest, str):
177 raise exceptions.DataError(
178 _(
179 "keys and values of mapping must be sources and destinations ticket fields"
180 )
181 )
182 if source in ticket_data:
183 value = ticket_data.pop(source)
184 if dest in FIELDS_LIST:
185 values = ticket_data[dest] = ticket_data.get(dest, [])
186 values.append(value)
187 else:
188 if dest in ticket_data:
189 ticket_data[dest] = ticket_data[dest] + "\n" + value
190 else:
191 ticket_data[dest] = value