2022-02-26 04:25:04 +00:00
|
|
|
import {rawEmojis} from "./emojis";
|
2022-02-24 17:26:07 +00:00
|
|
|
|
2022-02-18 19:41:01 +00:00
|
|
|
export const topicUrl = (baseUrl, topic) => `${baseUrl}/${topic}`;
|
2022-02-24 15:30:58 +00:00
|
|
|
export const topicUrlWs = (baseUrl, topic) => `${topicUrl(baseUrl, topic)}/ws`
|
2022-02-18 19:41:01 +00:00
|
|
|
.replaceAll("https://", "wss://")
|
|
|
|
.replaceAll("http://", "ws://");
|
2022-02-23 04:22:30 +00:00
|
|
|
export const topicUrlJson = (baseUrl, topic) => `${topicUrl(baseUrl, topic)}/json`;
|
|
|
|
export const topicUrlJsonPoll = (baseUrl, topic) => `${topicUrlJson(baseUrl, topic)}?poll=1`;
|
2022-02-26 16:45:39 +00:00
|
|
|
export const topicUrlJsonPollWithSince = (baseUrl, topic, since) => `${topicUrlJson(baseUrl, topic)}?poll=1&since=${since}`;
|
2022-02-25 18:40:03 +00:00
|
|
|
export const topicUrlAuth = (baseUrl, topic) => `${topicUrl(baseUrl, topic)}/auth`;
|
2022-02-26 04:25:04 +00:00
|
|
|
export const topicShortUrl = (baseUrl, topic) => shortUrl(topicUrl(baseUrl, topic));
|
2022-02-18 19:41:01 +00:00
|
|
|
export const shortUrl = (url) => url.replaceAll(/https?:\/\//g, "");
|
2022-02-23 04:22:30 +00:00
|
|
|
|
2022-02-26 16:45:39 +00:00
|
|
|
export const validTopic = (topic) => {
|
|
|
|
return topic.match(/^([-_a-zA-Z0-9]{1,64})$/) // Regex must match Go & Android app!
|
|
|
|
}
|
|
|
|
|
2022-02-24 17:26:07 +00:00
|
|
|
// Format emojis (see emoji.js)
|
|
|
|
const emojis = {};
|
|
|
|
rawEmojis.forEach(emoji => {
|
|
|
|
emoji.aliases.forEach(alias => {
|
|
|
|
emojis[alias] = emoji.emoji;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
const toEmojis = (tags) => {
|
|
|
|
if (!tags) return [];
|
|
|
|
else return tags.filter(tag => tag in emojis).map(tag => emojis[tag]);
|
|
|
|
}
|
|
|
|
|
2022-02-26 16:45:39 +00:00
|
|
|
|
|
|
|
export const formatTitleWithFallback = (m, fallback) => {
|
|
|
|
if (m.title) {
|
|
|
|
return formatTitle(m);
|
|
|
|
}
|
|
|
|
return fallback;
|
|
|
|
};
|
|
|
|
|
2022-02-24 17:26:07 +00:00
|
|
|
export const formatTitle = (m) => {
|
|
|
|
const emojiList = toEmojis(m.tags);
|
|
|
|
if (emojiList.length > 0) {
|
|
|
|
return `${emojiList.join(" ")} ${m.title}`;
|
|
|
|
} else {
|
|
|
|
return m.title;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export const formatMessage = (m) => {
|
|
|
|
if (m.title) {
|
|
|
|
return m.message;
|
|
|
|
} else {
|
|
|
|
const emojiList = toEmojis(m.tags);
|
|
|
|
if (emojiList.length > 0) {
|
|
|
|
return `${emojiList.join(" ")} ${m.message}`;
|
|
|
|
} else {
|
|
|
|
return m.message;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export const unmatchedTags = (tags) => {
|
|
|
|
if (!tags) return [];
|
|
|
|
else return tags.filter(tag => !(tag in emojis));
|
|
|
|
}
|
|
|
|
|
2022-02-26 04:25:04 +00:00
|
|
|
|
|
|
|
export const maybeWithBasicAuth = (headers, user) => {
|
|
|
|
if (user) {
|
|
|
|
headers['Authorization'] = `Basic ${encodeBase64(`${user.username}:${user.password}`)}`;
|
|
|
|
}
|
|
|
|
return headers;
|
|
|
|
}
|
|
|
|
|
|
|
|
export const basicAuth = (username, password) => {
|
|
|
|
return `Basic ${encodeBase64(`${username}:${password}`)}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
export const encodeBase64 = (s) => {
|
|
|
|
return new Buffer(s).toString('base64');
|
|
|
|
}
|
|
|
|
|
|
|
|
export const encodeBase64Url = (s) => {
|
|
|
|
return encodeBase64(s)
|
|
|
|
.replaceAll('+', '-')
|
|
|
|
.replaceAll('/', '_')
|
|
|
|
.replaceAll('=', '');
|
|
|
|
}
|
|
|
|
|
2022-02-23 04:22:30 +00:00
|
|
|
// From: https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API/Using_Fetch
|
2022-02-26 04:25:04 +00:00
|
|
|
export async function* fetchLinesIterator(fileURL, headers) {
|
2022-02-23 04:22:30 +00:00
|
|
|
const utf8Decoder = new TextDecoder('utf-8');
|
2022-02-26 04:25:04 +00:00
|
|
|
const response = await fetch(fileURL, {
|
|
|
|
headers: headers
|
|
|
|
});
|
2022-02-23 04:22:30 +00:00
|
|
|
const reader = response.body.getReader();
|
|
|
|
let { value: chunk, done: readerDone } = await reader.read();
|
|
|
|
chunk = chunk ? utf8Decoder.decode(chunk) : '';
|
|
|
|
|
|
|
|
const re = /\n|\r|\r\n/gm;
|
|
|
|
let startIndex = 0;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
let result = re.exec(chunk);
|
|
|
|
if (!result) {
|
|
|
|
if (readerDone) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
let remainder = chunk.substr(startIndex);
|
|
|
|
({ value: chunk, done: readerDone } = await reader.read());
|
|
|
|
chunk = remainder + (chunk ? utf8Decoder.decode(chunk) : '');
|
|
|
|
startIndex = re.lastIndex = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
yield chunk.substring(startIndex, result.index);
|
|
|
|
startIndex = re.lastIndex;
|
|
|
|
}
|
|
|
|
if (startIndex < chunk.length) {
|
|
|
|
yield chunk.substr(startIndex); // last line didn't end in a newline char
|
|
|
|
}
|
|
|
|
}
|