nodebb-plugin-mentions 4.5.3 → 4.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (45) hide show
  1. package/.eslintrc +2 -2
  2. package/LICENSE +7 -7
  3. package/README.md +13 -13
  4. package/languages/ar/notifications.json +2 -2
  5. package/languages/bg/notifications.json +2 -2
  6. package/languages/bn/notifications.json +2 -2
  7. package/languages/da/notifications.json +2 -2
  8. package/languages/de/notifications.json +5 -5
  9. package/languages/el/notifications.json +2 -2
  10. package/languages/en@pirate/notifications.json +4 -4
  11. package/languages/en_GB/mentions.json +3 -3
  12. package/languages/en_GB/notifications.json +7 -7
  13. package/languages/en_US/notifications.json +4 -4
  14. package/languages/es/notifications.json +3 -3
  15. package/languages/et/notifications.json +2 -2
  16. package/languages/fa_IR/notifications.json +4 -4
  17. package/languages/fi/notifications.json +2 -2
  18. package/languages/fr/notifications.json +5 -5
  19. package/languages/gl/notifications.json +2 -2
  20. package/languages/id/notifications.json +2 -2
  21. package/languages/it/notifications.json +3 -3
  22. package/languages/ja/notifications.json +4 -4
  23. package/languages/jbo/notifications.json +4 -4
  24. package/languages/ko/notifications.json +2 -2
  25. package/languages/lt/notifications.json +2 -2
  26. package/languages/ms/notifications.json +2 -2
  27. package/languages/nl/notifications.json +2 -2
  28. package/languages/pl/mentions.json +3 -3
  29. package/languages/pl/notifications.json +6 -6
  30. package/languages/pt-PT/notifications.json +6 -6
  31. package/languages/pt_BR/notifications.json +3 -3
  32. package/languages/ro/notifications.json +2 -2
  33. package/languages/rw/notifications.json +2 -2
  34. package/languages/sl/notifications.json +2 -2
  35. package/languages/sr/notifications.json +2 -2
  36. package/languages/sv/notifications.json +2 -2
  37. package/languages/tr/notifications.json +6 -6
  38. package/languages/vi/notifications.json +2 -2
  39. package/languages/zh-CN/mentions.json +3 -3
  40. package/languages/zh-CN/notifications.json +5 -5
  41. package/languages/zh_TW/notifications.json +2 -2
  42. package/library.js +62 -13
  43. package/package.json +4 -2
  44. package/plugin.json +31 -31
  45. package/static/.eslintrc +3 -3
package/.eslintrc CHANGED
@@ -1,3 +1,3 @@
1
- {
2
- "extends": "nodebb/lib"
1
+ {
2
+ "extends": "nodebb/lib"
3
3
  }
package/LICENSE CHANGED
@@ -1,8 +1,8 @@
1
- Copyright (c) 2013-2014, Julian Lam <julian@designcreateplay.com>
2
- All rights reserved.
3
-
4
- Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
5
-
6
- Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
7
- Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
1
+ Copyright (c) 2013-2014, Julian Lam <julian@designcreateplay.com>
2
+ All rights reserved.
3
+
4
+ Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
5
+
6
+ Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
7
+ Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
8
8
  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
package/README.md CHANGED
@@ -1,14 +1,14 @@
1
- # Username/Group Mentions
2
-
3
- This NodeBB plugin allows posters to reference (or *mention*) other users or groups on a NodeBB by simply
4
- precluding the `@` symbol before a username.
5
-
6
- A link is automatically added to the post.
7
-
8
- ## Installation
9
-
10
- This plugin is bundled with every NodeBB install. If not, you can install it via the Plugins page of the ACP.
11
-
12
- Alternatively,
13
-
1
+ # Username/Group Mentions
2
+
3
+ This NodeBB plugin allows posters to reference (or *mention*) other users or groups on a NodeBB by simply
4
+ precluding the `@` symbol before a username.
5
+
6
+ A link is automatically added to the post.
7
+
8
+ ## Installation
9
+
10
+ This plugin is bundled with every NodeBB install. If not, you can install it via the Plugins page of the ACP.
11
+
12
+ Alternatively,
13
+
14
14
  npm install nodebb-plugin-mentions
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> ذكرَ اسمك في <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> ذكرَ اسمك في <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> Ви спомена в <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> Ви спомена в <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong>, <strong>%2</strong> এ আপনার নাম উল্লেখ করেছেন"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong>, <strong>%2</strong> এ আপনার নাম উল্লেখ করেছেন"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> nævnte dig i <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> nævnte dig i <strong>%2</strong>"
3
3
  }
