summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authordakkar <dakkar@thenautilus.net>2024-05-31 11:43:09 +0100
committerdakkar <dakkar@thenautilus.net>2024-05-31 11:43:09 +0100
commit7b630b48b5e37b4c2dd04cd88365971a84d08a3d (patch)
treebeea4e4df3f61ce253b2c4ba72bf84ea7f242121
parentminimal fixes, thanks tests (diff)
parentmerge: lint our MRs (!506) (diff)
downloadsharkey-7b630b48b5e37b4c2dd04cd88365971a84d08a3d.tar.gz
sharkey-7b630b48b5e37b4c2dd04cd88365971a84d08a3d.tar.bz2
sharkey-7b630b48b5e37b4c2dd04cd88365971a84d08a3d.zip
Merge branch 'develop' into future
-rw-r--r--.gitlab-ci.yml2
-rw-r--r--README.md2
-rw-r--r--packages/backend/src/config.ts18
-rw-r--r--packages/backend/src/core/activitypub/misc/check-against-url.ts22
-rw-r--r--packages/backend/test/unit/misc/check-against-url.ts51
-rw-r--r--packages/frontend/src/pages/settings/general.vue10
6 files changed, 88 insertions, 17 deletions
diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index 1da34db4e6..f725dbf163 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -19,6 +19,8 @@ testCommit:
- pnpm install --frozen-lockfile
- pnpm run build
- pnpm run migrate
+ - pnpm run --filter=backend lint
+ - pnpm run --filter=frontend eslint
cache:
key: test
policy: pull-push
diff --git a/README.md b/README.md
index c002acbff9..6840503243 100644
--- a/README.md
+++ b/README.md
@@ -34,7 +34,7 @@
- **ActivityPub support**\
Not on Sharkey? No problem! Not only can Sharkey instances talk to each other, but you can make friends with people on other networks like Mastodon and Pixelfed!
- **Federated Backgrounds and Music status**\
-You can add a background to your profile as well as a music status via ListenBrainz, show everyone what music you are currently listening too
+You can add a background to your profile as well as a music status via ListenBrainz, show everyone what music you are currently listening to
- **Mastodon API**\
Sharkey implements the Mastodon API unlike normal Misskey
- **UI/UX Improvements**\
diff --git a/packages/backend/src/config.ts b/packages/backend/src/config.ts
index f6ce9b3cdf..7eb5b86b18 100644
--- a/packages/backend/src/config.ts
+++ b/packages/backend/src/config.ts
@@ -327,11 +327,11 @@ function applyEnvOverrides(config: Source) {
// these inner functions recurse through the config structure, using
// the given steps, building the env variable name
- function _apply_top(steps: (string | number)[]) {
+ function _apply_top(steps: (string | string[] | number | number[])[]) {
_walk('', [], steps);
}
- function _walk(name: string, path: (string | number)[], steps: (string | number)[]) {
+ function _walk(name: string, path: (string | number)[], steps: (string | string[] | number | number[])[]) {
// are there more steps after this one? recurse
if (steps.length > 1) {
const thisStep = steps.shift();
@@ -368,7 +368,7 @@ function applyEnvOverrides(config: Source) {
}
// this recurses down, bailing out if there's no config to override
- function _descend(name: string, path: (string | number)[], thisStep: string | number, steps: (string | number)[]) {
+ function _descend(name: string, path: (string | number)[], thisStep: string | number, steps: (string | string[] | number | number[])[]) {
name = `${name}${_step2name(thisStep)}_`;
path = [ ...path, thisStep ];
_walk(name, path, steps);
@@ -390,10 +390,10 @@ function applyEnvOverrides(config: Source) {
}
}
- const alwaysStrings = { 'chmodSocket': 1 };
+ const alwaysStrings = { 'chmodSocket': true } as { [key: string]: boolean };
function _assign(path: (string | number)[], lastStep: string | number, value: string) {
- let thisConfig = config;
+ let thisConfig = config as any;
for (const step of path) {
if (!thisConfig[step]) {
thisConfig[step] = {};
@@ -403,9 +403,11 @@ function applyEnvOverrides(config: Source) {
if (!alwaysStrings[lastStep]) {
if (value.match(/^[0-9]+$/)) {
- value = parseInt(value);
+ thisConfig[lastStep] = parseInt(value);
+ return;
} else if (value.match(/^(true|false)$/i)) {
- value = !!value.match(/^true$/i);
+ thisConfig[lastStep] = !!value.match(/^true$/i);
+ return;
}
}
@@ -416,7 +418,7 @@ function applyEnvOverrides(config: Source) {
_apply_top([['url', 'port', 'socket', 'chmodSocket', 'disableHsts']]);
_apply_top(['db', ['host', 'port', 'db', 'user', 'pass']]);
- _apply_top(['dbSlaves', config.dbSlaves?.keys(), ['host', 'port', 'db', 'user', 'pass']]);
+ _apply_top(['dbSlaves', Array.from((config.dbSlaves ?? []).keys()), ['host', 'port', 'db', 'user', 'pass']]);
_apply_top([
['redis', 'redisForPubsub', 'redisForJobQueue', 'redisForTimelines'],
['host','port','username','pass','db','prefix'],
diff --git a/packages/backend/src/core/activitypub/misc/check-against-url.ts b/packages/backend/src/core/activitypub/misc/check-against-url.ts
index 78ba891a2e..34e4907267 100644
--- a/packages/backend/src/core/activitypub/misc/check-against-url.ts
+++ b/packages/backend/src/core/activitypub/misc/check-against-url.ts
@@ -4,16 +4,24 @@
*/
import type { IObject } from '../type.js';
+function getHrefFrom(one: IObject|string): string | undefined {
+ if (typeof(one) === 'string') return one;
+ return one.href;
+}
+
export function assertActivityMatchesUrls(activity: IObject, urls: string[]) {
const idOk = activity.id !== undefined && urls.includes(activity.id);
+ if (idOk) return;
- // technically `activity.url` could be an `ApObject = IObject |
- // string | (IObject | string)[]`, but if it's a complicated thing
- // and the `activity.id` doesn't match, I think we're fine
- // rejecting the activity
- const urlOk = typeof(activity.url) === 'string' && urls.includes(activity.url);
+ const url = activity.url;
+ if (url) {
+ // `activity.url` can be an `ApObject = IObject | string | (IObject
+ // | string)[]`, we have to look inside it
+ const activityUrls = Array.isArray(url) ? url.map(getHrefFrom) : [getHrefFrom(url)];
+ const goodUrl = activityUrls.find(u => u && urls.includes(u));
- if (!idOk && !urlOk) {
- throw new Error(`bad Activity: neither id(${activity?.id}) nor url(${activity?.url}) match location(${urls})`);
+ if (goodUrl) return;
}
+
+ throw new Error(`bad Activity: neither id(${activity?.id}) nor url(${JSON.stringify(activity?.url)}) match location(${urls})`);
}
diff --git a/packages/backend/test/unit/misc/check-against-url.ts b/packages/backend/test/unit/misc/check-against-url.ts
new file mode 100644
index 0000000000..1cc12cbea2
--- /dev/null
+++ b/packages/backend/test/unit/misc/check-against-url.ts
@@ -0,0 +1,51 @@
+/*
+ * SPDX-FileCopyrightText: dakkar and sharkey-project
+ * SPDX-License-Identifier: AGPL-3.0-only
+ */
+
+import type { IObject } from '@/core/activitypub/type.js';
+import { describe, expect, test } from '@jest/globals';
+import { assertActivityMatchesUrls } from '@/core/activitypub/misc/check-against-url.js';
+
+function assertOne(activity: IObject) {
+ // return a function so we can use `.toThrow`
+ return () => assertActivityMatchesUrls(activity, ['good']);
+}
+
+describe('assertActivityMatchesUrls', () => {
+ test('id', () => {
+ expect(assertOne({ id: 'bad' })).toThrow(/bad Activity/);
+ expect(assertOne({ id: 'good' })).not.toThrow();
+ });
+
+ test('simple url', () => {
+ expect(assertOne({ url: 'bad' })).toThrow(/bad Activity/);
+ expect(assertOne({ url: 'good' })).not.toThrow();
+ });
+
+ test('array of urls', () => {
+ expect(assertOne({ url: ['bad'] })).toThrow(/bad Activity/);
+ expect(assertOne({ url: ['bad', 'other'] })).toThrow(/bad Activity/);
+ expect(assertOne({ url: ['good'] })).not.toThrow();
+ expect(assertOne({ url: ['bad', 'good'] })).not.toThrow();
+ });
+
+ test('array of objects', () => {
+ expect(assertOne({ url: [{ href: 'bad' }] })).toThrow(/bad Activity/);
+ expect(assertOne({ url: [{ href: 'bad' }, { href: 'other' }] })).toThrow(/bad Activity/);
+ expect(assertOne({ url: [{ href: 'good' }] })).not.toThrow();
+ expect(assertOne({ url: [{ href: 'bad' }, { href: 'good' }] })).not.toThrow();
+ });
+
+ test('mixed array', () => {
+ expect(assertOne({ url: [{ href: 'bad' }, 'other'] })).toThrow(/bad Activity/);
+ expect(assertOne({ url: [{ href: 'bad' }, 'good'] })).not.toThrow();
+ expect(assertOne({ url: ['bad', { href: 'good' }] })).not.toThrow();
+ });
+
+ test('id and url', () => {
+ expect(assertOne({ id: 'other', url: 'bad' })).toThrow(/bad Activity/);
+ expect(assertOne({ id: 'bad', url: 'good' })).not.toThrow();
+ expect(assertOne({ id: 'good', url: 'bad' })).not.toThrow();
+ });
+});
diff --git a/packages/frontend/src/pages/settings/general.vue b/packages/frontend/src/pages/settings/general.vue
index f0df5f307f..e772b8c167 100644
--- a/packages/frontend/src/pages/settings/general.vue
+++ b/packages/frontend/src/pages/settings/general.vue
@@ -296,6 +296,7 @@ import { definePageMetadata } from '@/scripts/page-metadata.js';
import { miLocalStorage } from '@/local-storage.js';
import { globalEvents } from '@/events.js';
import { claimAchievement } from '@/scripts/achievements.js';
+import { deepMerge } from '@/scripts/merge.js';
const lang = ref(miLocalStorage.getItem('lang'));
const fontSize = ref(miLocalStorage.getItem('fontSize'));
@@ -322,7 +323,14 @@ const reactionsDisplaySize = computed(defaultStore.makeGetterSetter('reactionsDi
const limitWidthOfReaction = computed(defaultStore.makeGetterSetter('limitWidthOfReaction'));
const collapseRenotes = computed(defaultStore.makeGetterSetter('collapseRenotes'));
const clickToOpen = computed(defaultStore.makeGetterSetter('clickToOpen'));
-const showBots = computed(defaultStore.makeGetterSetter('tlWithBots'));
+// copied from src/pages/timeline.vue
+const showBots = computed<boolean>({
+ get: () => defaultStore.reactiveState.tl.value.filter.withBots,
+ set: (newValue) => {
+ const out = deepMerge({ filter: { withBots: newValue } }, defaultStore.state.tl);
+ defaultStore.set('tl', out);
+ },
+});
const collapseFiles = computed(defaultStore.makeGetterSetter('collapseFiles'));
const autoloadConversation = computed(defaultStore.makeGetterSetter('autoloadConversation'));
const reduceAnimation = computed(defaultStore.makeGetterSetter('animation', v => !v, v => !v));