Refactored Linkifier class into single function
Modified design so that linkify interacts with MessageBodyBuilder through callbacks. Signed-off-by: RMidhunSuresh <rmidhunsuresh@gmail.com>
This commit is contained in:
parent
dca649dfd4
commit
787d438a74
2 changed files with 12 additions and 173 deletions
|
@ -1,173 +0,0 @@
|
||||||
import { MessageBodyBuilder } from "./MessageBodyBuilder.js";
|
|
||||||
|
|
||||||
export class Linkifier {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param {String} text Text to linkify
|
|
||||||
*/
|
|
||||||
constructor(text) {
|
|
||||||
this._text = text;
|
|
||||||
this._curr = 0;
|
|
||||||
this._message = new MessageBodyBuilder();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Separate string into text, newlines and add them into message object.
|
|
||||||
* @param {String} text
|
|
||||||
*/
|
|
||||||
_addTextToMessage(text) {
|
|
||||||
const components = text.split("\n");
|
|
||||||
components.slice(0, -1).forEach(t => {
|
|
||||||
this._message.insertText(t);
|
|
||||||
this._message.insertNewline();
|
|
||||||
});
|
|
||||||
const [last] = components.slice(-1);
|
|
||||||
this._message.insertText(last);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add text from this._curr upto start of supplied match into message object.
|
|
||||||
* If match is not provided, everything from this._curr to the end of
|
|
||||||
* this._text is added as text to the message object.
|
|
||||||
* @param {Array} [match] regex match
|
|
||||||
*/
|
|
||||||
_handleText(match) {
|
|
||||||
const index = match?.index;
|
|
||||||
const text = this._text.slice(this._curr, index);
|
|
||||||
this._addTextToMessage(text);
|
|
||||||
const len = match?.[0].length;
|
|
||||||
this._curr = index + len;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Splits message text into parts (text, newline and links)
|
|
||||||
* @returns {MessageBodyBuilder} Object representation of chat message
|
|
||||||
*/
|
|
||||||
linkify() {
|
|
||||||
const regex = /(?:https|http|ftp):\/\/[a-zA-Z0-9:.\[\]#-]+(?:\/[^\s]*[^\s.,?!]|[^\s\u{80}-\u{10ffff}.,?!])/gui
|
|
||||||
const matches = this._text.matchAll(regex);
|
|
||||||
for (let match of matches) {
|
|
||||||
const link = match[0];
|
|
||||||
this._handleText(match);
|
|
||||||
this._message.insertLink(link, link);
|
|
||||||
}
|
|
||||||
this._handleText();
|
|
||||||
return this._message;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function tests() {
|
|
||||||
|
|
||||||
function linkify(text) {
|
|
||||||
const obj = new Linkifier(text);
|
|
||||||
return obj.linkify();
|
|
||||||
}
|
|
||||||
|
|
||||||
function test(assert, input, output) {
|
|
||||||
output = new MessageBodyBuilder(output);
|
|
||||||
input = linkify(input);
|
|
||||||
assert.deepEqual(input, output);
|
|
||||||
}
|
|
||||||
|
|
||||||
function testLink(assert, link, expectFail = false) {
|
|
||||||
const input = link;
|
|
||||||
const output = expectFail ? [{ type: "text", text: input }] :
|
|
||||||
[{ type: "link", url: input, text: input }];
|
|
||||||
test(assert, input, output);
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
// Tests for text
|
|
||||||
"Text only": assert => {
|
|
||||||
const input = "This is a sentence";
|
|
||||||
const output = [{ type: "text", text: input }];
|
|
||||||
test(assert, input, output);
|
|
||||||
},
|
|
||||||
|
|
||||||
"Text with newline": assert => {
|
|
||||||
const input = "This is a sentence.\nThis is another sentence.";
|
|
||||||
const output = [
|
|
||||||
{ type: "text", text: "This is a sentence." },
|
|
||||||
{ type: "newline" },
|
|
||||||
{ type: "text", text: "This is another sentence." }
|
|
||||||
];
|
|
||||||
test(assert, input, output);
|
|
||||||
},
|
|
||||||
|
|
||||||
"Text with newline & trailing newline": assert => {
|
|
||||||
const input = "This is a sentence.\nThis is another sentence.\n";
|
|
||||||
const output = [
|
|
||||||
{ type: "text", text: "This is a sentence." },
|
|
||||||
{ type: "newline" },
|
|
||||||
{ type: "text", text: "This is another sentence." },
|
|
||||||
{ type: "newline" }
|
|
||||||
];
|
|
||||||
test(assert, input, output);
|
|
||||||
},
|
|
||||||
|
|
||||||
// Tests for links
|
|
||||||
"Link with host": assert => {
|
|
||||||
testLink(assert, "https://matrix.org");
|
|
||||||
},
|
|
||||||
|
|
||||||
"Link with host & path": assert => {
|
|
||||||
testLink(assert, "https://matrix.org/docs/develop");
|
|
||||||
},
|
|
||||||
|
|
||||||
"Link with host & fragment": assert => {
|
|
||||||
testLink(assert, "https://matrix.org#test");
|
|
||||||
},
|
|
||||||
|
|
||||||
"Link with host & query": assert => {
|
|
||||||
testLink(assert, "https://matrix.org/?foo=bar");
|
|
||||||
},
|
|
||||||
|
|
||||||
"Complex link": assert => {
|
|
||||||
const link = "https://www.foobar.com/url?sa=t&rct=j&q=&esrc=s&source" +
|
|
||||||
"=web&cd=&cad=rja&uact=8&ved=2ahUKEwjyu7DJ-LHwAhUQyzgGHc" +
|
|
||||||
"OKA70QFjAAegQIBBAD&url=https%3A%2F%2Fmatrix.org%2Fdocs%" +
|
|
||||||
"2Fprojects%2Fclient%2Felement%2F&usg=AOvVaw0xpENrPHv_R-" +
|
|
||||||
"ERkyacR2Bd";
|
|
||||||
testLink(assert, link);
|
|
||||||
},
|
|
||||||
|
|
||||||
"Localhost link": assert => {
|
|
||||||
testLink(assert, "http://localhost");
|
|
||||||
testLink(assert, "http://localhost:3000");
|
|
||||||
},
|
|
||||||
|
|
||||||
"IPV4 link": assert => {
|
|
||||||
testLink(assert, "https://192.0.0.1");
|
|
||||||
testLink(assert, "https://250.123.67.23:5924");
|
|
||||||
},
|
|
||||||
|
|
||||||
"IPV6 link": assert => {
|
|
||||||
testLink(assert, "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]");
|
|
||||||
testLink(assert, "http://[2001:0db8:85a3:0000:0000:8a2e:0370:7334]:7000");
|
|
||||||
},
|
|
||||||
|
|
||||||
"Missing scheme must not linkify": assert => {
|
|
||||||
testLink(assert, "matrix.org/foo/bar", true);
|
|
||||||
},
|
|
||||||
|
|
||||||
"Punctuation at end of link must not linkify": assert => {
|
|
||||||
const link = "https://foo.bar/?nenjil=lal810";
|
|
||||||
const end = ".,? ";
|
|
||||||
for (const char of end) {
|
|
||||||
const out = [{ type: "link", url: link, text: link }, { type: "text", text: char }];
|
|
||||||
test(assert, link + char, out);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
"Unicode in hostname must not linkify": assert => {
|
|
||||||
const link = "https://foo.bar\uD83D\uDE03.com";
|
|
||||||
const out = [{ type: "link", url: "https://foo.bar", text: "https://foo.bar" },
|
|
||||||
{ type: "text", text: "\uD83D\uDE03.com" }];
|
|
||||||
test(assert, link, out);
|
|
||||||
},
|
|
||||||
|
|
||||||
"Link with unicode only after / must linkify": assert => {
|
|
||||||
testLink(assert, "https://foo.bar.com/\uD83D\uDE03");
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
12
src/domain/session/room/timeline/linkify.js
Normal file
12
src/domain/session/room/timeline/linkify.js
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
export function linkify(text, callback) {
|
||||||
|
const regex = /(?:https|http|ftp):\/\/[a-zA-Z0-9:.\[\]#-]+(?:\/[^\s]*[^\s.,?!]|[^\s\u{80}-\u{10ffff}.,?!])/gui
|
||||||
|
const matches = text.matchAll(regex);
|
||||||
|
let curr = 0;
|
||||||
|
for (let match of matches) {
|
||||||
|
callback(match[0], true);
|
||||||
|
callback(text.slice(curr, match.index), false);
|
||||||
|
const len = match[0].length;
|
||||||
|
curr = match.index + len;
|
||||||
|
}
|
||||||
|
callback(text.slice(curr), false);
|
||||||
|
}
|
Reference in a new issue