From b5fc4eeb5305f5b4976ddf39fd652d8b163ad2c1 Mon Sep 17 00:00:00 2001
From: Purexo <5005154+Purexo@users.noreply.github.com>
Date: Mon, 13 May 2024 18:08:00 +0200
Subject: [PATCH] fix: format-checkers regex to support `:emoji:` format (#97)
* fix invalid regex https://github.com/ES-Community/bot/assets/5005154/02494c78-25d3-4a80-be93-77309e211816
* add some unit tests
* change `FunctionChecker` type
use string instead Message for testing, all checks use `cleanContent` from `Message`
* test: disable WorkChronicles test if CI
---
src/format-checkers/Job.ts | 2 +-
src/format-checkers/Link.ts | 2 +-
src/format-checkers/Project.ts | 4 +-
src/framework/FormatChecker.ts | 11 +++--
tests/cron/WorkChronicles.spec.ts | 2 +-
tests/format-checkers/Link.test.ts | 75 ++++++++++++++++++++++++++++++
6 files changed, 87 insertions(+), 9 deletions(-)
create mode 100644 tests/format-checkers/Link.test.ts
diff --git a/src/format-checkers/Job.ts b/src/format-checkers/Job.ts
index 4df702a..0d656de 100644
--- a/src/format-checkers/Job.ts
+++ b/src/format-checkers/Job.ts
@@ -5,7 +5,7 @@ export default new FormatChecker({
name: 'Job',
description: 'Force le formatage du canal #jobs.',
channelName: 'jobs',
- checker({ cleanContent }, logger) {
+ checker(cleanContent, logger) {
const lines = cleanContent.split('\n');
const headerParts = lines[0].split(' - ');
diff --git a/src/format-checkers/Link.ts b/src/format-checkers/Link.ts
index 2c6e4db..2a4a1e3 100644
--- a/src/format-checkers/Link.ts
+++ b/src/format-checkers/Link.ts
@@ -4,7 +4,7 @@ import createRegExp from 'emoji-regex';
const unicodeEmojiRegexp = createRegExp().source;
const urlRegexp =
'?';
-const titleRegexp = `(?:[\\w- ]|(?:)|${unicodeEmojiRegexp})+`;
+const titleRegexp = `(?:[a-zA-Z0-9_\\- ]|(?:)|(?::\\w{2,32}:)|${unicodeEmojiRegexp})+`;
const linkRegexp = `^\\[( )?(\\*\\*)?${titleRegexp}\\2\\1\\](?:[^\\n])+ - ${urlRegexp}$`;
export default new FormatChecker({
diff --git a/src/format-checkers/Project.ts b/src/format-checkers/Project.ts
index f34eb91..e4f0fb5 100644
--- a/src/format-checkers/Project.ts
+++ b/src/format-checkers/Project.ts
@@ -4,7 +4,7 @@ import { FormatChecker } from '../framework/index.js';
const unicodeEmojiRegexp = createRegExp().source;
const urlRegexp =
'?';
-const headerRegexp = `(?:[\\w- ]|(?:)|${unicodeEmojiRegexp})`;
+const headerRegexp = `(?:[a-zA-Z0-9_\\- ]|(?:)|(?::\\w{2,32}:)|${unicodeEmojiRegexp})+`;
const projectRegexp = new RegExp(
`^\\*\\*${headerRegexp}+\\*\\*\\n\\n(?:.*\\n)+\n(?:(?:${headerRegexp}* )?${urlRegexp}\n)+$`,
);
@@ -14,7 +14,7 @@ export default new FormatChecker({
name: 'Project',
description: 'Force le formatage du canal #projets.',
channelName: 'projets',
- checker: ({ cleanContent }) => projectRegexp.test(cleanContent + '\n'),
+ checker: (cleanContent) => projectRegexp.test(cleanContent + '\n'),
examples: [
`**Nom du projet**\n\nDescription du projet\n\nhttps://github.com`,
],
diff --git a/src/framework/FormatChecker.ts b/src/framework/FormatChecker.ts
index 7f3b386..2526007 100644
--- a/src/framework/FormatChecker.ts
+++ b/src/framework/FormatChecker.ts
@@ -7,7 +7,7 @@ import { Base, BaseConfig } from './Base.js';
import { Bot } from './Bot.js';
import { findTextChannelByName, isTextChannel } from './helpers.js';
-type FunctionChecker = (message: Message, logger: Logger) => boolean;
+type FunctionChecker = (cleanContent: string, logger: Logger) => boolean;
export interface FormatCheckerConfig extends BaseConfig {
channelName: string;
@@ -38,6 +38,11 @@ export class FormatChecker extends Base {
this._messageUpdateHandler = this._messageUpdateHandler.bind(this);
}
+ isMessageValid(cleanContent: string, logger: Logger) {
+ if (this.checker instanceof RegExp) return this.checker.test(cleanContent);
+ return this.checker(cleanContent, logger);
+ }
+
async _messageHandler(message: Message): Promise {
if (this.bot === undefined) return;
if (
@@ -53,9 +58,7 @@ export class FormatChecker extends Base {
checkerName: this.name,
});
- if (this.checker instanceof RegExp) {
- if (this.checker.test(message.cleanContent)) return;
- } else if (this.checker(message, logger)) return;
+ if (this.isMessageValid(message.cleanContent, logger)) return;
const { cleanContent, author } = message;
diff --git a/tests/cron/WorkChronicles.spec.ts b/tests/cron/WorkChronicles.spec.ts
index 4fd1315..7efde1d 100644
--- a/tests/cron/WorkChronicles.spec.ts
+++ b/tests/cron/WorkChronicles.spec.ts
@@ -2,7 +2,7 @@ import { test, expect } from 'vitest';
import { getLastChronicle } from '../../src/crons/WorkChronicles.js';
-test('getLastChronicle', async () => {
+test.skipIf(process.env.CI)('getLastChronicle', async () => {
const chronicle = await getLastChronicle();
if (!chronicle) return;
diff --git a/tests/format-checkers/Link.test.ts b/tests/format-checkers/Link.test.ts
new file mode 100644
index 0000000..5ef1458
--- /dev/null
+++ b/tests/format-checkers/Link.test.ts
@@ -0,0 +1,75 @@
+import { describe, test, expect } from 'vitest';
+import Link from '../../src/format-checkers/Link.js';
+import { pino, transport } from 'pino';
+import { randomUUID } from 'node:crypto';
+
+describe('Link', () => {
+ const logger = pino(transport({ target: 'pino-pretty' })).child({
+ id: randomUUID(),
+ type: 'FormatChecker',
+ checkerName: 'Link',
+ });
+
+ test('valid without emoji', () => {
+ const message = `[**SUJET**] Votre description ici - https://github.com/es-community`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(true);
+ });
+
+ test('valid with unicode emoji', () => {
+ const message = `[👍] Votre description ici - https://github.com/es-community`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(true);
+ });
+
+ test('valid with multiple unicode emojis', () => {
+ const message = `[👍👌] Votre description ici - https://github.com/es-community`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(true);
+ });
+
+ test('valid with discord emoji', () => {
+ const message = `[<:adonis:793993785712312361>] Breadcrumbs automatiques pour les routes sur Adonis V6 - https://adonis-breadcrumbs.pages.dev`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(true);
+ });
+
+ test('valid with multiple discord emojis', () => {
+ const message = `[<:adonis:793993785712312361><:firefox:793993785712312369>] Breadcrumbs automatiques pour les routes sur Adonis V6 - https://adonis-breadcrumbs.pages.dev`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(true);
+ });
+
+ test('valid with multiple discord emojis mixed with unicode emojis', () => {
+ const message = `[<:adonis:793993785712312361>🌈<:firefox:793993785712312369>👍👌] Breadcrumbs automatiques pour les routes sur Adonis V6 - https://adonis-breadcrumbs.pages.dev`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(true);
+ });
+
+ test('valid with discord dismissed emoji', () => {
+ const message = `[:adonis:] Breadcrumbs automatiques pour les routes sur Adonis V6 - https://adonis-breadcrumbs.pages.dev`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(true);
+ });
+
+ test('invalid, link is missing', () => {
+ const message = `[**SUJET**] Votre description ici`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(false);
+ });
+
+ test('invalid, separator is missing', () => {
+ const message = `[**SUJET**] Votre description ici https://github.com/es-community`;
+ const isValid = Link.isMessageValid(message, logger);
+
+ expect(isValid).toBe(false);
+ });
+});