diff --git a/sanitize_html/lib/sanitize_html.dart b/sanitize_html/lib/sanitize_html.dart
index dc298d48..fb4a6134 100644
--- a/sanitize_html/lib/sanitize_html.dart
+++ b/sanitize_html/lib/sanitize_html.dart
@@ -76,10 +76,14 @@ String sanitizeHtml(
bool Function(String)? allowElementId,
bool Function(String)? allowClassName,
Iterable? Function(String)? addLinkRel,
+ List? allowAttributes,
+ List? allowTags,
}) {
return SaneHtmlValidator(
allowElementId: allowElementId,
allowClassName: allowClassName,
addLinkRel: addLinkRel,
+ allowAttributes: allowAttributes,
+ allowTags: allowTags,
).sanitize(htmlString);
}
diff --git a/sanitize_html/lib/src/sane_html_validator.dart b/sanitize_html/lib/src/sane_html_validator.dart
index 32e39a81..caae2821 100644
--- a/sanitize_html/lib/src/sane_html_validator.dart
+++ b/sanitize_html/lib/src/sane_html_validator.dart
@@ -14,6 +14,7 @@
import 'package:html/dom.dart';
import 'package:html/parser.dart' as html_parser;
+import 'package:meta/meta.dart';
final _allowedElements = {
'H1',
@@ -177,6 +178,33 @@ bool _validUrl(String url) {
}
}
+bool _validBase64Image(String base64String) {
+ try {
+ final regex = RegExp(r'^data:image\/(png|jpeg|jpg|gif|bmp|svg\+xml);base64,[A-Za-z0-9+/]+={0,2}$');
+ return regex.hasMatch(base64String);
+ } catch (e) {
+ return false;
+ }
+}
+
+bool _validCIDImage(String cidString) {
+ try {
+ return cidString.startsWith('cid:');
+ } catch (e) {
+ return false;
+ }
+}
+
+@visibleForTesting
+bool validateBase64Image(String base64String) => _validBase64Image(base64String);
+
+@visibleForTesting
+bool validateCIDImage(String cidString) => _validCIDImage(cidString);
+
+bool _validImageSource(String url) {
+ return _validUrl(url) || _validBase64Image(url) || validateCIDImage(url);
+}
+
final _citeAttributeValidator = {
'cite': _validUrl,
};
@@ -187,8 +215,8 @@ final _elementAttributeValidators =
'href': _validLink,
},
'IMG': {
- 'src': _validUrl,
- 'longdesc': _validUrl,
+ 'src': _validImageSource,
+ 'longdesc': _validImageSource,
},
'DIV': {
'itemscope': _alwaysAllowed,
@@ -212,11 +240,15 @@ class SaneHtmlValidator {
final bool Function(String)? allowElementId;
final bool Function(String)? allowClassName;
final Iterable? Function(String)? addLinkRel;
+ final List? allowAttributes;
+ final List? allowTags;
SaneHtmlValidator({
required this.allowElementId,
required this.allowClassName,
required this.addLinkRel,
+ required this.allowAttributes,
+ required this.allowTags,
});
String sanitize(String htmlString) {
@@ -228,16 +260,19 @@ class SaneHtmlValidator {
void _sanitize(Node node) {
if (node is Element) {
final tagName = node.localName!.toUpperCase();
- if (!_allowedElements.contains(tagName)) {
+ if (!_allowedElements.contains(tagName)
+ && !(allowTags?.contains(tagName.toLowerCase()) ?? false)) {
node.remove();
return;
}
node.attributes.removeWhere((k, v) {
final attrName = k.toString();
if (attrName == 'id') {
- return allowElementId == null || !allowElementId!(v);
+ return allowAttributes?.contains('id') != true &&
+ (allowElementId == null || !allowElementId!(v));
}
if (attrName == 'class') {
+ if (allowAttributes?.contains('class') == true) return false;
if (allowClassName == null) return true;
node.classes.removeWhere((cn) => !allowClassName!(cn));
return node.classes.isEmpty;
@@ -269,6 +304,8 @@ class SaneHtmlValidator {
}
bool _isAttributeAllowed(String tagName, String attrName, String value) {
+ if (allowAttributes?.contains(attrName.toLowerCase()) == true) return true;
+
if (_alwaysAllowedAttributes.contains(attrName)) return true;
// Special validators for special attributes on special tags (href/src/cite)
diff --git a/sanitize_html/test/validate_base64_image_test.dart b/sanitize_html/test/validate_base64_image_test.dart
new file mode 100644
index 00000000..33d34271
--- /dev/null
+++ b/sanitize_html/test/validate_base64_image_test.dart
@@ -0,0 +1,46 @@
+import 'package:sanitize_html/src/sane_html_validator.dart';
+import 'package:test/test.dart';
+
+void main() {
+ group('validateBase64Image', () {
+ test('Valid Base64 PNG image string', () {
+ String validBase64PNG = '';
+ expect(validateBase64Image(validBase64PNG), isTrue);
+ });
+
+ test('Valid Base64 JPEG image string', () {
+ String validBase64JPEG = '';
+ expect(validateBase64Image(validBase64JPEG), isTrue);
+ });
+
+ test('Invalid Base64 image string (missing data:image/)', () {
+ String invalidBase64 = 'base64,iVBORw0KGgoAAAANSUhEUgAAAAUA';
+ expect(validateBase64Image(invalidBase64), isFalse);
+ });
+
+ test('Invalid Base64 image string (not base64 encoded)', () {
+ String invalidBase64 = 'data:image/png;notabase64string';
+ expect(validateBase64Image(invalidBase64), isFalse);
+ });
+
+ test('Valid Base64 SVG image string', () {
+ String validBase64SVG = '';
+ expect(validateBase64Image(validBase64SVG), isTrue);
+ });
+
+ test('Invalid Base64 image string (wrong image type)', () {
+ String invalidBase64Type = '';
+ expect(validateBase64Image(invalidBase64Type), isFalse);
+ });
+
+ test('Empty string', () {
+ String emptyString = '';
+ expect(validateBase64Image(emptyString), isFalse);
+ });
+
+ test('Non-image Base64 string', () {
+ String nonImageBase64 = 'data:text/plain;base64,dGVzdA=='; // Plain text Base64-encoded
+ expect(validateBase64Image(nonImageBase64), isFalse);
+ });
+ });
+}
diff --git a/sanitize_html/test/validate_cid_image_test.dart b/sanitize_html/test/validate_cid_image_test.dart
new file mode 100644
index 00000000..8cc0879a
--- /dev/null
+++ b/sanitize_html/test/validate_cid_image_test.dart
@@ -0,0 +1,18 @@
+import 'package:sanitize_html/src/sane_html_validator.dart';
+import 'package:test/test.dart';
+
+void main() {
+ group('validateCIDImage', () {
+ test('returns true for valid cid string', () {
+ expect(validateCIDImage('cid:12345'), true);
+ });
+
+ test('returns false for string without cid', () {
+ expect(validateCIDImage('https://example.com/image.png'), false);
+ });
+
+ test('returns false for empty string', () {
+ expect(validateCIDImage(''), false);
+ });
+ });
+}