@@ -1,6 +1,6 @@
1
- {
2
- "mentions": "Erwähnungen",
3
- "user-mentioned-you-in": "<strong>%1</strong> erwähnte dich in <strong>%2</strong>",
4
- "user-mentioned-group-in": "<strong>%1</strong> erwähnte <strong>%2</strong> in <strong>%3</strong>",
5
- "notificationType_mention": "Wenn dich jemand erwähnt"
1
+ {
2
+ "mentions": "Erwähnungen",
3
+ "user-mentioned-you-in": "<strong>%1</strong> erwähnte dich in <strong>%2</strong>",
4
+ "user-mentioned-group-in": "<strong>%1</strong> erwähnte <strong>%2</strong> in <strong>%3</strong>",
5
+ "notificationType_mention": "Wenn dich jemand erwähnt"
6
6
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "Ο/Η <strong>%1</strong> σε ανέφερε στο <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "Ο/Η <strong>%1</strong> σε ανέφερε στο <strong>%2</strong>"
3
3
  }
@@ -1,4 +1,4 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> spoke yer name in <strong>%2</strong>",
3
- "user-mentioned-group-in": "<strong>%1</strong> called fer <strong>%2</strong> in <strong>%3</strong>"
4
- }
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> spoke yer name in <strong>%2</strong>",
3
+ "user-mentioned-group-in": "<strong>%1</strong> called fer <strong>%2</strong> in <strong>%3</strong>"
4
+ }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> mentioned you in <strong>%2</strong>"
3
- }
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> mentioned you in <strong>%2</strong>"
3
+ }
@@ -1,7 +1,7 @@
1
- {
2
- "mentions": "Mentions",
3
- "user-mentioned-you-in": "<strong>%1</strong> mentioned you in <strong>%2</strong>",
4
- "user-mentioned-you-in-room": "<strong>%1</strong> mentioned you in <strong class=\"text-nowrap\"><i class=\"fa %2\"></i>%3</strong>",
5
- "user-mentioned-group-in": "<strong>%1</strong> mentioned <strong>%2</strong> in <strong>%3</strong>",
6
- "notificationType-mention": "When someone mentions you"
7
- }
1
+ {
2
+ "mentions": "Mentions",
3
+ "user-mentioned-you-in": "<strong>%1</strong> mentioned you in <strong>%2</strong>",
4
+ "user-mentioned-you-in-room": "<strong>%1</strong> mentioned you in <strong class=\"text-nowrap\"><i class=\"fa %2\"></i>%3</strong>",
5
+ "user-mentioned-group-in": "<strong>%1</strong> mentioned <strong>%2</strong> in <strong>%3</strong>",
6
+ "notificationType-mention": "When someone mentions you"
7
+ }
@@ -1,4 +1,4 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> mentioned you in <strong>%2</strong>",
3
- "user-mentioned-group-in": "<strong>%1</strong> mentioned <strong>%2</strong> in <strong>%3</strong>"
4
- }
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> mentioned you in <strong>%2</strong>",
3
+ "user-mentioned-group-in": "<strong>%1</strong> mentioned <strong>%2</strong> in <strong>%3</strong>"
4
+ }
@@ -1,4 +1,4 @@
1
- {
2
- "mentions": "Menciones",
3
- "user-mentioned-you-in": "<strong>%1</strong> te mencionó en <strong>%2</strong>"
1
+ {
2
+ "mentions": "Menciones",
3
+ "user-mentioned-you-in": "<strong>%1</strong> te mencionó en <strong>%2</strong>"
4
4
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> mainis sind postituses <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> mainis sind postituses <strong>%2</strong>"
3
3
  }
@@ -1,4 +1,4 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> شما رو در <strong>%2</strong> منشن کرده",
3
- "user-mentioned-group-in": "<strong>%1</strong> گروه <strong>%2</strong> رو در <strong>%3</strong> منشن کرده"
4
- }
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> شما رو در <strong>%2</strong> منشن کرده",
3
+ "user-mentioned-group-in": "<strong>%1</strong> گروه <strong>%2</strong> رو در <strong>%3</strong> منشن کرده"
4
+ }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> mainitsi sinut viestissä <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> mainitsi sinut viestissä <strong>%2</strong>"
3
3
  }
