query-classes.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503
  1. import {
  2. OPERATOR_ASSIGNEE,
  3. OPERATOR_BOARD,
  4. OPERATOR_COMMENT,
  5. OPERATOR_CREATED_AT,
  6. OPERATOR_DUE,
  7. OPERATOR_HAS,
  8. OPERATOR_LABEL,
  9. OPERATOR_LIMIT,
  10. OPERATOR_LIST,
  11. OPERATOR_MEMBER,
  12. OPERATOR_MODIFIED_AT,
  13. OPERATOR_SORT,
  14. OPERATOR_STATUS,
  15. OPERATOR_SWIMLANE,
  16. OPERATOR_UNKNOWN,
  17. OPERATOR_USER,
  18. ORDER_ASCENDING,
  19. ORDER_DESCENDING,
  20. PREDICATE_ALL,
  21. PREDICATE_ARCHIVED,
  22. PREDICATE_ASSIGNEES,
  23. PREDICATE_ATTACHMENT,
  24. PREDICATE_CHECKLIST,
  25. PREDICATE_CREATED_AT,
  26. PREDICATE_DESCRIPTION,
  27. PREDICATE_DUE_AT,
  28. PREDICATE_END_AT,
  29. PREDICATE_ENDED,
  30. PREDICATE_MEMBERS,
  31. PREDICATE_MODIFIED_AT,
  32. PREDICATE_MONTH,
  33. PREDICATE_OPEN,
  34. PREDICATE_OVERDUE,
  35. PREDICATE_PRIVATE,
  36. PREDICATE_PUBLIC,
  37. PREDICATE_QUARTER,
  38. PREDICATE_START_AT,
  39. PREDICATE_WEEK,
  40. PREDICATE_YEAR,
  41. } from './search-const';
  42. import Boards from '../models/boards';
  43. import moment from 'moment';
  44. export class QueryParams {
  45. text = '';
  46. constructor(params = {}) {
  47. this.params = params;
  48. }
  49. hasOperator(operator) {
  50. return this.params[operator];
  51. }
  52. addPredicate(operator, predicate) {
  53. if (!this.hasOperator(operator)) {
  54. this.params[operator] = [];
  55. }
  56. this.params[operator].push(predicate);
  57. }
  58. setPredicate(operator, predicate) {
  59. this.params[operator] = predicate;
  60. }
  61. getPredicate(operator) {
  62. return this.params[operator][0];
  63. }
  64. getPredicates(operator) {
  65. return this.params[operator];
  66. }
  67. getParams() {
  68. return this.params;
  69. }
  70. }
  71. export class QueryErrors {
  72. operatorTagMap = [
  73. [OPERATOR_BOARD, 'board-title-not-found'],
  74. [OPERATOR_SWIMLANE, 'swimlane-title-not-found'],
  75. [
  76. OPERATOR_LABEL,
  77. label => {
  78. if (Boards.labelColors().includes(label)) {
  79. return {
  80. tag: 'label-color-not-found',
  81. value: label,
  82. color: true,
  83. };
  84. } else {
  85. return {
  86. tag: 'label-not-found',
  87. value: label,
  88. color: false,
  89. };
  90. }
  91. },
  92. ],
  93. [OPERATOR_LIST, 'list-title-not-found'],
  94. [OPERATOR_COMMENT, 'comment-not-found'],
  95. [OPERATOR_USER, 'user-username-not-found'],
  96. [OPERATOR_ASSIGNEE, 'user-username-not-found'],
  97. [OPERATOR_MEMBER, 'user-username-not-found'],
  98. ];
  99. constructor() {
  100. this._errors = {};
  101. this.operatorTags = {};
  102. this.operatorTagMap.forEach(([operator, tag]) => {
  103. this.operatorTags[operator] = tag;
  104. });
  105. this.colorMap = Boards.colorMap();
  106. }
  107. addError(operator, error) {
  108. if (!this._errors[operator]) {
  109. this._errors[operator] = [];
  110. }
  111. this._errors[operator].push(error);
  112. }
  113. addNotFound(operator, value) {
  114. if (typeof this.operatorTags[operator] === 'function') {
  115. this.addError(operator, this.operatorTags[operator](value));
  116. } else {
  117. this.addError(operator, { tag: this.operatorTags[operator], value });
  118. }
  119. }
  120. hasErrors() {
  121. return Object.entries(this._errors).length > 0;
  122. }
  123. errors() {
  124. const errs = [];
  125. // eslint-disable-next-line no-unused-vars
  126. Object.entries(this._errors).forEach(([, errors]) => {
  127. errors.forEach(err => {
  128. errs.push(err);
  129. });
  130. });
  131. return errs;
  132. }
  133. errorMessages() {
  134. const messages = [];
  135. // eslint-disable-next-line no-unused-vars
  136. Object.entries(this._errors).forEach(([, errors]) => {
  137. errors.forEach(err => {
  138. messages.push(TAPi18n.__(err.tag, err.value));
  139. });
  140. });
  141. return messages;
  142. }
  143. }
  144. export class Query {
  145. selector = {};
  146. projection = {};
  147. constructor(selector, projection) {
  148. this._errors = new QueryErrors();
  149. this.queryParams = new QueryParams();
  150. this.colorMap = Boards.colorMap();
  151. if (selector) {
  152. this.selector = selector;
  153. }
  154. if (projection) {
  155. this.projection = projection;
  156. }
  157. }
  158. hasErrors() {
  159. return this._errors.hasErrors();
  160. }
  161. errors() {
  162. return this._errors.errors();
  163. }
  164. errorMessages() {
  165. return this._errors.errorMessages();
  166. }
  167. getParams() {
  168. return this.queryParams.getParams();
  169. }
  170. addPredicate(operator, predicate) {
  171. this.queryParams.addPredicate(operator, predicate);
  172. }
  173. buildParams(queryText) {
  174. queryText = queryText.trim();
  175. // eslint-disable-next-line no-console
  176. //console.log('query:', query);
  177. if (!queryText) {
  178. return;
  179. }
  180. const reOperator1 = new RegExp(
  181. '^((?<operator>[\\p{Letter}\\p{Mark}]+):|(?<abbrev>[#@]))(?<value>[\\p{Letter}\\p{Mark}]+)(\\s+|$)',
  182. 'iu',
  183. );
  184. const reOperator2 = new RegExp(
  185. '^((?<operator>[\\p{Letter}\\p{Mark}]+):|(?<abbrev>[#@]))(?<quote>["\']*)(?<value>.*?)\\k<quote>(\\s+|$)',
  186. 'iu',
  187. );
  188. const reText = new RegExp('^(?<text>\\S+)(\\s+|$)', 'u');
  189. const reQuotedText = new RegExp(
  190. '^(?<quote>["\'])(?<text>.*?)\\k<quote>(\\s+|$)',
  191. 'u',
  192. );
  193. const reNegatedOperator = new RegExp('^-(?<operator>.*)$');
  194. const operators = {
  195. 'operator-board': OPERATOR_BOARD,
  196. 'operator-board-abbrev': OPERATOR_BOARD,
  197. 'operator-swimlane': OPERATOR_SWIMLANE,
  198. 'operator-swimlane-abbrev': OPERATOR_SWIMLANE,
  199. 'operator-list': OPERATOR_LIST,
  200. 'operator-list-abbrev': OPERATOR_LIST,
  201. 'operator-label': OPERATOR_LABEL,
  202. 'operator-label-abbrev': OPERATOR_LABEL,
  203. 'operator-user': OPERATOR_USER,
  204. 'operator-user-abbrev': OPERATOR_USER,
  205. 'operator-member': OPERATOR_MEMBER,
  206. 'operator-member-abbrev': OPERATOR_MEMBER,
  207. 'operator-assignee': OPERATOR_ASSIGNEE,
  208. 'operator-assignee-abbrev': OPERATOR_ASSIGNEE,
  209. 'operator-status': OPERATOR_STATUS,
  210. 'operator-due': OPERATOR_DUE,
  211. 'operator-created': OPERATOR_CREATED_AT,
  212. 'operator-modified': OPERATOR_MODIFIED_AT,
  213. 'operator-comment': OPERATOR_COMMENT,
  214. 'operator-has': OPERATOR_HAS,
  215. 'operator-sort': OPERATOR_SORT,
  216. 'operator-limit': OPERATOR_LIMIT,
  217. };
  218. const predicates = {
  219. due: {
  220. 'predicate-overdue': PREDICATE_OVERDUE,
  221. },
  222. durations: {
  223. 'predicate-week': PREDICATE_WEEK,
  224. 'predicate-month': PREDICATE_MONTH,
  225. 'predicate-quarter': PREDICATE_QUARTER,
  226. 'predicate-year': PREDICATE_YEAR,
  227. },
  228. status: {
  229. 'predicate-archived': PREDICATE_ARCHIVED,
  230. 'predicate-all': PREDICATE_ALL,
  231. 'predicate-open': PREDICATE_OPEN,
  232. 'predicate-ended': PREDICATE_ENDED,
  233. 'predicate-public': PREDICATE_PUBLIC,
  234. 'predicate-private': PREDICATE_PRIVATE,
  235. },
  236. sorts: {
  237. 'predicate-due': PREDICATE_DUE_AT,
  238. 'predicate-created': PREDICATE_CREATED_AT,
  239. 'predicate-modified': PREDICATE_MODIFIED_AT,
  240. },
  241. has: {
  242. 'predicate-description': PREDICATE_DESCRIPTION,
  243. 'predicate-checklist': PREDICATE_CHECKLIST,
  244. 'predicate-attachment': PREDICATE_ATTACHMENT,
  245. 'predicate-start': PREDICATE_START_AT,
  246. 'predicate-end': PREDICATE_END_AT,
  247. 'predicate-due': PREDICATE_DUE_AT,
  248. 'predicate-assignee': PREDICATE_ASSIGNEES,
  249. 'predicate-member': PREDICATE_MEMBERS,
  250. },
  251. };
  252. const predicateTranslations = {};
  253. Object.entries(predicates).forEach(([category, catPreds]) => {
  254. predicateTranslations[category] = {};
  255. Object.entries(catPreds).forEach(([tag, value]) => {
  256. predicateTranslations[category][TAPi18n.__(tag)] = value;
  257. });
  258. });
  259. // eslint-disable-next-line no-console
  260. // console.log('predicateTranslations:', predicateTranslations);
  261. const operatorMap = {};
  262. Object.entries(operators).forEach(([key, value]) => {
  263. operatorMap[TAPi18n.__(key).toLowerCase()] = value;
  264. });
  265. // eslint-disable-next-line no-console
  266. // console.log('operatorMap:', operatorMap);
  267. let text = '';
  268. while (queryText) {
  269. let m = queryText.match(reOperator1);
  270. if (!m) {
  271. m = queryText.match(reOperator2);
  272. if (m) {
  273. queryText = queryText.replace(reOperator2, '');
  274. }
  275. } else {
  276. queryText = queryText.replace(reOperator1, '');
  277. }
  278. if (m) {
  279. let op;
  280. if (m.groups.operator) {
  281. op = m.groups.operator.toLowerCase();
  282. } else {
  283. op = m.groups.abbrev.toLowerCase();
  284. }
  285. // eslint-disable-next-line no-prototype-builtins
  286. if (operatorMap.hasOwnProperty(op)) {
  287. const operator = operatorMap[op];
  288. let value = m.groups.value;
  289. if (operator === OPERATOR_LABEL) {
  290. if (value in this.colorMap) {
  291. value = this.colorMap[value];
  292. // console.log('found color:', value);
  293. }
  294. } else if (
  295. [OPERATOR_DUE, OPERATOR_CREATED_AT, OPERATOR_MODIFIED_AT].includes(
  296. operator,
  297. )
  298. ) {
  299. const days = parseInt(value, 10);
  300. let duration = null;
  301. if (isNaN(days)) {
  302. // duration was specified as text
  303. if (predicateTranslations.durations[value]) {
  304. duration = predicateTranslations.durations[value];
  305. let date = null;
  306. switch (duration) {
  307. case PREDICATE_WEEK:
  308. // eslint-disable-next-line no-case-declarations
  309. const week = moment().week();
  310. if (week === 52) {
  311. date = moment(1, 'W');
  312. date.set('year', date.year() + 1);
  313. } else {
  314. date = moment(week + 1, 'W');
  315. }
  316. break;
  317. case PREDICATE_MONTH:
  318. // eslint-disable-next-line no-case-declarations
  319. const month = moment().month();
  320. // .month() is zero indexed
  321. if (month === 11) {
  322. date = moment(1, 'M');
  323. date.set('year', date.year() + 1);
  324. } else {
  325. date = moment(month + 2, 'M');
  326. }
  327. break;
  328. case PREDICATE_QUARTER:
  329. // eslint-disable-next-line no-case-declarations
  330. const quarter = moment().quarter();
  331. if (quarter === 4) {
  332. date = moment(1, 'Q');
  333. date.set('year', date.year() + 1);
  334. } else {
  335. date = moment(quarter + 1, 'Q');
  336. }
  337. break;
  338. case PREDICATE_YEAR:
  339. date = moment(moment().year() + 1, 'YYYY');
  340. break;
  341. }
  342. if (date) {
  343. value = {
  344. operator: '$lt',
  345. value: date.format('YYYY-MM-DD'),
  346. };
  347. }
  348. } else if (
  349. operator === OPERATOR_DUE &&
  350. value === PREDICATE_OVERDUE
  351. ) {
  352. value = {
  353. operator: '$lt',
  354. value: moment().format('YYYY-MM-DD'),
  355. };
  356. } else {
  357. this.errors.addError(OPERATOR_DUE, {
  358. tag: 'operator-number-expected',
  359. value: { operator: op, value },
  360. });
  361. continue;
  362. }
  363. } else if (operator === OPERATOR_DUE) {
  364. value = {
  365. operator: '$lt',
  366. value: moment(moment().format('YYYY-MM-DD'))
  367. .add(days + 1, duration ? duration : 'days')
  368. .format(),
  369. };
  370. } else {
  371. value = {
  372. operator: '$gte',
  373. value: moment(moment().format('YYYY-MM-DD'))
  374. .subtract(days, duration ? duration : 'days')
  375. .format(),
  376. };
  377. }
  378. } else if (operator === OPERATOR_SORT) {
  379. let negated = false;
  380. const m = value.match(reNegatedOperator);
  381. if (m) {
  382. value = m.groups.operator;
  383. negated = true;
  384. }
  385. if (!predicateTranslations.sorts[value]) {
  386. this.errors.addError(OPERATOR_SORT, {
  387. tag: 'operator-sort-invalid',
  388. value,
  389. });
  390. continue;
  391. } else {
  392. value = {
  393. name: predicateTranslations.sorts[value],
  394. order: negated ? ORDER_DESCENDING : ORDER_ASCENDING,
  395. };
  396. }
  397. } else if (operator === OPERATOR_STATUS) {
  398. if (!predicateTranslations.status[value]) {
  399. this.errors.addError(OPERATOR_STATUS, {
  400. tag: 'operator-status-invalid',
  401. value,
  402. });
  403. continue;
  404. } else {
  405. value = predicateTranslations.status[value];
  406. }
  407. } else if (operator === OPERATOR_HAS) {
  408. let negated = false;
  409. const m = value.match(reNegatedOperator);
  410. if (m) {
  411. value = m.groups.operator;
  412. negated = true;
  413. }
  414. if (!predicateTranslations.has[value]) {
  415. this.errors.addError(OPERATOR_HAS, {
  416. tag: 'operator-has-invalid',
  417. value,
  418. });
  419. continue;
  420. } else {
  421. value = {
  422. field: predicateTranslations.has[value],
  423. exists: !negated,
  424. };
  425. }
  426. } else if (operator === OPERATOR_LIMIT) {
  427. const limit = parseInt(value, 10);
  428. if (isNaN(limit) || limit < 1) {
  429. this.errors.addError(OPERATOR_LIMIT, {
  430. tag: 'operator-limit-invalid',
  431. value,
  432. });
  433. continue;
  434. } else {
  435. value = limit;
  436. }
  437. }
  438. this.queryParams.addPredicate(operator, value);
  439. } else {
  440. this.errors.addError(OPERATOR_UNKNOWN, {
  441. tag: 'operator-unknown-error',
  442. value: op,
  443. });
  444. }
  445. continue;
  446. }
  447. m = queryText.match(reQuotedText);
  448. if (!m) {
  449. m = queryText.match(reText);
  450. if (m) {
  451. queryText = queryText.replace(reText, '');
  452. }
  453. } else {
  454. queryText = queryText.replace(reQuotedText, '');
  455. }
  456. if (m) {
  457. text += (text ? ' ' : '') + m.groups.text;
  458. }
  459. }
  460. // eslint-disable-next-line no-console
  461. // console.log('text:', text);
  462. this.queryParams.text = text;
  463. // eslint-disable-next-line no-console
  464. console.log('queryParams:', this.queryParams);
  465. }
  466. }