You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

TicketNotifier.java 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617
  1. /*
  2. * Copyright 2014 gitblit.com.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.gitblit.tickets;
  17. import java.io.IOException;
  18. import java.io.InputStream;
  19. import java.text.DateFormat;
  20. import java.text.MessageFormat;
  21. import java.text.SimpleDateFormat;
  22. import java.util.ArrayList;
  23. import java.util.Arrays;
  24. import java.util.Collections;
  25. import java.util.HashMap;
  26. import java.util.HashSet;
  27. import java.util.List;
  28. import java.util.Map;
  29. import java.util.Set;
  30. import java.util.TreeMap;
  31. import java.util.TreeSet;
  32. import java.util.regex.Matcher;
  33. import java.util.regex.Pattern;
  34. import org.apache.commons.io.IOUtils;
  35. import org.apache.log4j.Logger;
  36. import org.eclipse.jgit.diff.DiffEntry.ChangeType;
  37. import org.eclipse.jgit.lib.Repository;
  38. import org.eclipse.jgit.revwalk.RevCommit;
  39. import org.slf4j.LoggerFactory;
  40. import com.gitblit.Constants;
  41. import com.gitblit.IStoredSettings;
  42. import com.gitblit.Keys;
  43. import com.gitblit.git.PatchsetCommand;
  44. import com.gitblit.manager.INotificationManager;
  45. import com.gitblit.manager.IRepositoryManager;
  46. import com.gitblit.manager.IRuntimeManager;
  47. import com.gitblit.manager.IUserManager;
  48. import com.gitblit.models.Mailing;
  49. import com.gitblit.models.PathModel.PathChangeModel;
  50. import com.gitblit.models.RepositoryModel;
  51. import com.gitblit.models.TicketModel;
  52. import com.gitblit.models.TicketModel.Change;
  53. import com.gitblit.models.TicketModel.Field;
  54. import com.gitblit.models.TicketModel.Patchset;
  55. import com.gitblit.models.TicketModel.Review;
  56. import com.gitblit.models.TicketModel.Status;
  57. import com.gitblit.models.UserModel;
  58. import com.gitblit.utils.ArrayUtils;
  59. import com.gitblit.utils.DiffUtils;
  60. import com.gitblit.utils.DiffUtils.DiffStat;
  61. import com.gitblit.utils.JGitUtils;
  62. import com.gitblit.utils.MarkdownUtils;
  63. import com.gitblit.utils.StringUtils;
  64. /**
  65. * Formats and queues ticket/patch notifications for dispatch to the
  66. * mail executor upon completion of a push or a ticket update. Messages are
  67. * created as Markdown and then transformed to html.
  68. *
  69. * @author James Moger
  70. *
  71. */
  72. public class TicketNotifier {
  73. protected final Map<Long, Mailing> queue = new TreeMap<Long, Mailing>();
  74. private final String SOFT_BRK = "\n";
  75. private final String HARD_BRK = "\n\n";
  76. private final String HR = "----\n\n";
  77. private final IStoredSettings settings;
  78. private final INotificationManager notificationManager;
  79. private final IUserManager userManager;
  80. private final IRepositoryManager repositoryManager;
  81. private final ITicketService ticketService;
  82. private final String addPattern = "<span style=\"color:darkgreen;\">+{0}</span>";
  83. private final String delPattern = "<span style=\"color:darkred;\">-{0}</span>";
  84. public TicketNotifier(
  85. IRuntimeManager runtimeManager,
  86. INotificationManager notificationManager,
  87. IUserManager userManager,
  88. IRepositoryManager repositoryManager,
  89. ITicketService ticketService) {
  90. this.settings = runtimeManager.getSettings();
  91. this.notificationManager = notificationManager;
  92. this.userManager = userManager;
  93. this.repositoryManager = repositoryManager;
  94. this.ticketService = ticketService;
  95. }
  96. public void sendAll() {
  97. for (Mailing mail : queue.values()) {
  98. notificationManager.send(mail);
  99. }
  100. }
  101. public void sendMailing(TicketModel ticket) {
  102. queueMailing(ticket);
  103. sendAll();
  104. }
  105. /**
  106. * Queues an update notification.
  107. *
  108. * @param ticket
  109. * @return a notification object used for testing
  110. */
  111. public Mailing queueMailing(TicketModel ticket) {
  112. try {
  113. // format notification message
  114. String markdown = formatLastChange(ticket);
  115. StringBuilder html = new StringBuilder();
  116. html.append("<head>");
  117. html.append(readStyle());
  118. html.append("</head>");
  119. html.append("<body>");
  120. html.append(MarkdownUtils.transformGFM(settings, markdown, ticket.repository));
  121. html.append("</body>");
  122. Mailing mailing = Mailing.newHtml();
  123. mailing.from = getUserModel(ticket.updatedBy == null ? ticket.createdBy : ticket.updatedBy).getDisplayName();
  124. mailing.subject = getSubject(ticket);
  125. mailing.content = html.toString();
  126. mailing.id = "ticket." + ticket.number + "." + StringUtils.getSHA1(ticket.repository + ticket.number);
  127. setRecipients(ticket, mailing);
  128. queue.put(ticket.number, mailing);
  129. return mailing;
  130. } catch (Exception e) {
  131. Logger.getLogger(getClass()).error("failed to queue mailing for #" + ticket.number, e);
  132. }
  133. return null;
  134. }
  135. protected String getSubject(TicketModel ticket) {
  136. Change lastChange = ticket.changes.get(ticket.changes.size() - 1);
  137. boolean newTicket = lastChange.isStatusChange() && ticket.changes.size() == 1;
  138. String re = newTicket ? "" : "Re: ";
  139. String subject = MessageFormat.format("{0}[{1}] {2} (#{3,number,0})",
  140. re, StringUtils.stripDotGit(ticket.repository), ticket.title, ticket.number);
  141. return subject;
  142. }
  143. protected String formatLastChange(TicketModel ticket) {
  144. Change lastChange = ticket.changes.get(ticket.changes.size() - 1);
  145. UserModel user = getUserModel(lastChange.author);
  146. // define the fields we do NOT want to see in an email notification
  147. Set<TicketModel.Field> fieldExclusions = new HashSet<TicketModel.Field>();
  148. fieldExclusions.addAll(Arrays.asList(Field.watchers, Field.voters));
  149. StringBuilder sb = new StringBuilder();
  150. boolean newTicket = false;
  151. boolean isFastForward = true;
  152. List<RevCommit> commits = null;
  153. DiffStat diffstat = null;
  154. String pattern;
  155. if (lastChange.isStatusChange()) {
  156. Status state = lastChange.getStatus();
  157. switch (state) {
  158. case New:
  159. // new ticket
  160. newTicket = true;
  161. fieldExclusions.add(Field.status);
  162. fieldExclusions.add(Field.title);
  163. fieldExclusions.add(Field.body);
  164. if (lastChange.hasPatchset()) {
  165. pattern = "**{0}** is proposing a change.";
  166. } else {
  167. pattern = "**{0}** created this ticket.";
  168. }
  169. sb.append(MessageFormat.format(pattern, user.getDisplayName()));
  170. break;
  171. default:
  172. // some form of resolved
  173. if (lastChange.hasField(Field.mergeSha)) {
  174. // closed by push (merged patchset)
  175. pattern = "**{0}** closed this ticket by pushing {1} to {2}.";
  176. // identify patch that closed the ticket
  177. String merged = ticket.mergeSha;
  178. for (Patchset patchset : ticket.getPatchsets()) {
  179. if (patchset.tip.equals(ticket.mergeSha)) {
  180. merged = patchset.toString();
  181. break;
  182. }
  183. }
  184. sb.append(MessageFormat.format(pattern, user.getDisplayName(), merged, ticket.mergeTo));
  185. } else {
  186. // workflow status change by user
  187. pattern = "**{0}** changed the status of this ticket to **{1}**.";
  188. sb.append(MessageFormat.format(pattern, user.getDisplayName(), lastChange.getStatus().toString().toUpperCase()));
  189. }
  190. break;
  191. }
  192. sb.append(HARD_BRK);
  193. } else if (lastChange.hasPatchset()) {
  194. // patchset uploaded
  195. Patchset patchset = lastChange.patchset;
  196. String base = "";
  197. // determine the changed paths
  198. Repository repo = null;
  199. try {
  200. repo = repositoryManager.getRepository(ticket.repository);
  201. if (patchset.isFF() && (patchset.rev > 1)) {
  202. // fast-forward update, just show the new data
  203. isFastForward = true;
  204. Patchset prev = ticket.getPatchset(patchset.number, patchset.rev - 1);
  205. base = prev.tip;
  206. } else {
  207. // proposal OR non-fast-forward update
  208. isFastForward = false;
  209. base = patchset.base;
  210. }
  211. diffstat = DiffUtils.getDiffStat(repo, base, patchset.tip);
  212. commits = JGitUtils.getRevLog(repo, base, patchset.tip);
  213. } catch (Exception e) {
  214. Logger.getLogger(getClass()).error("failed to get changed paths", e);
  215. } finally {
  216. repo.close();
  217. }
  218. // describe the patchset
  219. String compareUrl = ticketService.getCompareUrl(ticket, base, patchset.tip);
  220. if (patchset.isFF()) {
  221. pattern = "**{0}** added {1} {2} to patchset {3}.";
  222. sb.append(MessageFormat.format(pattern, user.getDisplayName(), patchset.added, patchset.added == 1 ? "commit" : "commits", patchset.number));
  223. } else {
  224. pattern = "**{0}** uploaded patchset {1}. *({2})*";
  225. sb.append(MessageFormat.format(pattern, user.getDisplayName(), patchset.number, patchset.type.toString().toUpperCase()));
  226. }
  227. sb.append(HARD_BRK);
  228. sb.append(MessageFormat.format("{0} {1}, {2} {3}, <span style=\"color:darkgreen;\">+{4} insertions</span>, <span style=\"color:darkred;\">-{5} deletions</span> from {6}. [compare]({7})",
  229. commits.size(), commits.size() == 1 ? "commit" : "commits",
  230. diffstat.paths.size(),
  231. diffstat.paths.size() == 1 ? "file" : "files",
  232. diffstat.getInsertions(),
  233. diffstat.getDeletions(),
  234. isFastForward ? "previous revision" : "merge base",
  235. compareUrl));
  236. // note commit additions on a rebase,if any
  237. switch (lastChange.patchset.type) {
  238. case Rebase:
  239. if (lastChange.patchset.added > 0) {
  240. sb.append(SOFT_BRK);
  241. sb.append(MessageFormat.format("{0} {1} added.", lastChange.patchset.added, lastChange.patchset.added == 1 ? "commit" : "commits"));
  242. }
  243. break;
  244. default:
  245. break;
  246. }
  247. sb.append(HARD_BRK);
  248. } else if (lastChange.hasReview()) {
  249. // review
  250. Review review = lastChange.review;
  251. pattern = "**{0}** has reviewed patchset {1,number,0} revision {2,number,0}.";
  252. sb.append(MessageFormat.format(pattern, user.getDisplayName(), review.patchset, review.rev));
  253. sb.append(HARD_BRK);
  254. String d = settings.getString(Keys.web.datestampShortFormat, "yyyy-MM-dd");
  255. String t = settings.getString(Keys.web.timeFormat, "HH:mm");
  256. DateFormat df = new SimpleDateFormat(d + " " + t);
  257. List<Change> reviews = ticket.getReviews(ticket.getPatchset(review.patchset, review.rev));
  258. sb.append("| Date | Reviewer | Score | Description |\n");
  259. sb.append("| :--- | :------------ | :---: | :----------- |\n");
  260. for (Change change : reviews) {
  261. String name = change.author;
  262. UserModel u = userManager.getUserModel(change.author);
  263. if (u != null) {
  264. name = u.getDisplayName();
  265. }
  266. String score;
  267. switch (change.review.score) {
  268. case approved:
  269. score = MessageFormat.format(addPattern, change.review.score.getValue());
  270. break;
  271. case vetoed:
  272. score = MessageFormat.format(delPattern, Math.abs(change.review.score.getValue()));
  273. break;
  274. default:
  275. score = "" + change.review.score.getValue();
  276. }
  277. String date = df.format(change.date);
  278. sb.append(String.format("| %1$s | %2$s | %3$s | %4$s |\n",
  279. date, name, score, change.review.score.toString()));
  280. }
  281. sb.append(HARD_BRK);
  282. } else if (lastChange.hasComment()) {
  283. // comment update
  284. sb.append(MessageFormat.format("**{0}** commented on this ticket.", user.getDisplayName()));
  285. sb.append(HARD_BRK);
  286. } else {
  287. // general update
  288. pattern = "**{0}** has updated this ticket.";
  289. sb.append(MessageFormat.format(pattern, user.getDisplayName()));
  290. sb.append(HARD_BRK);
  291. }
  292. // ticket link
  293. sb.append(MessageFormat.format("[view ticket {0,number,0}]({1})",
  294. ticket.number, ticketService.getTicketUrl(ticket)));
  295. sb.append(HARD_BRK);
  296. if (newTicket) {
  297. // ticket title
  298. sb.append(MessageFormat.format("### {0}", ticket.title));
  299. sb.append(HARD_BRK);
  300. // ticket description, on state change
  301. if (StringUtils.isEmpty(ticket.body)) {
  302. sb.append("<span style=\"color: #888;\">no description entered</span>");
  303. } else {
  304. sb.append(ticket.body);
  305. }
  306. sb.append(HARD_BRK);
  307. sb.append(HR);
  308. }
  309. // field changes
  310. if (lastChange.hasFieldChanges()) {
  311. Map<Field, String> filtered = new HashMap<Field, String>();
  312. for (Map.Entry<Field, String> fc : lastChange.fields.entrySet()) {
  313. if (!fieldExclusions.contains(fc.getKey())) {
  314. // field is included
  315. filtered.put(fc.getKey(), fc.getValue());
  316. }
  317. }
  318. // sort by field ordinal
  319. List<Field> fields = new ArrayList<Field>(filtered.keySet());
  320. Collections.sort(fields);
  321. if (filtered.size() > 0) {
  322. sb.append(HARD_BRK);
  323. sb.append("| Field Changes ||\n");
  324. sb.append("| ------------: | :----------- |\n");
  325. for (Field field : fields) {
  326. String value;
  327. if (filtered.get(field) == null) {
  328. value = "";
  329. } else {
  330. value = filtered.get(field).replace("\r\n", "<br/>").replace("\n", "<br/>").replace("|", "&#124;");
  331. }
  332. sb.append(String.format("| **%1$s:** | %2$s |\n", field.name(), value));
  333. }
  334. sb.append(HARD_BRK);
  335. }
  336. }
  337. // new comment
  338. if (lastChange.hasComment()) {
  339. sb.append(HR);
  340. sb.append(lastChange.comment.text);
  341. sb.append(HARD_BRK);
  342. }
  343. // insert the patchset details and review instructions
  344. if (lastChange.hasPatchset() && ticket.isOpen()) {
  345. if (commits != null && commits.size() > 0) {
  346. // append the commit list
  347. String title = isFastForward ? "Commits added to previous patchset revision" : "All commits in patchset";
  348. sb.append(MessageFormat.format("| {0} |||\n", title));
  349. sb.append("| SHA | Author | Title |\n");
  350. sb.append("| :-- | :----- | :---- |\n");
  351. for (RevCommit commit : commits) {
  352. sb.append(MessageFormat.format("| {0} | {1} | {2} |\n",
  353. commit.getName(), commit.getAuthorIdent().getName(),
  354. StringUtils.trimString(commit.getShortMessage(), Constants.LEN_SHORTLOG).replace("|", "&#124;")));
  355. }
  356. sb.append(HARD_BRK);
  357. }
  358. if (diffstat != null) {
  359. // append the changed path list
  360. String title = isFastForward ? "Files changed since previous patchset revision" : "All files changed in patchset";
  361. sb.append(MessageFormat.format("| {0} |||\n", title));
  362. sb.append("| :-- | :----------- | :-: |\n");
  363. for (PathChangeModel path : diffstat.paths) {
  364. String add = MessageFormat.format(addPattern, path.insertions);
  365. String del = MessageFormat.format(delPattern, path.deletions);
  366. String diff = null;
  367. switch (path.changeType) {
  368. case ADD:
  369. diff = add;
  370. break;
  371. case DELETE:
  372. diff = del;
  373. break;
  374. case MODIFY:
  375. if (path.insertions > 0 && path.deletions > 0) {
  376. // insertions & deletions
  377. diff = add + "/" + del;
  378. } else if (path.insertions > 0) {
  379. // just insertions
  380. diff = add;
  381. } else {
  382. // just deletions
  383. diff = del;
  384. }
  385. break;
  386. default:
  387. diff = path.changeType.name();
  388. break;
  389. }
  390. sb.append(MessageFormat.format("| {0} | {1} | {2} |\n",
  391. getChangeType(path.changeType), path.name, diff));
  392. }
  393. sb.append(HARD_BRK);
  394. }
  395. sb.append(formatPatchsetInstructions(ticket, lastChange.patchset));
  396. }
  397. return sb.toString();
  398. }
  399. protected String getChangeType(ChangeType type) {
  400. String style = null;
  401. switch (type) {
  402. case ADD:
  403. style = "color:darkgreen;";
  404. break;
  405. case COPY:
  406. style = "";
  407. break;
  408. case DELETE:
  409. style = "color:darkred;";
  410. break;
  411. case MODIFY:
  412. style = "";
  413. break;
  414. case RENAME:
  415. style = "";
  416. break;
  417. default:
  418. break;
  419. }
  420. String code = type.name().toUpperCase().substring(0, 1);
  421. if (style == null) {
  422. return code;
  423. } else {
  424. return MessageFormat.format("<strong><span style=\"{0}padding:2px;margin:2px;border:1px solid #ddd;\">{1}</span></strong>", style, code);
  425. }
  426. }
  427. /**
  428. * Generates patchset review instructions for command-line git
  429. *
  430. * @param patchset
  431. * @return instructions
  432. */
  433. protected String formatPatchsetInstructions(TicketModel ticket, Patchset patchset) {
  434. String canonicalUrl = settings.getString(Keys.web.canonicalUrl, "https://localhost:8443");
  435. String repositoryUrl = canonicalUrl + Constants.R_PATH + ticket.repository;
  436. String ticketBranch = Repository.shortenRefName(PatchsetCommand.getTicketBranch(ticket.number));
  437. String patchsetBranch = PatchsetCommand.getPatchsetBranch(ticket.number, patchset.number);
  438. String reviewBranch = PatchsetCommand.getReviewBranch(ticket.number);
  439. String instructions = readResource("commands.md");
  440. instructions = instructions.replace("${ticketId}", "" + ticket.number);
  441. instructions = instructions.replace("${patchset}", "" + patchset.number);
  442. instructions = instructions.replace("${repositoryUrl}", repositoryUrl);
  443. instructions = instructions.replace("${ticketRef}", ticketBranch);
  444. instructions = instructions.replace("${patchsetRef}", patchsetBranch);
  445. instructions = instructions.replace("${reviewBranch}", reviewBranch);
  446. return instructions;
  447. }
  448. /**
  449. * Gets the usermodel for the username. Creates a temp model, if required.
  450. *
  451. * @param username
  452. * @return a usermodel
  453. */
  454. protected UserModel getUserModel(String username) {
  455. UserModel user = userManager.getUserModel(username);
  456. if (user == null) {
  457. // create a temporary user model (for unit tests)
  458. user = new UserModel(username);
  459. }
  460. return user;
  461. }
  462. /**
  463. * Set the proper recipients for a ticket.
  464. *
  465. * @param ticket
  466. * @param mailing
  467. */
  468. protected void setRecipients(TicketModel ticket, Mailing mailing) {
  469. RepositoryModel repository = repositoryManager.getRepositoryModel(ticket.repository);
  470. //
  471. // Direct TO recipients
  472. //
  473. Set<String> toAddresses = new TreeSet<String>();
  474. for (String name : ticket.getParticipants()) {
  475. UserModel user = userManager.getUserModel(name);
  476. if (user != null) {
  477. if (!StringUtils.isEmpty(user.emailAddress)) {
  478. if (user.canView(repository)) {
  479. toAddresses.add(user.emailAddress);
  480. } else {
  481. LoggerFactory.getLogger(getClass()).warn(
  482. MessageFormat.format("ticket {0}-{1,number,0}: {2} can not receive notification",
  483. repository.name, ticket.number, user.username));
  484. }
  485. }
  486. }
  487. }
  488. mailing.setRecipients(toAddresses);
  489. //
  490. // CC recipients
  491. //
  492. Set<String> ccs = new TreeSet<String>();
  493. // cc users mentioned in last comment
  494. Change lastChange = ticket.changes.get(ticket.changes.size() - 1);
  495. if (lastChange.hasComment()) {
  496. Pattern p = Pattern.compile("\\s@([A-Za-z0-9-_]+)");
  497. Matcher m = p.matcher(lastChange.comment.text);
  498. while (m.find()) {
  499. String username = m.group();
  500. ccs.add(username);
  501. }
  502. }
  503. // cc users who are watching the ticket
  504. ccs.addAll(ticket.getWatchers());
  505. // TODO cc users who are watching the repository
  506. Set<String> ccAddresses = new TreeSet<String>();
  507. for (String name : ccs) {
  508. UserModel user = userManager.getUserModel(name);
  509. if (user != null) {
  510. if (!StringUtils.isEmpty(user.emailAddress)) {
  511. if (user.canView(repository)) {
  512. ccAddresses.add(user.emailAddress);
  513. } else {
  514. LoggerFactory.getLogger(getClass()).warn(
  515. MessageFormat.format("ticket {0}-{1,number,0}: {2} can not receive notification",
  516. repository.name, ticket.number, user.username));
  517. }
  518. }
  519. }
  520. }
  521. // cc repository mailing list addresses
  522. if (!ArrayUtils.isEmpty(repository.mailingLists)) {
  523. ccAddresses.addAll(repository.mailingLists);
  524. }
  525. ccAddresses.addAll(settings.getStrings(Keys.mail.mailingLists));
  526. mailing.setCCs(ccAddresses);
  527. }
  528. protected String readStyle() {
  529. StringBuilder sb = new StringBuilder();
  530. sb.append("<style>\n");
  531. sb.append(readResource("email.css"));
  532. sb.append("</style>\n");
  533. return sb.toString();
  534. }
  535. protected String readResource(String resource) {
  536. StringBuilder sb = new StringBuilder();
  537. InputStream is = null;
  538. try {
  539. is = getClass().getResourceAsStream(resource);
  540. List<String> lines = IOUtils.readLines(is);
  541. for (String line : lines) {
  542. sb.append(line).append('\n');
  543. }
  544. } catch (IOException e) {
  545. } finally {
  546. if (is != null) {
  547. try {
  548. is.close();
  549. } catch (IOException e) {
  550. }
  551. }
  552. }
  553. return sb.toString();
  554. }
  555. }