glin-profanity 2.0.0 → 2.3.3

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 (124) hide show
  1. package/README.md +338 -74
  2. package/lib/cjs/packages/js/src/core/index.d.ts +4 -0
  3. package/lib/cjs/packages/js/src/core/index.js +55 -0
  4. package/lib/cjs/packages/js/src/core/index.js.map +1 -0
  5. package/lib/cjs/packages/js/src/core/types.d.ts +20 -0
  6. package/lib/cjs/packages/js/src/core/types.js +3 -0
  7. package/lib/cjs/packages/js/src/core/types.js.map +1 -0
  8. package/lib/cjs/{data → packages/js/src/data}/dictionary.d.ts +1 -0
  9. package/lib/cjs/packages/js/src/data/dictionary.js +54 -0
  10. package/lib/cjs/packages/js/src/data/dictionary.js.map +1 -0
  11. package/lib/cjs/{filters → packages/js/src/filters}/Filter.d.ts +13 -14
  12. package/lib/cjs/packages/js/src/filters/Filter.js +246 -0
  13. package/lib/cjs/packages/js/src/filters/Filter.js.map +1 -0
  14. package/lib/cjs/packages/js/src/hooks/useProfanityChecker.d.ts +11 -0
  15. package/lib/cjs/packages/js/src/hooks/useProfanityChecker.js +42 -0
  16. package/lib/cjs/packages/js/src/hooks/useProfanityChecker.js.map +1 -0
  17. package/lib/cjs/packages/js/src/index.d.ts +12 -0
  18. package/lib/cjs/packages/js/src/index.js +24 -0
  19. package/lib/cjs/packages/js/src/index.js.map +1 -0
  20. package/lib/cjs/packages/js/src/nlp/contextAnalyzer.d.ts +35 -0
  21. package/lib/cjs/packages/js/src/nlp/contextAnalyzer.js +203 -0
  22. package/lib/cjs/packages/js/src/nlp/contextAnalyzer.js.map +1 -0
  23. package/lib/cjs/packages/js/src/types/types.d.ts +56 -0
  24. package/lib/cjs/packages/js/src/types/types.js +14 -0
  25. package/lib/cjs/packages/js/src/types/types.js.map +1 -0
  26. package/lib/cjs/shared/dictionaries/spanish.json +72 -0
  27. package/lib/esm/packages/js/src/core/index.d.ts +4 -0
  28. package/lib/esm/packages/js/src/core/index.js +47 -0
  29. package/lib/esm/packages/js/src/core/index.js.map +1 -0
  30. package/lib/esm/packages/js/src/core/types.d.ts +20 -0
  31. package/lib/esm/packages/js/src/core/types.js +2 -0
  32. package/lib/esm/packages/js/src/core/types.js.map +1 -0
  33. package/lib/esm/{data → packages/js/src/data}/dictionary.d.ts +1 -0
  34. package/lib/esm/packages/js/src/data/dictionary.js +49 -0
  35. package/lib/esm/packages/js/src/data/dictionary.js.map +1 -0
  36. package/lib/esm/{filters → packages/js/src/filters}/Filter.d.ts +13 -14
  37. package/lib/esm/packages/js/src/filters/Filter.js +240 -0
  38. package/lib/esm/packages/js/src/filters/Filter.js.map +1 -0
  39. package/lib/esm/packages/js/src/hooks/useProfanityChecker.d.ts +11 -0
  40. package/lib/esm/packages/js/src/hooks/useProfanityChecker.js +38 -0
  41. package/lib/esm/packages/js/src/hooks/useProfanityChecker.js.map +1 -0
  42. package/lib/esm/packages/js/src/index.d.ts +12 -0
  43. package/lib/esm/packages/js/src/index.js +15 -0
  44. package/lib/esm/packages/js/src/index.js.map +1 -0
  45. package/lib/esm/packages/js/src/nlp/contextAnalyzer.d.ts +35 -0
  46. package/lib/esm/packages/js/src/nlp/contextAnalyzer.js +199 -0
  47. package/lib/esm/packages/js/src/nlp/contextAnalyzer.js.map +1 -0
  48. package/lib/esm/packages/js/src/types/types.d.ts +56 -0
  49. package/lib/esm/packages/js/src/types/types.js +11 -0
  50. package/lib/esm/packages/js/src/types/types.js.map +1 -0
  51. package/lib/esm/shared/dictionaries/spanish.json +72 -0
  52. package/package.json +26 -3
  53. package/lib/cjs/data/dictionary.js +0 -52
  54. package/lib/cjs/data/dictionary.js.map +0 -1
  55. package/lib/cjs/filters/Filter.js +0 -137
  56. package/lib/cjs/filters/Filter.js.map +0 -1
  57. package/lib/cjs/hooks/useProfanityChecker.d.ts +0 -20
  58. package/lib/cjs/hooks/useProfanityChecker.js +0 -58
  59. package/lib/cjs/hooks/useProfanityChecker.js.map +0 -1
  60. package/lib/cjs/index.d.ts +0 -3
  61. package/lib/cjs/index.js +0 -8
  62. package/lib/cjs/index.js.map +0 -1
  63. package/lib/cjs/types/types.d.ts +0 -14
  64. package/lib/cjs/types/types.js +0 -11
  65. package/lib/cjs/types/types.js.map +0 -1
  66. package/lib/esm/data/dictionary.js +0 -47
  67. package/lib/esm/data/dictionary.js.map +0 -1
  68. package/lib/esm/filters/Filter.js +0 -131
  69. package/lib/esm/filters/Filter.js.map +0 -1
  70. package/lib/esm/hooks/useProfanityChecker.d.ts +0 -20
  71. package/lib/esm/hooks/useProfanityChecker.js +0 -51
  72. package/lib/esm/hooks/useProfanityChecker.js.map +0 -1
  73. package/lib/esm/index.d.ts +0 -3
  74. package/lib/esm/index.js +0 -3
  75. package/lib/esm/index.js.map +0 -1
  76. package/lib/esm/types/types.d.ts +0 -14
  77. package/lib/esm/types/types.js +0 -8
  78. package/lib/esm/types/types.js.map +0 -1
  79. /package/lib/cjs/{data → shared/dictionaries}/Norwegian.json +0 -0
  80. /package/lib/cjs/{data → shared/dictionaries}/arabic.json +0 -0
  81. /package/lib/cjs/{data → shared/dictionaries}/chinese.json +0 -0
  82. /package/lib/cjs/{data → shared/dictionaries}/czech.json +0 -0
  83. /package/lib/cjs/{data → shared/dictionaries}/danish.json +0 -0
  84. /package/lib/cjs/{data → shared/dictionaries}/english.json +0 -0
  85. /package/lib/cjs/{data → shared/dictionaries}/esperanto.json +0 -0
  86. /package/lib/cjs/{data → shared/dictionaries}/finnish.json +0 -0
  87. /package/lib/cjs/{data → shared/dictionaries}/french.json +0 -0
  88. /package/lib/cjs/{data → shared/dictionaries}/german.json +0 -0
  89. /package/lib/cjs/{data → shared/dictionaries}/globalWhitelist.json +0 -0
  90. /package/lib/cjs/{data → shared/dictionaries}/hindi.json +0 -0
  91. /package/lib/cjs/{data → shared/dictionaries}/hungarian.json +0 -0
  92. /package/lib/cjs/{data → shared/dictionaries}/italian.json +0 -0
  93. /package/lib/cjs/{data → shared/dictionaries}/japanese.json +0 -0
  94. /package/lib/cjs/{data → shared/dictionaries}/korean.json +0 -0
  95. /package/lib/cjs/{data → shared/dictionaries}/persian.json +0 -0
  96. /package/lib/cjs/{data → shared/dictionaries}/polish.json +0 -0
  97. /package/lib/cjs/{data → shared/dictionaries}/portuguese.json +0 -0
  98. /package/lib/cjs/{data → shared/dictionaries}/russian.json +0 -0
  99. /package/lib/cjs/{data → shared/dictionaries}/swedish.json +0 -0
  100. /package/lib/cjs/{data → shared/dictionaries}/thai.json +0 -0
  101. /package/lib/cjs/{data → shared/dictionaries}/turkish.json +0 -0
  102. /package/lib/esm/{data → shared/dictionaries}/Norwegian.json +0 -0
  103. /package/lib/esm/{data → shared/dictionaries}/arabic.json +0 -0
  104. /package/lib/esm/{data → shared/dictionaries}/chinese.json +0 -0
  105. /package/lib/esm/{data → shared/dictionaries}/czech.json +0 -0
  106. /package/lib/esm/{data → shared/dictionaries}/danish.json +0 -0
  107. /package/lib/esm/{data → shared/dictionaries}/english.json +0 -0
  108. /package/lib/esm/{data → shared/dictionaries}/esperanto.json +0 -0
  109. /package/lib/esm/{data → shared/dictionaries}/finnish.json +0 -0
  110. /package/lib/esm/{data → shared/dictionaries}/french.json +0 -0
  111. /package/lib/esm/{data → shared/dictionaries}/german.json +0 -0
  112. /package/lib/esm/{data → shared/dictionaries}/globalWhitelist.json +0 -0
  113. /package/lib/esm/{data → shared/dictionaries}/hindi.json +0 -0
  114. /package/lib/esm/{data → shared/dictionaries}/hungarian.json +0 -0
  115. /package/lib/esm/{data → shared/dictionaries}/italian.json +0 -0
  116. /package/lib/esm/{data → shared/dictionaries}/japanese.json +0 -0
  117. /package/lib/esm/{data → shared/dictionaries}/korean.json +0 -0
  118. /package/lib/esm/{data → shared/dictionaries}/persian.json +0 -0
  119. /package/lib/esm/{data → shared/dictionaries}/polish.json +0 -0
  120. /package/lib/esm/{data → shared/dictionaries}/portuguese.json +0 -0
  121. /package/lib/esm/{data → shared/dictionaries}/russian.json +0 -0
  122. /package/lib/esm/{data → shared/dictionaries}/swedish.json +0 -0
  123. /package/lib/esm/{data → shared/dictionaries}/thai.json +0 -0
  124. /package/lib/esm/{data → shared/dictionaries}/turkish.json +0 -0
