diff --git a/__tests__/ExpensiMark-HTML-test.js b/__tests__/ExpensiMark-HTML-test.js
index f0408322..67dabe84 100644
--- a/__tests__/ExpensiMark-HTML-test.js
+++ b/__tests__/ExpensiMark-HTML-test.js
@@ -2100,7 +2100,7 @@ describe('Video markdown conversion to html tag', () => {
expect(parser.replace(testString, {shouldKeepRawInput: true})).toBe(resultString);
})
- test('Single video with extra cached attribues', () => {
+ test('Single video with extra cached attributes with videoAttributeCache', () => {
const testString = '![test](https://example.com/video.mp4)';
const resultString = '';
expect(parser.replace(testString, {
@@ -2112,6 +2112,18 @@ describe('Video markdown conversion to html tag', () => {
})).toBe(resultString);
})
+ test('Single video with extra cached attributes with mediaAttributeCache', () => {
+ const testString = '![test](https://example.com/video.mp4)';
+ const resultString = '';
+ expect(parser.replace(testString, {
+ extras: {
+ mediaAttributeCache: {
+ 'https://example.com/video.mp4': 'data-expensify-height="100" data-expensify-width="100"'
+ }
+ }
+ })).toBe(resultString);
+ })
+
test('Text containing image and video', () => {
const testString = 'An image of a banana: ![banana](https://example.com/banana.png) and a video of a banana: ![banana](https://example.com/banana.mp4)';
const resultString = 'An image of a banana: and a video of a banana: ';
@@ -2223,6 +2235,31 @@ describe('Image markdown conversion to html tag', () => {
'';
expect(parser.replace(testString, {shouldKeepRawInput: true})).toBe(resultString);
});
+
+ test('Single image with extra cached attributes using mediaAttributeCache', () => {
+ const testString = '![test](https://example.com/image.jpg)';
+ const resultString = '';
+ expect(parser.replace(testString, {
+ extras: {
+ mediaAttributeCache: {
+ 'https://example.com/image.jpg': 'data-expensify-height="100" data-expensify-width="100"'
+ }
+ }
+ })).toBe(resultString);
+ })
+
+ test('Single image with extra cached attributes using videAttributeCache', () => {
+ const testString = '![test](https://example.com/image.jpg)';
+ const resultString = '';
+ expect(parser.replace(testString, {
+ extras: {
+ videoAttributeCache: {
+ 'https://example.com/image.jpg': 'data-expensify-height="100" data-expensify-width="100"'
+ }
+ }
+ })).toBe(resultString);
+ })
+
});
describe('room mentions', () => {
diff --git a/__tests__/ExpensiMark-Markdown-test.js b/__tests__/ExpensiMark-Markdown-test.js
index 1ebfd21a..0a70d3de 100644
--- a/__tests__/ExpensiMark-Markdown-test.js
+++ b/__tests__/ExpensiMark-Markdown-test.js
@@ -867,6 +867,50 @@ describe('Image tag conversion to markdown', () => {
const resultString = '![```code```](https://example.com/image.png)';
expect(parser.htmlToMarkdown(testString)).toBe(resultString);
});
+
+ test('Cache extra attributes for img with alt with mediaAttributeCachingFn', () => {
+ const mediaAttributeCachingFn = jest.fn();
+ const testString = '';
+ const resultString = '![altText](https://example.com/image.png)';
+ const extras = {
+ mediaAttributeCachingFn,
+ };
+ expect(parser.htmlToMarkdown(testString, extras)).toBe(resultString);
+ expect(mediaAttributeCachingFn).toHaveBeenCalledWith("https://example.com/image.png", 'data-expensify-width="100" data-expensify-height="500" data-name="newName" data-expensify-source="expensify-source"')
+ });
+
+ test('Cache extra attributes for img with alt with cacheVideoAttributes', () => {
+ const cacheVideoAttributes = jest.fn();
+ const testString = '';
+ const resultString = '![altText](https://example.com/image.png)';
+ const extras = {
+ cacheVideoAttributes,
+ };
+ expect(parser.htmlToMarkdown(testString, extras)).toBe(resultString);
+ expect(cacheVideoAttributes).toHaveBeenCalledWith("https://example.com/image.png", 'data-expensify-width="100" data-expensify-height="500" data-name="newName" data-expensify-source="expensify-source"')
+ });
+
+ test('Cache extra attributes for img without alt with mediaAttributeCachingFn', () => {
+ const mediaAttributeCachingFn = jest.fn();
+ const testString = '';
+ const resultString = '!(https://example.com/image.png)';
+ const extras = {
+ mediaAttributeCachingFn,
+ };
+ expect(parser.htmlToMarkdown(testString, extras)).toBe(resultString);
+ expect(mediaAttributeCachingFn).toHaveBeenCalledWith("https://example.com/image.png", 'data-expensify-width="100" data-expensify-height="500" data-name="newName" data-expensify-source="expensify-source"')
+ });
+
+ test('Cache extra attributes for img without alt with cacheVideoAttributes', () => {
+ const cacheVideoAttributes = jest.fn();
+ const testString = '';
+ const resultString = '!(https://example.com/image.png)';
+ const extras = {
+ cacheVideoAttributes,
+ };
+ expect(parser.htmlToMarkdown(testString, extras)).toBe(resultString);
+ expect(cacheVideoAttributes).toHaveBeenCalledWith("https://example.com/image.png", 'data-expensify-width="100" data-expensify-height="500" data-name="newName" data-expensify-source="expensify-source"')
+ });
});
describe('Video tag conversion to markdown', () => {
@@ -882,7 +926,7 @@ describe('Video tag conversion to markdown', () => {
expect(parser.htmlToMarkdown(testString)).toBe(resultString);
})
- test('While convert video, cache some extra attributes from the video tag', () => {
+ test('Video with extra attributes to be cached with cacheVideoAttributes', () => {
const cacheVideoAttributes = jest.fn();
const testString = '';
const resultString = '![video](https://example.com/video.mp4)';
@@ -892,6 +936,17 @@ describe('Video tag conversion to markdown', () => {
expect(parser.htmlToMarkdown(testString, extras)).toBe(resultString);
expect(cacheVideoAttributes).toHaveBeenCalledWith("https://example.com/video.mp4", ' data-expensify-width="100" data-expensify-height="500" data-expensify-thumbnail-url="https://image.com/img.jpg"')
})
+
+ test('Video with extra attributes to be cached with mediaAttributeCachingFn', () => {
+ const mediaAttributeCachingFn = jest.fn();
+ const testString = '';
+ const resultString = '![video](https://example.com/video.mp4)';
+ const extras = {
+ mediaAttributeCachingFn,
+ };
+ expect(parser.htmlToMarkdown(testString, extras)).toBe(resultString);
+ expect(mediaAttributeCachingFn).toHaveBeenCalledWith("https://example.com/video.mp4", ' data-expensify-width="100" data-expensify-height="500" data-expensify-thumbnail-url="https://image.com/img.jpg"')
+ })
})
describe('Tag names starting with common charaters', () => {
diff --git a/__tests__/ExpensiMark-test.js b/__tests__/ExpensiMark-test.js
index 31983401..41c0a2c9 100644
--- a/__tests__/ExpensiMark-test.js
+++ b/__tests__/ExpensiMark-test.js
@@ -1,6 +1,7 @@
/* eslint-disable max-len */
import ExpensiMark from '../lib/ExpensiMark';
import * as Utils from '../lib/utils';
+import {any, string} from "prop-types";
const parser = new ExpensiMark();
@@ -48,3 +49,55 @@ test('Test extract link from Markdown link syntax', () => {
const links = ['https://new.expensify.com/'];
expect(parser.extractLinksInMarkdownComment(comment)).toStrictEqual(links);
});
+
+describe('Test ExpensiMark getAttributeCache', () => {
+ const expensiMark = new ExpensiMark();
+
+ describe('For attrCachingFn', () => {
+ test('If mediaAttributeCachingFn is provided, returns it', () => {
+ const extras = {
+ mediaAttributeCachingFn: jest.fn(),
+ }
+ expect(expensiMark.getAttributeCache(extras).attrCachingFn).toBe(extras.mediaAttributeCachingFn);
+ })
+
+ test('If mediaAttributeCachingFn is not provided, returns cacheVideoAttributes', () => {
+ const extras = {
+ cacheVideoAttributes: jest.fn(),
+ }
+ expect(expensiMark.getAttributeCache(extras).attrCachingFn).toBe(extras.cacheVideoAttributes);
+ })
+
+ test('If mediaAttributeCachingFn and cacheVideoAttributes are not provided, returns undefined', () => {
+ const extras = {}
+ expect(expensiMark.getAttributeCache(extras).attrCachingFn).toBe(undefined);
+ })
+ });
+
+ describe('For attrCache', () => {
+ test('If mediaAttributeCache is provided, returns it', () => {
+ const extras = {
+ mediaAttributeCache: jest.fn(),
+ }
+ expect(expensiMark.getAttributeCache(extras).attrCache).toBe(extras.mediaAttributeCache);
+ })
+
+ test('If mediaAttributeCache is not provided, returns videoAttributeCache', () => {
+ const extras = {
+ videoAttributeCache: jest.fn(),
+ }
+ expect(expensiMark.getAttributeCache(extras).attrCache).toBe(extras.videoAttributeCache);
+ })
+
+ test('If mediaAttributeCache and videoAttributeCache are not provided, returns undefined', () => {
+ const extras = {}
+ expect(expensiMark.getAttributeCache(extras).attrCache).toBe(undefined);
+ })
+ });
+
+ test('If no extras are undefined, returns undefined for both attrCachingFn and attrCache', () => {
+ const {attrCachingFn, attrCache} = expensiMark.getAttributeCache(undefined);
+ expect(attrCachingFn).toBe(undefined);
+ expect(attrCache).toBe(undefined);
+ })
+});
diff --git a/lib/ExpensiMark.ts b/lib/ExpensiMark.ts
index 65e72d3f..0d870fd2 100644
--- a/lib/ExpensiMark.ts
+++ b/lib/ExpensiMark.ts
@@ -9,9 +9,31 @@ import * as Utils from './utils';
type Extras = {
reportIDToName?: Record;
accountIDToName?: Record;
+
+ /**
+ * @deprecated Replaced with mediaAttributeCachingFn
+ */
cacheVideoAttributes?: (vidSource: string, attrs: string) => void;
+
+ /**
+ * @deprecated Replaced with mediaAttributeCache
+ */
videoAttributeCache?: Record;
+
+ /**
+ * Function used to cache HTML tag attributes during conversion to and from Markdown
+ * @param mediaSource URI to media source
+ * @param attrs Additional attributes to be cached
+ */
+ mediaAttributeCachingFn?: (mediaSource: string, attrs: string) => void;
+
+ /**
+ * Key/value cache for HTML tag attributes, where the key is media source URI, value is cached attributes
+ */
+ mediaAttributeCache?: Record;
};
+export type {Extras};
+
const EXTRAS_DEFAULT = {};
type ReplacementFn = (extras: Extras, ...matches: string[]) => string;
@@ -62,6 +84,17 @@ const MARKDOWN_VIDEO_REGEX = new RegExp(
const SLACK_SPAN_NEW_LINE_TAG = '';
export default class ExpensiMark {
+ getAttributeCache = (extras?: Extras) => {
+ if (!extras) {
+ return {attrCachingFn: undefined, attrCache: undefined};
+ }
+
+ return {
+ attrCachingFn: extras.mediaAttributeCachingFn ?? extras.cacheVideoAttributes,
+ attrCache: extras.mediaAttributeCache ?? extras.videoAttributeCache,
+ };
+ };
+
static Log = new Logger({
serverLoggingCallback: () => undefined,
// eslint-disable-next-line no-console
@@ -171,11 +204,13 @@ export default class ExpensiMark {
* @return Returns the HTML video tag
*/
replacement: (extras, _match, videoName, videoSource) => {
- const extraAttrs = extras && extras.videoAttributeCache && extras.videoAttributeCache[videoSource];
+ const attrCache = this.getAttributeCache(extras).attrCache;
+ const extraAttrs = attrCache && attrCache[videoSource];
return ``;
},
rawInputReplacement: (extras, _match, videoName, videoSource) => {
- const extraAttrs = extras && extras.videoAttributeCache && extras.videoAttributeCache[videoSource];
+ const attrCache = this.getAttributeCache(extras).attrCache;
+ const extraAttrs = attrCache && attrCache[videoSource];
return ``;
},
},
@@ -245,13 +280,21 @@ export default class ExpensiMark {
* tag.
* Additional sanitization is done to the alt attribute to prevent parsing it further to html by later
* rules.
+ * Additional tags from cache are applied to the result HTML.
*/
{
name: 'image',
regex: MARKDOWN_IMAGE_REGEX,
- replacement: (_extras, _match, g1, g2) => ``,
- rawInputReplacement: (_extras, _match, g1, g2) =>
- ``,
+ replacement: (extras, _match, imgAlt, imgSource) => {
+ const attrCache = this.getAttributeCache(extras).attrCache;
+ const extraAttrs = attrCache && attrCache[imgSource];
+ return ``;
+ },
+ rawInputReplacement: (extras, _match, imgAlt, imgSource) => {
+ const attrCache = this.getAttributeCache(extras).attrCache;
+ const extraAttrs = attrCache && attrCache[imgSource];
+ return ``;
+ },
},
/**
@@ -658,13 +701,38 @@ export default class ExpensiMark {
{
name: 'image',
- regex: /<]*src\s*=\s*(['"])(.*?)\1(?:[^><]*alt\s*=\s*(['"])(.*?)\3)?[^><]*>*(?![^<][\s\S]*?(<\/pre>|<\/code>))/gi,
- replacement: (_extras, _match, _g1, g2, _g3, g4) => {
- if (g4) {
- return `![${g4}](${g2})`;
+ regex: /<]*src\s*=\s*(['"])(.*?)\1(.*?)>(?![^<][\s\S]*?(<\/pre>|<\/code>))/gi,
+ /**
+ * @param extras - The extras object
+ * @param _match - The full match
+ * @param _g1 - The first capture group (the quote)
+ * @param imgSource - The second capture group - src attribute value
+ * @param imgAttrs - The third capture group - any attributes after src
+ * @returns The markdown image tag
+ */
+ replacement: (extras, _match, _g1, imgSource, imgAttrs) => {
+ // Extract alt attribute from imgAttrs
+ let altText = '';
+ const altRegex = /alt\s*=\s*(['"])(.*?)\1/i;
+ const altMatch = imgAttrs.match(altRegex);
+ const attrCachingFn = this.getAttributeCache(extras).attrCachingFn;
+ let attributes = imgAttrs;
+ if (altMatch) {
+ altText = altMatch[2];
+ // Remove the alt attribute from imgAttrs
+ attributes = attributes.replace(altRegex, '');
}
-
- return `!(${g2})`;
+ // Remove trailing slash and extra whitespace
+ attributes = attributes.replace(/\s*\/\s*$/, '').trim();
+ // Cache attributes without alt and trailing slash
+ if (imgAttrs && attrCachingFn && typeof attrCachingFn === 'function') {
+ attrCachingFn(imgSource, attributes);
+ }
+ // Return the markdown image tag
+ if (altText) {
+ return `![${altText}](${imgSource})`;
+ }
+ return `!(${imgSource})`;
},
},
@@ -681,8 +749,10 @@ export default class ExpensiMark {
* @returns The markdown video tag
*/
replacement: (extras, _match, _g1, videoSource, videoAttrs, videoName) => {
- if (videoAttrs && extras && extras.cacheVideoAttributes && typeof extras.cacheVideoAttributes === 'function') {
- extras.cacheVideoAttributes(videoSource, videoAttrs);
+ const attrCachingFn = this.getAttributeCache(extras).attrCachingFn;
+
+ if (videoAttrs && attrCachingFn && typeof attrCachingFn === 'function') {
+ attrCachingFn(videoSource, videoAttrs);
}
if (videoName) {
return `![${videoName}](${videoSource})`;