query-classes.js 15 KB

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