@@ -1,6 +1,6 @@
1
- {
2
- "mentions": "Mentions",
3
- "user-mentioned-you-in": "<strong>%1</strong> vous a mentionné dans <strong>%2</strong>",
4
- "user-mentioned-group-in": "<strong>%1</strong> a mentionné <strong>%2</strong> dans <strong>%3</strong>",
5
- "notificationType_mention": "Lorsque quelqu'un vous mentionne"
1
+ {
2
+ "mentions": "Mentions",
3
+ "user-mentioned-you-in": "<strong>%1</strong> vous a mentionné dans <strong>%2</strong>",
4
+ "user-mentioned-group-in": "<strong>%1</strong> a mentionné <strong>%2</strong> dans <strong>%3</strong>",
5
+ "notificationType_mention": "Lorsque quelqu'un vous mentionne"
6
6
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> mencionóute en <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> mencionóute en <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> menyebut mu di <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> menyebut mu di <strong>%2</strong>"
3
3
  }
@@ -1,4 +1,4 @@
1
- {
2
- "mentions": "Menzioni",
3
- "user-mentioned-you-in": "<strong>%1</strong> ti ha menzionato in <strong>%2</strong>"
1
+ {
2
+ "mentions": "Menzioni",
3
+ "user-mentioned-you-in": "<strong>%1</strong> ti ha menzionato in <strong>%2</strong>"
4
4
  }
@@ -1,5 +1,5 @@
1
- {
2
- "mentions": "メンション",
3
- "user-mentioned-you-in": "<strong>%1</strong> mentioned you in <strong>%2</strong>",
4
- "user-mentioned-group-in": "<strong>%1</strong> mentioned <strong>%2</strong> in <strong>%3</strong>"
1
+ {
2
+ "mentions": "メンション",
3
+ "user-mentioned-you-in": "<strong>%1</strong> mentioned you in <strong>%2</strong>",
4
+ "user-mentioned-group-in": "<strong>%1</strong> mentioned <strong>%2</strong> in <strong>%3</strong>"
5
5
  }
@@ -1,4 +1,4 @@
1
- {
2
- "user-mentioned-you-in": "la .<strong>%1</strong>. klacpe do la .<strong>%2</strong>.",
3
- "user-mentioned-group-in": "la .<strong>%1</strong>. klacpe la .<strong>%2</strong>. la .<strong>%3</strong>."
4
- }
1
+ {
2
+ "user-mentioned-you-in": "la .<strong>%1</strong>. klacpe do la .<strong>%2</strong>.",
3
+ "user-mentioned-group-in": "la .<strong>%1</strong>. klacpe la .<strong>%2</strong>. la .<strong>%3</strong>."
4
+ }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong>님이 <strong>%2</strong>에서 나를 언급했습니다."
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong>님이 <strong>%2</strong>에서 나를 언급했습니다."
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> paminėjo Jus <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> paminėjo Jus <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> sebut anda di <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> sebut anda di <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "Onze naam is genoemd door <strong>%1</strong> in <strong>%2</strong>."
1
+ {
2
+ "user-mentioned-you-in": "Onze naam is genoemd door <strong>%1</strong> in <strong>%2</strong>."
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> wspomniał o Tobie w <strong>%2</strong>"
3
- }
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> wspomniał o Tobie w <strong>%2</strong>"
3
+ }
@@ -1,6 +1,6 @@
1
- {
2
- "mentions": "Wspomnienia",
3
- "user-mentioned-you-in": "<strong>%1</strong> wspomniał o Tobie w <strong>%2</strong>",
4
- "user-mentioned-group-in": "<strong>%1</strong> wspomniał <strong>%2</strong> w <strong>%3</strong>",
5
- "notificationType_mention": "Kiedy ktoś o Tobie wspomni"
6
- }
1
+ {
2
+ "mentions": "Wspomnienia",
3
+ "user-mentioned-you-in": "<strong>%1</strong> wspomniał o Tobie w <strong>%2</strong>",
4
+ "user-mentioned-group-in": "<strong>%1</strong> wspomniał <strong>%2</strong> w <strong>%3</strong>",
5
+ "notificationType_mention": "Kiedy ktoś o Tobie wspomni"
6
+ }
@@ -1,6 +1,6 @@
1
- {
2
- "mentions": "Menções",
3
- "user-mentioned-you-in": "<strong>%1</strong> mencionou-te no tópico <strong>%2</strong>",
4
- "user-mentioned-group-in": "<strong>%1</strong> mencionou o grupo <strong>%2</strong> no tópico <strong>%3</strong>",
5
- "notificationType_mention": "Quando alguém te menciona"
6
- }
1
+ {
2
+ "mentions": "Menções",
3
+ "user-mentioned-you-in": "<strong>%1</strong> mencionou-te no tópico <strong>%2</strong>",
4
+ "user-mentioned-group-in": "<strong>%1</strong> mencionou o grupo <strong>%2</strong> no tópico <strong>%3</strong>",
5
+ "notificationType_mention": "Quando alguém te menciona"
6
+ }
@@ -1,4 +1,4 @@
1
- {
2
- "mentions": "Menções",
3
- "user-mentioned-you-in": "<strong>%1</strong> mencionou você em <strong>%2</strong>"
1
+ {
2
+ "mentions": "Menções",
3
+ "user-mentioned-you-in": "<strong>%1</strong> mencionou você em <strong>%2</strong>"
4
4
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> te-a menționat în <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> te-a menționat în <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> yakuvuze muri <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> yakuvuze muri <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> te je omenil v <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> te je omenil v <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> вас помену у <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> вас помену у <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> nämnde dig i <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> nämnde dig i <strong>%2</strong>"
3
3
  }
@@ -1,6 +1,6 @@
1
- {
2
- "mentions": "Bahsedilmeler",
3
- "user-mentioned-you-in": "<strong>%1</strong> sizden şu konuda bahsetti: <strong>%2</strong>",
4
- "user-mentioned-group-in": "<strong>%1</strong> kullanıcısı <strong>%2</strong> grubundan şu konuda bahsetti: <strong>%3</strong>",
5
- "notificationType_mention": "Biri sizden bahsettiğinde"
6
- }
1
+ {
2
+ "mentions": "Bahsedilmeler",
3
+ "user-mentioned-you-in": "<strong>%1</strong> sizden şu konuda bahsetti: <strong>%2</strong>",
4
+ "user-mentioned-group-in": "<strong>%1</strong> kullanıcısı <strong>%2</strong> grubundan şu konuda bahsetti: <strong>%3</strong>",
5
+ "notificationType_mention": "Biri sizden bahsettiğinde"
6
+ }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> nhắc đến bạn trong <strong>%2</strong>"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> nhắc đến bạn trong <strong>%2</strong>"
3
3
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> 在 <strong>%2</strong> 中提到了您"
3
- }
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> 在 <strong>%2</strong> 中提到了您"
3
+ }
@@ -1,6 +1,6 @@
1
- {
2
- "mentions": "提到",
3
- "user-mentioned-you-in": "<strong>%1</strong> 在 <strong>%2</strong> 中提到了你",
4
- "user-mentioned-group-in": "<strong>%1</strong> 在 <strong>%3</strong> 中提到了 <strong>%2</strong>",
5
- "notificationType_mention": "当有人提到你时"
1
+ {
2
+ "mentions": "提到",
3
+ "user-mentioned-you-in": "<strong>%1</strong> 在 <strong>%2</strong> 中提到了你",
4
+ "user-mentioned-group-in": "<strong>%1</strong> 在 <strong>%3</strong> 中提到了 <strong>%2</strong>",
5
+ "notificationType_mention": "当有人提到你时"
6
6
  }
@@ -1,3 +1,3 @@
1
- {
2
- "user-mentioned-you-in": "<strong>%1</strong> 在 <strong>%2</strong>提到你"
1
+ {
2
+ "user-mentioned-you-in": "<strong>%1</strong> 在 <strong>%2</strong>提到你"
3
3
  }
package/library.js CHANGED
@@ -5,6 +5,7 @@
5
5
  const _ = require('lodash');
6
6
  const validator = require('validator');
7
7
  const entitiesDecode = require('html-entities').decode;
8
+ const cheerio = require('cheerio');
8
9
 
9
10
  const nconf = require.main.require('nconf');
10
11
  const winston = require.main.require('winston');
@@ -332,11 +333,12 @@ async function createNotification(postData, nidType, notificationText) {
332
333
  }
333
334
 
334
335
  Mentions.parsePost = async (data) => {
335
- if (!data || !data.postData || !data.postData.content) {
336
+ const { postData, type } = data;
337
+ if (!postData.content) {
336
338
  return data;
337
339
  }
338
340
 
339
- const parsed = await Mentions.parseRaw(data.postData.content);
341
+ const parsed = await Mentions.parseRaw(postData.content, type);
340
342
  data.postData.content = parsed;
341
343
  return data;
342
344
  };
@@ -345,7 +347,7 @@ function removePunctuationSuffix(string) {
345
347
  return string.replace(/[!?.]*$/, '');
346
348
  }
347
349
 
348
- function getMatches(content, isMarkdown = false) {
350
+ async function getMatches(content, isMarkdown = false) {
349
351
  const splitContent = utility.split(content, isMarkdown, false, true);
350
352
  let matches = [];
351
353
  splitContent.forEach((cleanedContent, i) => {
@@ -354,12 +356,38 @@ function getMatches(content, isMarkdown = false) {
354
356
  }
355
357
  });
356
358
 
357
- return { splitContent, matches };
359
+ const $ = cheerio.load(splitContent.join(''));
360
+ const anchors = $('a');
361
+ const urls = new Set();
362
+ Array.from(anchors).forEach((anchor) => {
363
+ const text = $(anchor).prop('innerText');
364
+ const match = text.match(regex);
365
+ if (match) {
366
+ urls.add($(anchor).attr('href'));
367
+ }
368
+ });
369
+
370
+ // Filter out urls that don't backreference to a remote id
371
+ const backrefs = await db.getObjectFields('remoteUrl:uid', Array.from(urls));
372
+ const urlAsIdExists = await db.isSortedSetMembers('usersRemote:lastCrawled', Array.from(urls));
373
+ const urlMap = new Map();
374
+ Array.from(urls).map(async (url, index) => {
375
+ if (backrefs[url] || urlAsIdExists[index]) {
376
+ urlMap.set(url, backrefs[url] || url);
377
+ }
378
+ });
379
+ let slugs = await User.getUsersFields(Array.from(urlMap.values()), ['userslug']);
380
+ slugs = slugs.map(({ userslug }) => userslug);
381
+ Array.from(urlMap.keys()).forEach((url, idx) => {
382
+ urlMap.set(url, `/user/${encodeURIComponent(slugs[idx])}`);
383
+ });
384
+
385
+ return { splitContent, matches, urlMap };
358
386
  }
359
387
 
360
388
  Mentions.getMatches = async (content) => {
361
389
  // Exported method only accepts markdown, also filters out dupes and matches to ensure slugs exist
362
- let { matches } = getMatches(content, true);
390
+ let { matches } = await getMatches(content, true);
363
391
  matches = await filterMatches(matches);
364
392
  const ids = await Promise.all(matches.map(async m => User.getUidByUserslug(m.slice(1).toLowerCase())));
365
393
  matches = matches.map((slug, idx) => (ids[idx] ? {
@@ -377,11 +405,15 @@ async function filterMatches(matches) {
377
405
  return matches.filter((m, i) => exists[[i]]);
378
406
  }
379
407
 
380
- Mentions.parseRaw = async (content) => {
408
+ Mentions.parseRaw = async (content, type = 'default') => {
409
+ if (type === 'plaintext') {
410
+ return content;
411
+ }
412
+
381
413
  // Note: Mentions.clean explicitly can't be called here because I need the content unstripped
382
- let { splitContent, matches } = getMatches(content);
414
+ let { splitContent, matches, urlMap } = await getMatches(content);
383
415
 
384
- if (!matches.length) {
416
+ if (!matches.length && !urlMap.size) {
385
417
  return content;
386
418
  }
387
419
 
@@ -395,6 +427,7 @@ Mentions.parseRaw = async (content) => {
395
427
  return atIndex !== 0 ? match.slice(atIndex) : match;
396
428
  });
397
429
 
430
+ // Convert matches to anchor html
398
431
  await Promise.all(matches.map(async (match) => {
399
432
  const slug = slugify(match.slice(1));
400
433
  match = removePunctuationSuffix(match);
@@ -402,7 +435,7 @@ Mentions.parseRaw = async (content) => {
402
435
  const cid = await categories.getCidByHandle(slug);
403
436
  const { groupExists, user, category } = await utils.promiseParallel({
404
437
  groupExists: Groups.existsBySlug(slug),
405
- user: User.getUserFields(uid, ['uid', 'username', 'fullname', 'url']),
438
+ user: User.getUserFields(uid, ['uid', 'username', 'userslug', 'fullname', 'url']),
406
439
  category: categories.getCategoryFields(cid, ['slug']),
407
440
  });
408
441
 
@@ -411,17 +444,20 @@ Mentions.parseRaw = async (content) => {
411
444
 
412
445
  switch (true) {
413
446
  case !!uid: {
414
- url = utils.isNumber(user.uid) ? `${nconf.get('url')}/user/${slug}` : (user.url || user.uid);
447
+ url = `/user/${encodeURIComponent(user.userslug)}`;
448
+ if (type.startsWith('activitypub') && !utils.isNumber(uid)) {
449
+ url = user.url || user.uid;
450
+ }
415
451
  break;
416
452
  }
417
453
 
418
454
  case !!cid: {
419
- url = `${nconf.get('url')}/category/${category.slug}`;
455
+ url = `/category/${category.slug}`;
420
456
  break;
421
457
  }
422
458
 
423
459
  case !!groupExists: {
424
- url = `${nconf.get('url')}/groups/${slug}`;
460
+ url = `/groups/${slug}`;
425
461
  break;
426
462
  }
427
463
  }
@@ -461,7 +497,20 @@ Mentions.parseRaw = async (content) => {
461
497
  }
462
498
  }));
463
499
 
464
- return splitContent.join('');
500
+ const parsed = splitContent.join('');
501
+
502
+ // Modify existing anchors to local profile
503
+ const $ = cheerio.load(parsed);
504
+ const anchors = $('a');
505
+ Array.from(anchors).forEach((anchor) => {
506
+ const $anchor = $(anchor);
507
+ const url = $anchor.attr('href');
508
+ if (urlMap.has(url)) {
509
+ $anchor.attr('href', urlMap.get(url));
510
+ }
511
+ });
512
+
513
+ return $.html();
465
514
  };
466
515
 
467
516
  Mentions.clean = function (input, isMarkdown, stripBlockquote, stripCode) {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "nodebb-plugin-mentions",
3
- "version": "4.5.3",
3
+ "version": "4.6.0",
4
4
  "description": "NodeBB Plugin that allows users to mention other users by prepending an '@' sign to their username",
5
5
  "main": "library.js",
6
6
  "repository": {
@@ -22,13 +22,15 @@
22
22
  "compatibility": "^4.0.0"
23
23
  },
24
24
  "dependencies": {
25
+ "cheerio": "^1.0.0-rc.12",
25
26
  "html-entities": "^2.3.2",
26
27
  "lodash": "4.17.21",
28
+ "sanitize-html": "^2.13.0",
27
29
  "validator": "^13.0.0"
28
30
  },
29
31
  "devDependencies": {
30
32
  "mocha": "10.4.0",
31
- "eslint": "9.2.0",
33
+ "eslint": "9.3.0",
32
34
  "eslint-config-nodebb": "0.2.1",
33
35
  "eslint-plugin-import": "2.29.1"
34
36
  }
package/plugin.json CHANGED
@@ -1,32 +1,32 @@
1
- {
2
- "id": "nodebb-plugin-mentions",
3
- "name": "Username Mentions",
4
- "description": "NodeBB Plugin that allows users to mention other users by prepending an '@' sign to their username",
5
- "url": "https://github.com/julianlam/nodebb-plugin-mentions",
6
- "library": "./library.js",
7
- "upgrades": [
8
- "upgrades/mentions_delete_mentions_set_zset.js"
9
- ],
10
- "hooks": [
11
- { "hook": "static:app.load", "method": "init" },
12
- { "hook": "filter:admin.header.build", "method": "addAdminNavigation" },
13
- { "hook": "filter:parse.post", "method": "parsePost" },
14
- { "hook": "filter:parse.raw", "method": "parseRaw" },
15
- { "hook": "action:post.save", "method": "notify" },
16
- { "hook": "action:post.edit", "method": "notify" },
17
- { "hook": "action:posts.purge", "method": "actionPostsPurge" },
18
- { "hook": "action:messaging.save", "method": "notifyMessage" },
19
- { "hook": "filter:notifications.addFilters", "method": "addFilters" },
20
- { "hook": "filter:user.notificationTypes", "method": "notificationTypes" },
21
- { "hook": "filter:users.addFields", "method": "addFields" }
22
- ],
23
- "scripts": [
24
- "static/autofill.js"
25
- ],
26
- "modules": {
27
- "../admin/plugins/mentions.js": "./static/admin.js"
28
- },
29
- "languages": "languages",
30
- "defaultLang": "en_GB",
31
- "templates": "templates"
1
+ {
2
+ "id": "nodebb-plugin-mentions",
3
+ "name": "Username Mentions",
4
+ "description": "NodeBB Plugin that allows users to mention other users by prepending an '@' sign to their username",
5
+ "url": "https://github.com/julianlam/nodebb-plugin-mentions",
6
+ "library": "./library.js",
7
+ "upgrades": [
8
+ "upgrades/mentions_delete_mentions_set_zset.js"
9
+ ],
10
+ "hooks": [
11
+ { "hook": "static:app.load", "method": "init" },
12
+ { "hook": "filter:admin.header.build", "method": "addAdminNavigation" },
13
+ { "hook": "filter:parse.post", "method": "parsePost" },
14
+ { "hook": "filter:parse.raw", "method": "parseRaw" },
15
+ { "hook": "action:post.save", "method": "notify" },
16
+ { "hook": "action:post.edit", "method": "notify" },
17
+ { "hook": "action:posts.purge", "method": "actionPostsPurge" },
18
+ { "hook": "action:messaging.save", "method": "notifyMessage" },
19
+ { "hook": "filter:notifications.addFilters", "method": "addFilters" },
20
+ { "hook": "filter:user.notificationTypes", "method": "notificationTypes" },
21
+ { "hook": "filter:users.addFields", "method": "addFields" }
22
+ ],
23
+ "scripts": [
24
+ "static/autofill.js"
25
+ ],
26
+ "modules": {
27
+ "../admin/plugins/mentions.js": "./static/admin.js"
28
+ },
29
+ "languages": "languages",
30
+ "defaultLang": "en_GB",
31
+ "templates": "templates"
32
32
  }
package/static/.eslintrc CHANGED
@@ -1,3 +1,3 @@
1
- {
2
- "extends": "nodebb/public"
3
- }
1
+ {
2
+ "extends": "nodebb/public"
3
+ }