@@ -1,17 +1,5 @@
1
- import { Language, CheckProfanityResult } from '../types/types';
2
- interface FilterConfig {
3
- languages?: Language[];
4
- allLanguages?: boolean;
5
- caseSensitive?: boolean;
6
- wordBoundaries?: boolean;
7
- customWords?: string[];
8
- replaceWith?: string;
9
- severityLevels?: boolean;
10
- ignoreWords?: string[];
11
- logProfanity?: boolean;
12
- allowObfuscatedMatch?: boolean;
13
- fuzzyToleranceLevel?: number;
14
- }
1
+ import { CheckProfanityResult, SeverityLevel, FilterConfig } from '../types/types';
2
+ export type { FilterConfig };
15
3
  declare class Filter {
16
4
  private words;
17
5
  private caseSensitive;
@@ -22,12 +10,23 @@ declare class Filter {
22
10
  private logProfanity;
23
11
  private allowObfuscatedMatch;
24
12
  private fuzzyToleranceLevel;
13
+ private enableContextAware;
14
+ private contextWindow;
15
+ private confidenceThreshold;
16
+ private contextAnalyzer?;
17
+ private primaryLanguage;
25
18
  constructor(config?: FilterConfig);
19
+ private debugLog;
26
20
  private normalizeObfuscated;
27
21
  private getRegex;
28
22
  private isFuzzyToleranceMatch;
29
23
  private evaluateSeverity;
30
24
  isProfane(value: string): boolean;
25
+ matches(word: string): boolean;
31
26
  checkProfanity(text: string): CheckProfanityResult;
27
+ checkProfanityWithMinSeverity(text: string, minSeverity?: SeverityLevel): {
28
+ filteredWords: string[];
29
+ result: CheckProfanityResult;
30
+ };
32
31
  }
33
32
  export { Filter };
@@ -0,0 +1,246 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.Filter = void 0;
7
+ const dictionary_1 = __importDefault(require("../data/dictionary"));
8
+ const types_1 = require("../types/types");
9
+ const contextAnalyzer_1 = require("../nlp/contextAnalyzer");
10
+ class Filter {
11
+ constructor(config) {
12
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m;
13
+ const defaultLanguage = 'english';
14
+ this.enableContextAware = (_a = config === null || config === void 0 ? void 0 : config.enableContextAware) !== null && _a !== void 0 ? _a : false;
15
+ this.contextWindow = (_b = config === null || config === void 0 ? void 0 : config.contextWindow) !== null && _b !== void 0 ? _b : 3;
16
+ this.confidenceThreshold = (_c = config === null || config === void 0 ? void 0 : config.confidenceThreshold) !== null && _c !== void 0 ? _c : 0.7;
17
+ this.primaryLanguage = ((_d = config === null || config === void 0 ? void 0 : config.languages) === null || _d === void 0 ? void 0 : _d[0]) || defaultLanguage;
18
+ if (this.enableContextAware) {
19
+ this.contextAnalyzer = new contextAnalyzer_1.ContextAnalyzer({
20
+ contextWindow: this.contextWindow,
21
+ language: this.primaryLanguage,
22
+ domainWhitelists: ((_e = config === null || config === void 0 ? void 0 : config.domainWhitelists) === null || _e === void 0 ? void 0 : _e[this.primaryLanguage]) || []
23
+ });
24
+ }
25
+ let words = [];
26
+ this.caseSensitive = (_f = config === null || config === void 0 ? void 0 : config.caseSensitive) !== null && _f !== void 0 ? _f : false;
27
+ this.allowObfuscatedMatch = (_g = config === null || config === void 0 ? void 0 : config.allowObfuscatedMatch) !== null && _g !== void 0 ? _g : false;
28
+ this.wordBoundaries = (_h = config === null || config === void 0 ? void 0 : config.wordBoundaries) !== null && _h !== void 0 ? _h : !this.allowObfuscatedMatch;
29
+ this.replaceWith = config === null || config === void 0 ? void 0 : config.replaceWith;
30
+ this.severityLevels = (_j = config === null || config === void 0 ? void 0 : config.severityLevels) !== null && _j !== void 0 ? _j : false;
31
+ this.ignoreWords = new Set(((_k = config === null || config === void 0 ? void 0 : config.ignoreWords) === null || _k === void 0 ? void 0 : _k.map((word) => word.toLowerCase())) || []);
32
+ this.logProfanity = (_l = config === null || config === void 0 ? void 0 : config.logProfanity) !== null && _l !== void 0 ? _l : false;
33
+ this.fuzzyToleranceLevel = (_m = config === null || config === void 0 ? void 0 : config.fuzzyToleranceLevel) !== null && _m !== void 0 ? _m : 0.8;
34
+ if (config === null || config === void 0 ? void 0 : config.allLanguages) {
35
+ for (const lang in dictionary_1.default) {
36
+ if (dictionary_1.default.hasOwnProperty(lang)) {
37
+ words = [...words, ...dictionary_1.default[lang]];
38
+ }
39
+ }
40
+ }
41
+ else {
42
+ const languages = (config === null || config === void 0 ? void 0 : config.languages) || ['english'];
43
+ const languagesChecks = new Set(languages);
44
+ languagesChecks.forEach((lang) => {
45
+ words = [...words, ...dictionary_1.default[lang]];
46
+ });
47
+ }
48
+ if (config === null || config === void 0 ? void 0 : config.customWords) {
49
+ words = [...words, ...config.customWords];
50
+ }
51
+ this.words = new Map(words.map((word) => [word.toLowerCase(), 1]));
52
+ }
53
+ debugLog(...args) {
54
+ if (this.logProfanity) {
55
+ console.log('[glin-profanity]', ...args);
56
+ }
57
+ }
58
+ normalizeObfuscated(text) {
59
+ let normalized = text.replace(/([a-zA-Z])\1{1,}/g, '$1$1');
60
+ const charMap = {
61
+ '@': 'a',
62
+ $: 's',
63
+ '!': 'i',
64
+ '1': 'i',
65
+ '*': '',
66
+ };
67
+ normalized = normalized.replace(/[@$!1*]/g, (m) => charMap[m] || m);
68
+ return normalized;
69
+ }
70
+ getRegex(word) {
71
+ const flags = this.caseSensitive ? 'g' : 'gi';
72
+ const escapedWord = word.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
73
+ const boundary = this.wordBoundaries ? '\\b' : '';
74
+ return new RegExp(`${boundary}${escapedWord}${boundary}`, flags);
75
+ }
76
+ isFuzzyToleranceMatch(word, text) {
77
+ const simplifiedText = text.toLowerCase().replace(/[^a-z]/g, '');
78
+ const simplifiedWord = word.toLowerCase();
79
+ let matchCount = 0;
80
+ let index = 0;
81
+ for (let i = 0; i < simplifiedText.length; i++) {
82
+ if (simplifiedText[i] === simplifiedWord[index]) {
83
+ matchCount++;
84
+ index++;
85
+ if (index === simplifiedWord.length)
86
+ break;
87
+ }
88
+ }
89
+ const score = matchCount / simplifiedWord.length;
90
+ return score >= this.fuzzyToleranceLevel;
91
+ }
92
+ evaluateSeverity(word, text) {
93
+ if (this.wordBoundaries && this.getRegex(word).test(text)) {
94
+ return types_1.SeverityLevel.EXACT;
95
+ }
96
+ if (this.getRegex(word).test(text))
97
+ return types_1.SeverityLevel.EXACT;
98
+ if (this.isFuzzyToleranceMatch(word, text))
99
+ return types_1.SeverityLevel.FUZZY;
100
+ return undefined;
101
+ }
102
+ isProfane(value) {
103
+ let input = this.allowObfuscatedMatch
104
+ ? this.normalizeObfuscated(value)
105
+ : value;
106
+ for (const word of this.words.keys()) {
107
+ if (!this.ignoreWords.has(word.toLowerCase()) &&
108
+ this.evaluateSeverity(word, input) !== undefined) {
109
+ return true;
110
+ }
111
+ }
112
+ return false;
113
+ }
114
+ matches(word) {
115
+ return this.isProfane(word);
116
+ }
117
+ checkProfanity(text) {
118
+ // Backward compatibility: if not context-aware, run old logic
119
+ if (!this.enableContextAware) {
120
+ let input = this.allowObfuscatedMatch
121
+ ? this.normalizeObfuscated(text)
122
+ : text;
123
+ input = input.toLowerCase();
124
+ const profaneWords = [];
125
+ const severityMap = {};
126
+ for (const dictWord of this.words.keys()) {
127
+ if (this.ignoreWords.has(dictWord.toLowerCase()))
128
+ continue;
129
+ const severity = this.evaluateSeverity(dictWord, input);
130
+ if (severity !== undefined) {
131
+ const regex = this.getRegex(dictWord);
132
+ let match;
133
+ while ((match = regex.exec(input)) !== null) {
134
+ profaneWords.push(match[0]);
135
+ if (severityMap[match[0]] === undefined) {
136
+ severityMap[match[0]] = severity;
137
+ }
138
+ }
139
+ }
140
+ }
141
+ let processedText = text;
142
+ if (this.replaceWith && profaneWords.length > 0) {
143
+ const uniqueWords = Array.from(new Set(profaneWords));
144
+ for (const word of uniqueWords) {
145
+ const escaped = word.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
146
+ const replacementRegex = this.wordBoundaries
147
+ ? new RegExp(`\\b${escaped}\\b`, 'gi')
148
+ : new RegExp(escaped, 'gi');
149
+ processedText = processedText.replace(replacementRegex, this.replaceWith);
150
+ }
151
+ }
152
+ return {
153
+ containsProfanity: profaneWords.length > 0,
154
+ profaneWords: Array.from(new Set(profaneWords)),
155
+ processedText: this.replaceWith ? processedText : undefined,
156
+ severityMap: this.severityLevels && Object.keys(severityMap).length > 0
157
+ ? severityMap
158
+ : undefined,
159
+ };
160
+ }
161
+ // Context-aware path
162
+ let input = this.allowObfuscatedMatch
163
+ ? this.normalizeObfuscated(text)
164
+ : text;
165
+ input = input.toLowerCase();
166
+ const originalText = text;
167
+ const profaneWords = [];
168
+ const severityMap = {};
169
+ const matches = [];
170
+ for (const dictWord of this.words.keys()) {
171
+ if (this.ignoreWords.has(dictWord.toLowerCase()))
172
+ continue;
173
+ const severity = this.evaluateSeverity(dictWord, input);
174
+ if (severity !== undefined) {
175
+ const regex = this.getRegex(dictWord);
176
+ let match;
177
+ while ((match = regex.exec(input)) !== null) {
178
+ const matchedWord = match[0];
179
+ const matchIndex = match.index;
180
+ const matchObj = {
181
+ word: matchedWord,
182
+ index: matchIndex,
183
+ severity: severity
184
+ };
185
+ if (this.enableContextAware && this.contextAnalyzer) {
186
+ const contextResult = this.contextAnalyzer.analyzeContext(originalText, matchedWord, matchIndex);
187
+ matchObj.contextScore = contextResult.contextScore;
188
+ matchObj.reason = contextResult.reason;
189
+ matchObj.isWhitelisted = contextResult.isWhitelisted;
190
+ if (contextResult.isWhitelisted || (contextResult.contextScore > this.confidenceThreshold)) {
191
+ continue;
192
+ }
193
+ }
194
+ profaneWords.push(matchedWord);
195
+ if (severityMap[matchedWord] === undefined) {
196
+ severityMap[matchedWord] = severity;
197
+ }
198
+ matches.push(matchObj);
199
+ }
200
+ }
201
+ }
202
+ if (profaneWords.length > 0) {
203
+ this.debugLog('Detected:', profaneWords);
204
+ }
205
+ let processedText = text;
206
+ if (this.replaceWith && profaneWords.length > 0) {
207
+ const uniqueWords = Array.from(new Set(profaneWords));
208
+ for (const word of uniqueWords) {
209
+ const escaped = word.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
210
+ const replacementRegex = this.wordBoundaries
211
+ ? new RegExp(`\\b${escaped}\\b`, 'gi')
212
+ : new RegExp(escaped, 'gi');
213
+ processedText = processedText.replace(replacementRegex, this.replaceWith);
214
+ }
215
+ }
216
+ let contextScore;
217
+ if (this.enableContextAware && matches.length > 0) {
218
+ const totalScore = matches.reduce((sum, match) => sum + (match.contextScore || 0.5), 0);
219
+ contextScore = totalScore / matches.length;
220
+ }
221
+ return {
222
+ containsProfanity: profaneWords.length > 0,
223
+ profaneWords: Array.from(new Set(profaneWords)),
224
+ processedText: this.replaceWith ? processedText : undefined,
225
+ severityMap: this.severityLevels && Object.keys(severityMap).length > 0 ? severityMap : undefined,
226
+ matches: matches.length > 0 ? matches : undefined,
227
+ contextScore,
228
+ reason: matches.length > 0 ?
229
+ `Found ${matches.length} potential profanity matches` :
230
+ 'No profanity detected'
231
+ };
232
+ }
233
+ checkProfanityWithMinSeverity(text, minSeverity = types_1.SeverityLevel.EXACT) {
234
+ const result = this.checkProfanity(text);
235
+ const filteredWords = result.severityMap && result.profaneWords.length > 0
236
+ ? result.profaneWords.filter((word) => {
237
+ var _a;
238
+ const severity = (_a = result.severityMap) === null || _a === void 0 ? void 0 : _a[word];
239
+ return typeof severity === 'number' && severity >= minSeverity;
240
+ })
241
+ : [];
242
+ return { filteredWords, result };
243
+ }
244
+ }
245
+ exports.Filter = Filter;
246
+ //# sourceMappingURL=Filter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Filter.js","sourceRoot":"","sources":["../../../../../../src/filters/Filter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { ProfanityCheckerConfig, ProfanityCheckResult } from '../core/types';
2
+ import type { CheckProfanityResult } from '../types/types';
3
+ export type { ProfanityCheckerConfig };
4
+ export declare const useProfanityChecker: (config?: ProfanityCheckerConfig) => {
5
+ result: CheckProfanityResult;
6
+ checkText: (text: string) => ProfanityCheckResult;
7
+ checkTextAsync: (text: string) => Promise<ProfanityCheckResult>;
8
+ reset: () => void;
9
+ isDirty: boolean;
10
+ isWordProfane: (word: string) => boolean;
11
+ };
@@ -0,0 +1,42 @@
1
+ "use strict";
2
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
3
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
4
+ return new (P || (P = Promise))(function (resolve, reject) {
5
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
6
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
7
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
8
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
9
+ });
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.useProfanityChecker = void 0;
13
+ const react_1 = require("react");
14
+ const core_1 = require("../core");
15
+ const useProfanityChecker = (config) => {
16
+ var _a;
17
+ const [result, setResult] = (0, react_1.useState)(null);
18
+ const checkText = (0, react_1.useCallback)((text) => {
19
+ const checkResult = (0, core_1.checkProfanity)(text, config);
20
+ setResult(checkResult);
21
+ return checkResult;
22
+ }, [config]);
23
+ const checkTextAsync = (0, react_1.useCallback)((text) => __awaiter(void 0, void 0, void 0, function* () {
24
+ const checkResult = yield (0, core_1.checkProfanityAsync)(text, config);
25
+ setResult(checkResult);
26
+ return checkResult;
27
+ }), [config]);
28
+ const isWordProfaneCallback = (0, react_1.useCallback)((word) => {
29
+ return (0, core_1.isWordProfane)(word, config);
30
+ }, [config]);
31
+ const reset = (0, react_1.useCallback)(() => setResult(null), []);
32
+ return {
33
+ result,
34
+ checkText,
35
+ checkTextAsync,
36
+ reset,
37
+ isDirty: (_a = result === null || result === void 0 ? void 0 : result.containsProfanity) !== null && _a !== void 0 ? _a : false,
38
+ isWordProfane: isWordProfaneCallback,
39
+ };
40
+ };
41
+ exports.useProfanityChecker = useProfanityChecker;
42
+ //# sourceMappingURL=useProfanityChecker.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useProfanityChecker.js","sourceRoot":"","sources":["../../../../../../src/hooks/useProfanityChecker.ts"],"names":[],"mappings":";;;;;;;;;;;;AAAA,iCAA8C;AAC9C,kCAA6E;AAMtE,MAAM,mBAAmB,GAAG,CAAC,MAA+B,EAAE,EAAE;;IACrE,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,IAAA,gBAAQ,EAA8B,IAAI,CAAC,CAAC;IAExE,MAAM,SAAS,GAAG,IAAA,mBAAW,EAAC,CAAC,IAAY,EAAE,EAAE;QAC7C,MAAM,WAAW,GAAG,IAAA,qBAAc,EAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QACjD,SAAS,CAAC,WAAW,CAAC,CAAC;QACvB,OAAO,WAAW,CAAC;IACrB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC;IAEb,MAAM,cAAc,GAAG,IAAA,mBAAW,EAAC,CAAO,IAAY,EAAE,EAAE;QACxD,MAAM,WAAW,GAAG,MAAM,IAAA,0BAAmB,EAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QAC5D,SAAS,CAAC,WAAW,CAAC,CAAC;QACvB,OAAO,WAAW,CAAC;IACrB,CAAC,CAAA,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC;IAEb,MAAM,qBAAqB,GAAG,IAAA,mBAAW,EAAC,CAAC,IAAY,EAAE,EAAE;QACzD,OAAO,IAAA,oBAAa,EAAC,IAAI,EAAE,MAAM,CAAC,CAAC;IACrC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC;IAEb,MAAM,KAAK,GAAG,IAAA,mBAAW,EAAC,GAAG,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC;IAErD,OAAO;QACL,MAAM;QACN,SAAS;QACT,cAAc;QACd,KAAK;QACL,OAAO,EAAE,MAAA,MAAM,aAAN,MAAM,uBAAN,MAAM,CAAE,iBAAiB,mCAAI,KAAK;QAC3C,aAAa,EAAE,qBAAqB;KACrC,CAAC;AACJ,CAAC,CAAC;AA7BW,QAAA,mBAAmB,uBA6B9B"}
@@ -0,0 +1,12 @@
1
+ /**
2
+ * Glin-Profanity: Lightweight and efficient profanity detection and filtering.
3
+ *
4
+ * Provides multi-language support, context-aware filtering, and customizable
5
+ * configurations for detecting and filtering profane language in text inputs.
6
+ */
7
+ export { checkProfanity, checkProfanityAsync, isWordProfane } from './core';
8
+ export type { ProfanityCheckerConfig, ProfanityCheckResult } from './core/types';
9
+ export { useProfanityChecker } from './hooks/useProfanityChecker';
10
+ export { Filter } from './filters/Filter';
11
+ export { SeverityLevel } from './types/types';
12
+ export type { Language, CheckProfanityResult, FilterConfig, FilteredProfanityResult, Match, ContextAwareConfig } from './types/types';
@@ -0,0 +1,24 @@
1
+ "use strict";
2
+ /**
3
+ * Glin-Profanity: Lightweight and efficient profanity detection and filtering.
4
+ *
5
+ * Provides multi-language support, context-aware filtering, and customizable
6
+ * configurations for detecting and filtering profane language in text inputs.
7
+ */
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.SeverityLevel = exports.Filter = exports.useProfanityChecker = exports.isWordProfane = exports.checkProfanityAsync = exports.checkProfanity = void 0;
10
+ // Core API (framework-agnostic)
11
+ var core_1 = require("./core");
12
+ Object.defineProperty(exports, "checkProfanity", { enumerable: true, get: function () { return core_1.checkProfanity; } });
13
+ Object.defineProperty(exports, "checkProfanityAsync", { enumerable: true, get: function () { return core_1.checkProfanityAsync; } });
14
+ Object.defineProperty(exports, "isWordProfane", { enumerable: true, get: function () { return core_1.isWordProfane; } });
15
+ // React-specific
16
+ var useProfanityChecker_1 = require("./hooks/useProfanityChecker");
17
+ Object.defineProperty(exports, "useProfanityChecker", { enumerable: true, get: function () { return useProfanityChecker_1.useProfanityChecker; } });
18
+ // Advanced/Low-level
19
+ var Filter_1 = require("./filters/Filter");
20
+ Object.defineProperty(exports, "Filter", { enumerable: true, get: function () { return Filter_1.Filter; } });
21
+ // Legacy types (maintain compatibility)
22
+ var types_1 = require("./types/types");
23
+ Object.defineProperty(exports, "SeverityLevel", { enumerable: true, get: function () { return types_1.SeverityLevel; } });
24
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../src/index.ts"],"names":[],"mappings":";AAAA;;;;;GAKG;;;AAEH,gCAAgC;AAChC,+BAIgB;AAHd,sGAAA,cAAc,OAAA;AACd,2GAAA,mBAAmB,OAAA;AACnB,qGAAA,aAAa,OAAA;AAQf,iBAAiB;AACjB,mEAAkE;AAAzD,0HAAA,mBAAmB,OAAA;AAE5B,qBAAqB;AACrB,2CAA0C;AAAjC,gGAAA,MAAM,OAAA;AAEf,wCAAwC;AACxC,uCAA8C;AAArC,sGAAA,aAAa,OAAA"}
@@ -0,0 +1,35 @@
1
+ import { Language } from '../types/types';
2
+ export interface ContextAnalysisResult {
3
+ contextScore: number;
4
+ reason: string;
5
+ isWhitelisted: boolean;
6
+ }
7
+ export interface ContextConfig {
8
+ contextWindow: number;
9
+ language: Language;
10
+ domainWhitelists?: string[];
11
+ }
12
+ export declare class ContextAnalyzer {
13
+ private contextWindow;
14
+ private language;
15
+ private domainWhitelists;
16
+ constructor(config: ContextConfig);
17
+ /**
18
+ * Analyzes the context around a profanity match to determine if it should be flagged
19
+ */
20
+ analyzeContext(text: string, matchWord: string, matchIndex: number): ContextAnalysisResult;
21
+ private tokenize;
22
+ private findWordIndex;
23
+ private checkPhraseContext;
24
+ private isDomainWhitelisted;
25
+ private calculateSentimentScore;
26
+ private generateReason;
27
+ /**
28
+ * Updates the domain whitelist for this analyzer instance
29
+ */
30
+ updateDomainWhitelist(newWhitelist: string[]): void;
31
+ /**
32
+ * Adds words to the domain whitelist
33
+ */
34
+ addToDomainWhitelist(words: string[]): void;
35
+ }
@@ -0,0 +1,203 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ContextAnalyzer = void 0;
4
+ // Positive sentiment indicators
5
+ const POSITIVE_INDICATORS = new Set([
6
+ 'amazing', 'awesome', 'excellent', 'fantastic', 'great', 'love', 'wonderful',
7
+ 'brilliant', 'perfect', 'incredible', 'outstanding', 'superb', 'magnificent',
8
+ 'marvelous', 'spectacular', 'phenomenal', 'terrific', 'fabulous', 'divine',
9
+ 'best', 'good', 'nice', 'cool', 'sweet', 'rad', 'sick', 'dope', 'fire',
10
+ 'lit', 'epic', 'legendary', 'godlike', 'insane', 'crazy', 'wild', 'beast',
11
+ 'movie', 'film', 'show', 'song', 'music', 'game', 'book', 'restaurant',
12
+ 'food', 'dish', 'meal', 'place', 'spot', 'location', 'experience'
13
+ ]);
14
+ // Negative sentiment indicators
15
+ const NEGATIVE_INDICATORS = new Set([
16
+ 'hate', 'terrible', 'awful', 'horrible', 'disgusting', 'pathetic', 'stupid',
17
+ 'idiot', 'moron', 'loser', 'worthless', 'useless', 'garbage', 'trash',
18
+ 'suck', 'sucks', 'worst', 'bad', 'ugly', 'gross', 'nasty', 'annoying',
19
+ 'irritating', 'frustrating', 'disappointing', 'lame', 'weak', 'fail',
20
+ 'you', 'your', 'yourself', 'u', 'ur', 'ure', 'youre'
21
+ ]);
22
+ // Domain-specific positive contexts
23
+ const GAMING_POSITIVE = new Set([
24
+ 'player', 'gamer', 'team', 'squad', 'clan', 'guild', 'match', 'game',
25
+ 'round', 'level', 'boss', 'raid', 'quest', 'achievement', 'skill',
26
+ 'build', 'loadout', 'strategy', 'tactic', 'play', 'move', 'combo'
27
+ ]);
28
+ // Common positive phrases that might contain flagged words
29
+ const POSITIVE_PHRASES = new Map([
30
+ ['the bomb', 0.9], // "this movie is the bomb"
31
+ ['da bomb', 0.9], // slang for "the best"
32
+ ['bomb.com', 0.9], // website reference
33
+ ['bomb diggity', 0.9], // slang for excellent
34
+ ['photo bomb', 0.8], // photography term
35
+ ['bath bomb', 0.8], // cosmetic product
36
+ ['bomb squad', 0.7], // could be neutral/positive in gaming
37
+ ]);
38
+ // Negative phrases that should remain flagged
39
+ const NEGATIVE_PHRASES = new Map([
40
+ ['you are', 0.1], // "you are [profanity]"
41
+ ['ur a', 0.1], // "ur a [profanity]"
42
+ ['such a', 0.2], // "such a [profanity]"
43
+ ['fucking', 0.1], // intensifier, usually negative
44
+ ['damn', 0.2], // mild profanity, context dependent
45
+ ]);
46
+ class ContextAnalyzer {
47
+ constructor(config) {
48
+ this.contextWindow = config.contextWindow;
49
+ this.language = config.language;
50
+ this.domainWhitelists = new Set(config.domainWhitelists || []);
51
+ }
52
+ /**
53
+ * Analyzes the context around a profanity match to determine if it should be flagged
54
+ */
55
+ analyzeContext(text, matchWord, matchIndex) {
56
+ const words = this.tokenize(text);
57
+ const matchWordIndex = this.findWordIndex(words, matchIndex);
58
+ if (matchWordIndex === -1) {
59
+ return {
60
+ contextScore: 0.5,
61
+ reason: 'Could not locate match in tokenized text',
62
+ isWhitelisted: false
63
+ };
64
+ }
65
+ // Extract context window
66
+ const startIndex = Math.max(0, matchWordIndex - this.contextWindow);
67
+ const endIndex = Math.min(words.length, matchWordIndex + this.contextWindow + 1);
68
+ const contextWords = words.slice(startIndex, endIndex);
69
+ const contextText = contextWords.join(' ').toLowerCase();
70
+ // Check for exact phrase matches first
71
+ const phraseResult = this.checkPhraseContext(contextText, matchWord);
72
+ if (phraseResult) {
73
+ return phraseResult;
74
+ }
75
+ // Check domain-specific whitelists
76
+ if (this.isDomainWhitelisted(contextWords, matchWord)) {
77
+ return {
78
+ contextScore: 0.8,
79
+ reason: 'Domain-specific whitelist match',
80
+ isWhitelisted: true
81
+ };
82
+ }
83
+ // Perform sentiment analysis
84
+ const sentimentScore = this.calculateSentimentScore(contextWords, matchWordIndex - startIndex);
85
+ return {
86
+ contextScore: sentimentScore,
87
+ reason: this.generateReason(sentimentScore, contextWords),
88
+ isWhitelisted: false
89
+ };
90
+ }
91
+ tokenize(text) {
92
+ // Simple tokenization - split on whitespace and punctuation
93
+ return text.toLowerCase()
94
+ .replace(/[^\w\s]/g, ' ')
95
+ .split(/\s+/)
96
+ .filter(word => word.length > 0);
97
+ }
98
+ findWordIndex(words, charIndex) {
99
+ // This is a simplified approach - in production, you'd want more robust mapping
100
+ // For now, we'll estimate based on the character position
101
+ let currentPos = 0;
102
+ for (let i = 0; i < words.length; i++) {
103
+ if (currentPos >= charIndex) {
104
+ return Math.max(0, i - 1);
105
+ }
106
+ currentPos += words[i].length + 1; // +1 for space
107
+ }
108
+ return words.length - 1;
109
+ }
110
+ checkPhraseContext(contextText, matchWord) {
111
+ // Check positive phrases
112
+ for (const [phrase, score] of POSITIVE_PHRASES.entries()) {
113
+ if (contextText.includes(phrase)) {
114
+ return {
115
+ contextScore: score,
116
+ reason: `Positive phrase detected: "${phrase}"`,
117
+ isWhitelisted: true
118
+ };
119
+ }
120
+ }
121
+ // Check negative phrases
122
+ for (const [phrase, score] of NEGATIVE_PHRASES.entries()) {
123
+ if (contextText.includes(phrase)) {
124
+ return {
125
+ contextScore: score,
126
+ reason: `Negative phrase detected: "${phrase}"`,
127
+ isWhitelisted: false
128
+ };
129
+ }
130
+ }
131
+ return null;
132
+ }
133
+ isDomainWhitelisted(contextWords, matchWord) {
134
+ // Check if any domain whitelist words are present
135
+ for (const word of contextWords) {
136
+ if (this.domainWhitelists.has(word) || GAMING_POSITIVE.has(word)) {
137
+ return true;
138
+ }
139
+ }
140
+ return false;
141
+ }
142
+ calculateSentimentScore(contextWords, matchPosition) {
143
+ let positiveCount = 0;
144
+ let negativeCount = 0;
145
+ let totalWords = contextWords.length;
146
+ // Weight words closer to the match more heavily
147
+ for (let i = 0; i < contextWords.length; i++) {
148
+ const word = contextWords[i];
149
+ const distance = Math.abs(i - matchPosition);
150
+ const weight = Math.max(0.1, 1 - (distance * 0.2)); // Closer words have higher weight
151
+ if (POSITIVE_INDICATORS.has(word)) {
152
+ positiveCount += weight;
153
+ }
154
+ else if (NEGATIVE_INDICATORS.has(word)) {
155
+ negativeCount += weight;
156
+ }
157
+ }
158
+ // Calculate base score
159
+ const totalSentiment = positiveCount + negativeCount;
160
+ if (totalSentiment === 0) {
161
+ return 0.5; // Neutral if no sentiment indicators
162
+ }
163
+ const rawScore = positiveCount / totalSentiment;
164
+ // Apply context-specific adjustments
165
+ let adjustedScore = rawScore;
166
+ // If there are personal pronouns (you, your), lean towards negative
167
+ const hasPersonalPronouns = contextWords.some(word => ['you', 'your', 'u', 'ur'].includes(word));
168
+ if (hasPersonalPronouns && rawScore < 0.7) {
169
+ adjustedScore *= 0.7; // Reduce score when personal pronouns are present
170
+ }
171
+ // If there are object/thing references, lean towards positive
172
+ const hasObjectReferences = contextWords.some(word => ['movie', 'song', 'game', 'book', 'show', 'this', 'that', 'it'].includes(word));
173
+ if (hasObjectReferences && rawScore > 0.3) {
174
+ adjustedScore = Math.min(1, adjustedScore * 1.3); // Boost score for object references
175
+ }
176
+ return Math.max(0, Math.min(1, adjustedScore));
177
+ }
178
+ generateReason(score, contextWords) {
179
+ if (score >= 0.7) {
180
+ return 'Positive context detected - likely not profanity';
181
+ }
182
+ else if (score <= 0.3) {
183
+ return 'Negative context detected - likely profanity';
184
+ }
185
+ else {
186
+ return 'Neutral context - uncertain classification';
187
+ }
188
+ }
189
+ /**
190
+ * Updates the domain whitelist for this analyzer instance
191
+ */
192
+ updateDomainWhitelist(newWhitelist) {
193
+ this.domainWhitelists = new Set(newWhitelist);
194
+ }
195
+ /**
196
+ * Adds words to the domain whitelist
197
+ */
198
+ addToDomainWhitelist(words) {
199
+ words.forEach(word => this.domainWhitelists.add(word.toLowerCase()));
200
+ }
201
+ }
202
+ exports.ContextAnalyzer = ContextAnalyzer;
203
+ //# sourceMappingURL=contextAnalyzer.js.map