search.js 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. // eslint-disable-next-line no-unused-vars
  2. const logger = require('@alias/logger')('growi:service:search');
  3. class SearchService {
  4. constructor(crowi) {
  5. this.crowi = crowi;
  6. this.configManager = crowi.configManager;
  7. this.isErrorOccuredOnHealthcheck = null;
  8. this.isErrorOccuredOnSearching = null;
  9. try {
  10. this.delegator = this.generateDelegator();
  11. }
  12. catch (err) {
  13. logger.error(err);
  14. }
  15. if (this.isConfigured) {
  16. this.delegator.init();
  17. this.registerUpdateEvent();
  18. }
  19. }
  20. get isConfigured() {
  21. return this.delegator != null;
  22. }
  23. get isReachable() {
  24. return this.isConfigured && !this.isErrorOccuredOnHealthcheck && !this.isErrorOccuredOnSearching;
  25. }
  26. get isSearchboxEnabled() {
  27. return this.configManager.getConfig('crowi', 'app:searchboxSslUrl') != null;
  28. }
  29. get isElasticsearchEnabled() {
  30. return this.configManager.getConfig('crowi', 'app:elasticsearchUri') != null;
  31. }
  32. generateDelegator() {
  33. logger.info('Initializing search delegator');
  34. if (this.isSearchboxEnabled) {
  35. logger.info('Searchbox is enabled');
  36. const SearchboxDelegator = require('./search-delegator/searchbox.js');
  37. return new SearchboxDelegator(this.configManager, this.crowi.socketIoService);
  38. }
  39. if (this.isElasticsearchEnabled) {
  40. logger.info('Elasticsearch (not Searchbox) is enabled');
  41. const ElasticsearchDelegator = require('./search-delegator/elasticsearch.js');
  42. return new ElasticsearchDelegator(this.configManager, this.crowi.socketIoService);
  43. }
  44. }
  45. registerUpdateEvent() {
  46. const pageEvent = this.crowi.event('page');
  47. pageEvent.on('create', this.delegator.syncPageUpdated.bind(this.delegator));
  48. pageEvent.on('update', this.delegator.syncPageUpdated.bind(this.delegator));
  49. pageEvent.on('deleteCompletely', this.delegator.syncPagesDeletedCompletely.bind(this.delegator));
  50. pageEvent.on('delete', this.delegator.syncPageDeleted.bind(this.delegator));
  51. pageEvent.on('createMany', this.delegator.syncDescendantsPagesUpdated.bind(this.delegator));
  52. const bookmarkEvent = this.crowi.event('bookmark');
  53. bookmarkEvent.on('create', this.delegator.syncBookmarkChanged.bind(this.delegator));
  54. bookmarkEvent.on('delete', this.delegator.syncBookmarkChanged.bind(this.delegator));
  55. const tagEvent = this.crowi.event('tag');
  56. tagEvent.on('update', this.delegator.syncTagChanged.bind(this.delegator));
  57. }
  58. resetErrorStatus() {
  59. this.isErrorOccuredOnHealthcheck = false;
  60. this.isErrorOccuredOnSearching = false;
  61. }
  62. async reconnectClient() {
  63. logger.info('Try to reconnect...');
  64. this.delegator.initClient();
  65. try {
  66. await this.getInfoForHealth();
  67. logger.info('Reconnecting succeeded.');
  68. this.resetErrorStatus();
  69. }
  70. catch (err) {
  71. throw err;
  72. }
  73. }
  74. async getInfo() {
  75. try {
  76. return await this.delegator.getInfo();
  77. }
  78. catch (err) {
  79. logger.error(err);
  80. throw err;
  81. }
  82. }
  83. async getInfoForHealth() {
  84. try {
  85. const result = await this.delegator.getInfoForHealth();
  86. this.isErrorOccuredOnHealthcheck = false;
  87. return result;
  88. }
  89. catch (err) {
  90. logger.error(err);
  91. // switch error flag, `isErrorOccuredOnHealthcheck` to be `false`
  92. this.isErrorOccuredOnHealthcheck = true;
  93. throw err;
  94. }
  95. }
  96. async getInfoForAdmin() {
  97. return this.delegator.getInfoForAdmin();
  98. }
  99. async normalizeIndices() {
  100. return this.delegator.normalizeIndices();
  101. }
  102. async rebuildIndex() {
  103. return this.delegator.rebuildIndex();
  104. }
  105. async searchKeyword(keyword, user, userGroups, searchOpts) {
  106. try {
  107. return await this.delegator.searchKeyword(keyword, user, userGroups, searchOpts);
  108. }
  109. catch (err) {
  110. logger.error(err);
  111. // switch error flag, `isReachable` to be `false`
  112. this.isErrorOccuredOnSearching = true;
  113. throw err;
  114. }
  115. }
  116. }
  117. module.exports = SearchService;