Merge branch 'develop' into feat/scylladb
This commit is contained in:
commit
7e0fa532f7
120 changed files with 9659 additions and 134 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -57,6 +57,9 @@ packages/backend/assets/LICENSE
|
||||||
!/packages/backend/src/db
|
!/packages/backend/src/db
|
||||||
!/packages/backend/src/server/api/endpoints/drive/files
|
!/packages/backend/src/server/api/endpoints/drive/files
|
||||||
|
|
||||||
|
packages/megalodon/lib
|
||||||
|
packages/megalodon/.idea
|
||||||
|
|
||||||
# blender backups
|
# blender backups
|
||||||
*.blend1
|
*.blend1
|
||||||
*.blend2
|
*.blend2
|
||||||
|
|
|
@ -28,6 +28,7 @@ COPY packages/backend/package.json packages/backend/package.json
|
||||||
COPY packages/client/package.json packages/client/package.json
|
COPY packages/client/package.json packages/client/package.json
|
||||||
COPY packages/sw/package.json packages/sw/package.json
|
COPY packages/sw/package.json packages/sw/package.json
|
||||||
COPY packages/firefish-js/package.json packages/firefish-js/package.json
|
COPY packages/firefish-js/package.json packages/firefish-js/package.json
|
||||||
|
COPY packages/megalodon/package.json packages/megalodon/package.json
|
||||||
COPY packages/backend/native-utils/package.json packages/backend/native-utils/package.json
|
COPY packages/backend/native-utils/package.json packages/backend/native-utils/package.json
|
||||||
COPY packages/backend/native-utils/npm/linux-x64-musl/package.json packages/backend/native-utils/npm/linux-x64-musl/package.json
|
COPY packages/backend/native-utils/npm/linux-x64-musl/package.json packages/backend/native-utils/npm/linux-x64-musl/package.json
|
||||||
COPY packages/backend/native-utils/npm/linux-arm64-musl/package.json packages/backend/native-utils/npm/linux-arm64-musl/package.json
|
COPY packages/backend/native-utils/npm/linux-arm64-musl/package.json packages/backend/native-utils/npm/linux-arm64-musl/package.json
|
||||||
|
@ -57,6 +58,8 @@ RUN apt-get update && apt-get install -y libvips-dev zip unzip tini ffmpeg
|
||||||
|
|
||||||
COPY . ./
|
COPY . ./
|
||||||
|
|
||||||
|
COPY --from=build /firefish/packages/megalodon /firefish/packages/megalodon
|
||||||
|
|
||||||
# Copy node modules
|
# Copy node modules
|
||||||
COPY --from=build /firefish/node_modules /firefish/node_modules
|
COPY --from=build /firefish/node_modules /firefish/node_modules
|
||||||
COPY --from=build /firefish/packages/backend/node_modules /firefish/packages/backend/node_modules
|
COPY --from=build /firefish/packages/backend/node_modules /firefish/packages/backend/node_modules
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://git.joinfirefish.org/firefish/firefish.git"
|
"url": "https://git.joinfirefish.org/firefish/firefish.git"
|
||||||
},
|
},
|
||||||
"packageManager": "pnpm@8.8.0",
|
"packageManager": "pnpm@8.9.2",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"rebuild": "pnpm run clean && pnpm run build",
|
"rebuild": "pnpm run clean && pnpm run build",
|
||||||
|
@ -66,7 +66,7 @@
|
||||||
"gulp-replace": "1.1.4",
|
"gulp-replace": "1.1.4",
|
||||||
"gulp-terser": "2.1.0",
|
"gulp-terser": "2.1.0",
|
||||||
"install-peers": "^1.0.4",
|
"install-peers": "^1.0.4",
|
||||||
"pnpm": "8.8.0",
|
"pnpm": "8.9.2",
|
||||||
"start-server-and-test": "1.15.2",
|
"start-server-and-test": "1.15.2",
|
||||||
"typescript": "5.2.2"
|
"typescript": "5.2.2"
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,3 +7,4 @@ This directory contains all of the packages Firefish uses.
|
||||||
- `client`: Web interface written in Vue3 and TypeScript
|
- `client`: Web interface written in Vue3 and TypeScript
|
||||||
- `sw`: Web [Service Worker](https://developer.mozilla.org/en-US/docs/Web/API/Service_Worker_API) written in TypeScript
|
- `sw`: Web [Service Worker](https://developer.mozilla.org/en-US/docs/Web/API/Service_Worker_API) written in TypeScript
|
||||||
- `firefish-js`: TypeScript SDK for both backend and client, also published on [NPM](https://www.npmjs.com/package/firefish-js) for public use
|
- `firefish-js`: TypeScript SDK for both backend and client, also published on [NPM](https://www.npmjs.com/package/firefish-js) for public use
|
||||||
|
- `megalodon`: TypeScript library used for partial Mastodon API compatibility
|
||||||
|
|
|
@ -89,7 +89,7 @@
|
||||||
"koa-send": "5.0.1",
|
"koa-send": "5.0.1",
|
||||||
"koa-slow": "2.1.0",
|
"koa-slow": "2.1.0",
|
||||||
"koa-views": "7.0.2",
|
"koa-views": "7.0.2",
|
||||||
"megalodon": "8.1.1",
|
"megalodon": "workspace:*",
|
||||||
"meilisearch": "0.34.1",
|
"meilisearch": "0.34.1",
|
||||||
"mfm-js": "0.23.3",
|
"mfm-js": "0.23.3",
|
||||||
"mime-types": "2.1.35",
|
"mime-types": "2.1.35",
|
||||||
|
|
|
@ -24,11 +24,7 @@ export function getClient(
|
||||||
const accessTokenArr = authorization?.split(" ") ?? [null];
|
const accessTokenArr = authorization?.split(" ") ?? [null];
|
||||||
const accessToken = accessTokenArr[accessTokenArr.length - 1];
|
const accessToken = accessTokenArr[accessTokenArr.length - 1];
|
||||||
const generator = (megalodon as any).default;
|
const generator = (megalodon as any).default;
|
||||||
const client = generator(
|
const client = generator(BASE_URL, accessToken) as MegalodonInterface;
|
||||||
"firefish",
|
|
||||||
BASE_URL,
|
|
||||||
accessToken,
|
|
||||||
) as MegalodonInterface;
|
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -68,7 +68,7 @@ export function apiAuthMastodon(router: Router): void {
|
||||||
website: body.website,
|
website: body.website,
|
||||||
redirect_uri: red,
|
redirect_uri: red,
|
||||||
client_id: Buffer.from(appData.url || "").toString("base64"),
|
client_id: Buffer.from(appData.url || "").toString("base64"),
|
||||||
client_secret: appData.client_secret,
|
client_secret: appData.clientSecret,
|
||||||
};
|
};
|
||||||
console.log(returns);
|
console.log(returns);
|
||||||
ctx.body = returns;
|
ctx.body = returns;
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
|
import megalodon, { MegalodonInterface } from "megalodon";
|
||||||
import Router from "@koa/router";
|
import Router from "@koa/router";
|
||||||
import { getClient } from "../ApiMastodonCompatibleService.js";
|
import { getClient } from "../ApiMastodonCompatibleService.js";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import Converter from "megalodon";
|
import { Converter } from "megalodon";
|
||||||
import { convertTimelinesArgsId, limitToInt } from "./timeline.js";
|
import { convertTimelinesArgsId, limitToInt } from "./timeline.js";
|
||||||
import { convertAccount, convertStatus } from "../converters.js";
|
import { convertAccount, convertStatus } from "../converters.js";
|
||||||
|
|
||||||
|
|
|
@ -380,7 +380,7 @@ export function apiStatusMastodon(router: Router): void {
|
||||||
const accessTokens = ctx.headers.authorization;
|
const accessTokens = ctx.headers.authorization;
|
||||||
const client = getClient(BASE_URL, accessTokens);
|
const client = getClient(BASE_URL, accessTokens);
|
||||||
try {
|
try {
|
||||||
const data = await client.createEmojiReaction(
|
const data = await client.reactStatus(
|
||||||
convertId(ctx.params.id, IdType.FirefishId),
|
convertId(ctx.params.id, IdType.FirefishId),
|
||||||
ctx.params.name,
|
ctx.params.name,
|
||||||
);
|
);
|
||||||
|
@ -400,7 +400,7 @@ export function apiStatusMastodon(router: Router): void {
|
||||||
const accessTokens = ctx.headers.authorization;
|
const accessTokens = ctx.headers.authorization;
|
||||||
const client = getClient(BASE_URL, accessTokens);
|
const client = getClient(BASE_URL, accessTokens);
|
||||||
try {
|
try {
|
||||||
const data = await client.deleteEmojiReaction(
|
const data = await client.unreactStatus(
|
||||||
convertId(ctx.params.id, IdType.FirefishId),
|
convertId(ctx.params.id, IdType.FirefishId),
|
||||||
ctx.params.name,
|
ctx.params.name,
|
||||||
);
|
);
|
||||||
|
|
|
@ -25,7 +25,7 @@ import { readNotification } from "../common/read-notification.js";
|
||||||
import channels from "./channels/index.js";
|
import channels from "./channels/index.js";
|
||||||
import type Channel from "./channel.js";
|
import type Channel from "./channel.js";
|
||||||
import type { StreamEventEmitter, StreamMessages } from "./types.js";
|
import type { StreamEventEmitter, StreamMessages } from "./types.js";
|
||||||
import Converter from "megalodon";
|
import { Converter } from "megalodon";
|
||||||
import { getClient } from "../mastodon/ApiMastodonCompatibleService.js";
|
import { getClient } from "../mastodon/ApiMastodonCompatibleService.js";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -179,10 +179,10 @@ mastoRouter.post("/oauth/token", async (ctx) => {
|
||||||
ctx.body = ret;
|
ctx.body = ret;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let client_id: Array<string> | string | null = body.client_id;
|
let client_id: any = body.client_id;
|
||||||
const BASE_URL = `${ctx.request.protocol}://${ctx.request.hostname}`;
|
const BASE_URL = `${ctx.request.protocol}://${ctx.request.hostname}`;
|
||||||
const generator = (megalodon as any).default;
|
const generator = (megalodon as any).default;
|
||||||
const client = generator("firefish", BASE_URL, null) as MegalodonInterface;
|
const client = generator(BASE_URL, null) as MegalodonInterface;
|
||||||
let token = null;
|
let token = null;
|
||||||
if (body.code) {
|
if (body.code) {
|
||||||
//m = body.code.match(/^([a-zA-Z0-9]{8})([a-zA-Z0-9]{4})([a-zA-Z0-9]{4})([a-zA-Z0-9]{4})([a-zA-Z0-9]{12})/);
|
//m = body.code.match(/^([a-zA-Z0-9]{8})([a-zA-Z0-9]{4})([a-zA-Z0-9]{4})([a-zA-Z0-9]{4})([a-zA-Z0-9]{12})/);
|
||||||
|
@ -206,7 +206,7 @@ mastoRouter.post("/oauth/token", async (ctx) => {
|
||||||
token ? token : "",
|
token ? token : "",
|
||||||
);
|
);
|
||||||
const ret = {
|
const ret = {
|
||||||
access_token: atData.access_token,
|
access_token: atData.accessToken,
|
||||||
token_type: "Bearer",
|
token_type: "Bearer",
|
||||||
scope: body.scope || "read write follow push",
|
scope: body.scope || "read write follow push",
|
||||||
created_at: Math.floor(new Date().getTime() / 1000),
|
created_at: Math.floor(new Date().getTime() / 1000),
|
||||||
|
|
|
@ -11,32 +11,27 @@
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
v-else-if="!input && !select"
|
v-else-if="!input && !select"
|
||||||
:class="[$style.icon, $style['type_' + type]]"
|
:class="[$style.icon, $style[`type_${type}`]]"
|
||||||
>
|
>
|
||||||
<i
|
<i
|
||||||
v-if="type === 'success'"
|
v-if="type === 'success'"
|
||||||
:class="$style.iconInner"
|
:class="[$style.iconInner, iconClass('ph-check')]"
|
||||||
class="ph-check ph-lg"
|
|
||||||
></i>
|
></i>
|
||||||
<i
|
<i
|
||||||
v-else-if="type === 'error'"
|
v-else-if="type === 'error'"
|
||||||
:class="$style.iconInner"
|
:class="[$style.iconInner, iconClass('ph-circle-wavy-warning')]"
|
||||||
class="ph-circle-wavy-warning ph-lg"
|
|
||||||
></i>
|
></i>
|
||||||
<i
|
<i
|
||||||
v-else-if="type === 'warning'"
|
v-else-if="type === 'warning'"
|
||||||
:class="$style.iconInner"
|
:class="[$style.iconInner, iconClass('ph-warning')]"
|
||||||
class="ph-warning ph-lg"
|
|
||||||
></i>
|
></i>
|
||||||
<i
|
<i
|
||||||
v-else-if="type === 'info'"
|
v-else-if="type === 'info'"
|
||||||
:class="$style.iconInner"
|
:class="[$style.iconInner, iconClass('ph-info')]"
|
||||||
class="ph-info ph-lg"
|
|
||||||
></i>
|
></i>
|
||||||
<i
|
<i
|
||||||
v-else-if="type === 'question'"
|
v-else-if="type === 'question'"
|
||||||
:class="$style.iconInner"
|
:class="[$style.iconInner, iconClass('ph-question')]"
|
||||||
class="ph-circle-question ph-lg"
|
|
||||||
></i>
|
></i>
|
||||||
<MkLoading
|
<MkLoading
|
||||||
v-else-if="type === 'waiting'"
|
v-else-if="type === 'waiting'"
|
||||||
|
|
|
@ -10,10 +10,10 @@
|
||||||
:aria-controls="bodyId"
|
:aria-controls="bodyId"
|
||||||
>
|
>
|
||||||
<template v-if="showBody"
|
<template v-if="showBody"
|
||||||
><i :class="icon('ph-caret-up')"></i
|
><i class="ph-caret-up ph-bold ph-lg"></i
|
||||||
></template>
|
></template>
|
||||||
<template v-else
|
<template v-else
|
||||||
><i :class="icon('ph-caret-down')"></i
|
><i class="ph-caret-down ph-bold ph-lg"></i
|
||||||
></template>
|
></template>
|
||||||
</button>
|
</button>
|
||||||
</header>
|
</header>
|
||||||
|
@ -35,7 +35,7 @@
|
||||||
import { defineComponent } from "vue";
|
import { defineComponent } from "vue";
|
||||||
import { getUniqueId } from "@/os";
|
import { getUniqueId } from "@/os";
|
||||||
import { defaultStore } from "@/store";
|
import { defaultStore } from "@/store";
|
||||||
import icon from "@/scripts/icon";
|
// import icon from "@/scripts/icon";
|
||||||
|
|
||||||
const localStoragePrefix = "ui:folder:";
|
const localStoragePrefix = "ui:folder:";
|
||||||
|
|
||||||
|
|
|
@ -359,7 +359,7 @@ const isDeleted = ref(false);
|
||||||
const muted = ref(
|
const muted = ref(
|
||||||
getWordSoftMute(
|
getWordSoftMute(
|
||||||
note.value,
|
note.value,
|
||||||
$i.id,
|
$i?.id,
|
||||||
defaultStore.state.mutedWords,
|
defaultStore.state.mutedWords,
|
||||||
defaultStore.state.mutedLangs,
|
defaultStore.state.mutedLangs,
|
||||||
),
|
),
|
||||||
|
|
|
@ -235,7 +235,7 @@ const isDeleted = ref(false);
|
||||||
const muted = ref(
|
const muted = ref(
|
||||||
getWordSoftMute(
|
getWordSoftMute(
|
||||||
note.value,
|
note.value,
|
||||||
$i.id,
|
$i?.id,
|
||||||
defaultStore.state.mutedWords,
|
defaultStore.state.mutedWords,
|
||||||
defaultStore.state.mutedLangs,
|
defaultStore.state.mutedLangs,
|
||||||
),
|
),
|
||||||
|
|
|
@ -268,7 +268,7 @@ const isDeleted = ref(false);
|
||||||
const muted = ref(
|
const muted = ref(
|
||||||
getWordSoftMute(
|
getWordSoftMute(
|
||||||
note.value,
|
note.value,
|
||||||
$i.id,
|
$i?.id,
|
||||||
defaultStore.state.mutedWords,
|
defaultStore.state.mutedWords,
|
||||||
defaultStore.state.mutedLangs,
|
defaultStore.state.mutedLangs,
|
||||||
),
|
),
|
||||||
|
|
|
@ -73,13 +73,16 @@ useTooltip(buttonRef, async (showing) => {
|
||||||
});
|
});
|
||||||
|
|
||||||
const hasRenotedBefore = ref(false);
|
const hasRenotedBefore = ref(false);
|
||||||
os.api("notes/renotes", {
|
|
||||||
noteId: props.note.id,
|
if ($i != null) {
|
||||||
userId: $i.id,
|
os.api("notes/renotes", {
|
||||||
limit: 1,
|
noteId: props.note.id,
|
||||||
}).then((res) => {
|
userId: $i.id,
|
||||||
hasRenotedBefore.value = res.length > 0;
|
limit: 1,
|
||||||
});
|
}).then((res) => {
|
||||||
|
hasRenotedBefore.value = res.length > 0;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const renote = (viaKeyboard = false, ev?: MouseEvent) => {
|
const renote = (viaKeyboard = false, ev?: MouseEvent) => {
|
||||||
pleaseLogin();
|
pleaseLogin();
|
||||||
|
|
|
@ -14,8 +14,7 @@
|
||||||
>
|
>
|
||||||
<i
|
<i
|
||||||
v-if="success"
|
v-if="success"
|
||||||
:class="[$style.icon, $style.success]"
|
:class="[$style.icon, $style.success, iconClass('ph-check')]"
|
||||||
class="ph-check ph-lg"
|
|
||||||
></i>
|
></i>
|
||||||
<MkLoading
|
<MkLoading
|
||||||
v-else
|
v-else
|
||||||
|
@ -32,6 +31,7 @@
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { shallowRef, watch } from "vue";
|
import { shallowRef, watch } from "vue";
|
||||||
import MkModal from "@/components/MkModal.vue";
|
import MkModal from "@/components/MkModal.vue";
|
||||||
|
import iconClass from "@/scripts/icon"
|
||||||
|
|
||||||
const modal = shallowRef<InstanceType<typeof MkModal>>();
|
const modal = shallowRef<InstanceType<typeof MkModal>>();
|
||||||
|
|
||||||
|
|
|
@ -18,11 +18,14 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
|
import { ref } from "vue";
|
||||||
import icon from "@/scripts/icon";
|
import icon from "@/scripts/icon";
|
||||||
|
|
||||||
defineProps<{
|
const props = defineProps<{
|
||||||
defaultOpen: boolean;
|
defaultOpen: boolean;
|
||||||
}>();
|
}>();
|
||||||
|
|
||||||
|
const opened = ref(props.defaultOpen);
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
|
|
|
@ -12,11 +12,11 @@
|
||||||
<div v-else>
|
<div v-else>
|
||||||
<div class="wszdbhzo">
|
<div class="wszdbhzo">
|
||||||
<div>
|
<div>
|
||||||
<i :class="icon('ph-warning')"></i>
|
<i :class="iconClass('ph-warning')"></i>
|
||||||
{{ i18n.ts.somethingHappened }}
|
{{ i18n.ts.somethingHappened }}
|
||||||
</div>
|
</div>
|
||||||
<MkButton inline class="retry" @click="retry"
|
<MkButton inline class="retry" @click="retry">
|
||||||
><i :class="icon('ph-arrow-clockwise')"></i>
|
<i :class="iconClass('ph-arrow-clockwise')"></i>
|
||||||
{{ i18n.ts.retry }}</MkButton
|
{{ i18n.ts.retry }}</MkButton
|
||||||
>
|
>
|
||||||
</div>
|
</div>
|
||||||
|
@ -30,7 +30,7 @@ import { defineComponent, ref, watch } from "vue";
|
||||||
import MkButton from "@/components/MkButton.vue";
|
import MkButton from "@/components/MkButton.vue";
|
||||||
import { i18n } from "@/i18n";
|
import { i18n } from "@/i18n";
|
||||||
import { defaultStore } from "@/store";
|
import { defaultStore } from "@/store";
|
||||||
import icon from "@/scripts/icon";
|
import iconClass from "@/scripts/icon";
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: {
|
components: {
|
||||||
|
@ -90,6 +90,8 @@ export default defineComponent({
|
||||||
result,
|
result,
|
||||||
retry,
|
retry,
|
||||||
i18n,
|
i18n,
|
||||||
|
defaultStore,
|
||||||
|
iconClass,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -32,9 +32,7 @@
|
||||||
class="save"
|
class="save"
|
||||||
@click="updated"
|
@click="updated"
|
||||||
>
|
>
|
||||||
<!-- FIXME: icon function doesn't work here -->
|
<i :class="icon('ph-floppy-disk-back')"></i>
|
||||||
<!-- <i :class="icon('ph-floppy-disk-back')"></i> -->
|
|
||||||
<i class="ph-floppy-disk-back ph-bold ph-lg"></i>
|
|
||||||
{{ i18n.ts.save }}</MkButton
|
{{ i18n.ts.save }}</MkButton
|
||||||
>
|
>
|
||||||
</div>
|
</div>
|
||||||
|
@ -53,7 +51,7 @@ import {
|
||||||
import { debounce } from "throttle-debounce";
|
import { debounce } from "throttle-debounce";
|
||||||
import MkButton from "@/components/MkButton.vue";
|
import MkButton from "@/components/MkButton.vue";
|
||||||
import { i18n } from "@/i18n";
|
import { i18n } from "@/i18n";
|
||||||
// import icon from "@/scripts/icon";
|
import icon from "@/scripts/icon";
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: {
|
components: {
|
||||||
|
@ -192,6 +190,7 @@ export default defineComponent({
|
||||||
onKeydown,
|
onKeydown,
|
||||||
updated,
|
updated,
|
||||||
i18n,
|
i18n,
|
||||||
|
icon,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -75,6 +75,7 @@ export default defineComponent({
|
||||||
return {
|
return {
|
||||||
showBody: this.expanded,
|
showBody: this.expanded,
|
||||||
i18n,
|
i18n,
|
||||||
|
icon,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
>
|
>
|
||||||
<template #func>
|
<template #func>
|
||||||
<button class="_button" @click="changeType()">
|
<button class="_button" @click="changeType()">
|
||||||
<i :class="icon('ph-pencil')"></i>
|
<i :class="iconClass('ph-pencil')"></i>
|
||||||
</button>
|
</button>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
@ -158,7 +158,7 @@ import * as os from "@/os";
|
||||||
import { isLiteralValue } from "@/scripts/hpml/expr";
|
import { isLiteralValue } from "@/scripts/hpml/expr";
|
||||||
import { funcDefs } from "@/scripts/hpml/lib";
|
import { funcDefs } from "@/scripts/hpml/lib";
|
||||||
import { i18n } from "@/i18n";
|
import { i18n } from "@/i18n";
|
||||||
import icon from "@/scripts/icon";
|
import iconClass from "@/scripts/icon";
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: {
|
components: {
|
||||||
|
@ -207,6 +207,7 @@ export default defineComponent({
|
||||||
warn: null,
|
warn: null,
|
||||||
slots: "",
|
slots: "",
|
||||||
i18n,
|
i18n,
|
||||||
|
iconClass,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -79,7 +79,7 @@ export function getWordSoftMute(
|
||||||
mutedWords: Array<string | string[]>,
|
mutedWords: Array<string | string[]>,
|
||||||
mutedLangs: Array<string | string[]>,
|
mutedLangs: Array<string | string[]>,
|
||||||
): Muted {
|
): Muted {
|
||||||
if (note.userId === meId) return NotMuted;
|
if (meId == null || note.userId === meId) return NotMuted;
|
||||||
|
|
||||||
if (mutedWords.length > 0) {
|
if (mutedWords.length > 0) {
|
||||||
const noteMuted = checkWordMute(note, mutedWords);
|
const noteMuted = checkWordMute(note, mutedWords);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { defaultStore } from "@/store";
|
import { defaultStore } from "@/store";
|
||||||
|
|
||||||
export default function icon(name: string, large = true): string {
|
export default function (name: string, large = true): string {
|
||||||
return `${name} ${large ? "ph-lg" : ""} ${defaultStore.state.iconSet}`;
|
return `${name} ${large ? "ph-lg" : ""} ${defaultStore.state.iconSet}`;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ import * as mfm from "mfm-js";
|
||||||
import { defaultStore } from "@/store";
|
import { defaultStore } from "@/store";
|
||||||
import { expandKaTeXMacro } from "@/scripts/katex-macro";
|
import { expandKaTeXMacro } from "@/scripts/katex-macro";
|
||||||
|
|
||||||
export default function preprocess(text: string): string {
|
export default function (text: string): string {
|
||||||
if (defaultStore.state.enableCustomKaTeXMacro) {
|
if (defaultStore.state.enableCustomKaTeXMacro) {
|
||||||
const parsedKaTeXMacro =
|
const parsedKaTeXMacro =
|
||||||
localStorage.getItem("customKaTeXMacroParsed") ?? "{}";
|
localStorage.getItem("customKaTeXMacroParsed") ?? "{}";
|
||||||
|
|
|
@ -130,6 +130,7 @@ export default defineComponent({
|
||||||
!instance.disableLocalTimeline ||
|
!instance.disableLocalTimeline ||
|
||||||
!instance.disableRecommendedTimeline ||
|
!instance.disableRecommendedTimeline ||
|
||||||
!instance.disableGlobalTimeline,
|
!instance.disableGlobalTimeline,
|
||||||
|
icon,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
83
packages/megalodon/package.json
Normal file
83
packages/megalodon/package.json
Normal file
|
@ -0,0 +1,83 @@
|
||||||
|
{
|
||||||
|
"name": "megalodon",
|
||||||
|
"private": true,
|
||||||
|
"main": "./lib/src/index.js",
|
||||||
|
"typings": "./lib/src/index.d.ts",
|
||||||
|
"scripts": {
|
||||||
|
"build": "tsc -p ./",
|
||||||
|
"build:debug": "pnpm run build",
|
||||||
|
"lint": "pnpm biome check **/*.ts --apply",
|
||||||
|
"format": "pnpm biome format --write src/**/*.ts",
|
||||||
|
"doc": "typedoc --out ../docs ./src",
|
||||||
|
"test": "NODE_ENV=test jest -u --maxWorkers=3"
|
||||||
|
},
|
||||||
|
"jest": {
|
||||||
|
"moduleFileExtensions": [
|
||||||
|
"ts",
|
||||||
|
"js"
|
||||||
|
],
|
||||||
|
"moduleNameMapper": {
|
||||||
|
"^@/(.+)": "<rootDir>/src/$1",
|
||||||
|
"^~/(.+)": "<rootDir>/$1"
|
||||||
|
},
|
||||||
|
"testMatch": [
|
||||||
|
"**/test/**/*.spec.ts"
|
||||||
|
],
|
||||||
|
"preset": "ts-jest/presets/default",
|
||||||
|
"transform": {
|
||||||
|
"^.+\\.(ts|tsx)$": "ts-jest"
|
||||||
|
},
|
||||||
|
"globals": {
|
||||||
|
"ts-jest": {
|
||||||
|
"tsconfig": "tsconfig.json"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"testEnvironment": "node"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@types/oauth": "^0.9.0",
|
||||||
|
"@types/ws": "^8.5.4",
|
||||||
|
"axios": "1.2.2",
|
||||||
|
"dayjs": "^1.11.7",
|
||||||
|
"form-data": "^4.0.0",
|
||||||
|
"https-proxy-agent": "^5.0.1",
|
||||||
|
"oauth": "^0.10.0",
|
||||||
|
"object-assign-deep": "^0.4.0",
|
||||||
|
"parse-link-header": "^2.0.0",
|
||||||
|
"socks-proxy-agent": "^7.0.0",
|
||||||
|
"typescript": "4.9.4",
|
||||||
|
"uuid": "^9.0.0",
|
||||||
|
"ws": "8.12.0",
|
||||||
|
"async-lock": "1.4.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/core-js": "^2.5.0",
|
||||||
|
"@types/form-data": "^2.5.0",
|
||||||
|
"@types/jest": "^29.4.0",
|
||||||
|
"@types/object-assign-deep": "^0.4.0",
|
||||||
|
"@types/parse-link-header": "^2.0.0",
|
||||||
|
"@types/uuid": "^9.0.0",
|
||||||
|
"@types/node": "18.11.18",
|
||||||
|
"@typescript-eslint/eslint-plugin": "^5.49.0",
|
||||||
|
"@typescript-eslint/parser": "^5.49.0",
|
||||||
|
"@types/async-lock": "1.4.0",
|
||||||
|
"eslint": "^8.32.0",
|
||||||
|
"eslint-config-prettier": "^8.6.0",
|
||||||
|
"eslint-config-standard": "^16.0.3",
|
||||||
|
"eslint-plugin-import": "^2.27.5",
|
||||||
|
"eslint-plugin-node": "^11.0.0",
|
||||||
|
"eslint-plugin-prettier": "^4.2.1",
|
||||||
|
"eslint-plugin-promise": "^6.1.1",
|
||||||
|
"eslint-plugin-standard": "^5.0.0",
|
||||||
|
"jest": "^29.4.0",
|
||||||
|
"jest-worker": "^29.4.0",
|
||||||
|
"lodash": "^4.17.14",
|
||||||
|
"prettier": "^2.8.3",
|
||||||
|
"ts-jest": "^29.0.5",
|
||||||
|
"typedoc": "^0.23.24"
|
||||||
|
},
|
||||||
|
"directories": {
|
||||||
|
"lib": "lib",
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
}
|
1
packages/megalodon/src/axios.d.ts
vendored
Normal file
1
packages/megalodon/src/axios.d.ts
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
declare module "axios/lib/adapters/http";
|
13
packages/megalodon/src/cancel.ts
Normal file
13
packages/megalodon/src/cancel.ts
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
export class RequestCanceledError extends Error {
|
||||||
|
public isCancel: boolean;
|
||||||
|
|
||||||
|
constructor(msg: string) {
|
||||||
|
super(msg);
|
||||||
|
this.isCancel = true;
|
||||||
|
Object.setPrototypeOf(this, RequestCanceledError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const isCancel = (value: any): boolean => {
|
||||||
|
return value && value.isCancel;
|
||||||
|
};
|
3
packages/megalodon/src/converter.ts
Normal file
3
packages/megalodon/src/converter.ts
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
import MisskeyAPI from "./misskey/api_client";
|
||||||
|
|
||||||
|
export default MisskeyAPI.Converter;
|
3
packages/megalodon/src/default.ts
Normal file
3
packages/megalodon/src/default.ts
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
export const NO_REDIRECT = "urn:ietf:wg:oauth:2.0:oob";
|
||||||
|
export const DEFAULT_SCOPE = ["read", "write", "follow"];
|
||||||
|
export const DEFAULT_UA = "megalodon";
|
27
packages/megalodon/src/entities/account.ts
Normal file
27
packages/megalodon/src/entities/account.ts
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
/// <reference path="emoji.ts" />
|
||||||
|
/// <reference path="source.ts" />
|
||||||
|
/// <reference path="field.ts" />
|
||||||
|
namespace Entity {
|
||||||
|
export type Account = {
|
||||||
|
id: string;
|
||||||
|
username: string;
|
||||||
|
acct: string;
|
||||||
|
display_name: string;
|
||||||
|
locked: boolean;
|
||||||
|
created_at: string;
|
||||||
|
followers_count: number;
|
||||||
|
following_count: number;
|
||||||
|
statuses_count: number;
|
||||||
|
note: string;
|
||||||
|
url: string;
|
||||||
|
avatar: string;
|
||||||
|
avatar_static: string;
|
||||||
|
header: string;
|
||||||
|
header_static: string;
|
||||||
|
emojis: Array<Emoji>;
|
||||||
|
moved: Account | null;
|
||||||
|
fields: Array<Field>;
|
||||||
|
bot: boolean | null;
|
||||||
|
source?: Source;
|
||||||
|
};
|
||||||
|
}
|
8
packages/megalodon/src/entities/activity.ts
Normal file
8
packages/megalodon/src/entities/activity.ts
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Activity = {
|
||||||
|
week: string;
|
||||||
|
statuses: string;
|
||||||
|
logins: string;
|
||||||
|
registrations: string;
|
||||||
|
};
|
||||||
|
}
|
34
packages/megalodon/src/entities/announcement.ts
Normal file
34
packages/megalodon/src/entities/announcement.ts
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
/// <reference path="tag.ts" />
|
||||||
|
/// <reference path="emoji.ts" />
|
||||||
|
/// <reference path="reaction.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Announcement = {
|
||||||
|
id: string;
|
||||||
|
content: string;
|
||||||
|
starts_at: string | null;
|
||||||
|
ends_at: string | null;
|
||||||
|
published: boolean;
|
||||||
|
all_day: boolean;
|
||||||
|
published_at: string;
|
||||||
|
updated_at: string;
|
||||||
|
read?: boolean;
|
||||||
|
mentions: Array<AnnouncementAccount>;
|
||||||
|
statuses: Array<AnnouncementStatus>;
|
||||||
|
tags: Array<Tag>;
|
||||||
|
emojis: Array<Emoji>;
|
||||||
|
reactions: Array<Reaction>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type AnnouncementAccount = {
|
||||||
|
id: string;
|
||||||
|
username: string;
|
||||||
|
url: string;
|
||||||
|
acct: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type AnnouncementStatus = {
|
||||||
|
id: string;
|
||||||
|
url: string;
|
||||||
|
};
|
||||||
|
}
|
7
packages/megalodon/src/entities/application.ts
Normal file
7
packages/megalodon/src/entities/application.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Application = {
|
||||||
|
name: string;
|
||||||
|
website?: string | null;
|
||||||
|
vapid_key?: string | null;
|
||||||
|
};
|
||||||
|
}
|
14
packages/megalodon/src/entities/async_attachment.ts
Normal file
14
packages/megalodon/src/entities/async_attachment.ts
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
/// <reference path="attachment.ts" />
|
||||||
|
namespace Entity {
|
||||||
|
export type AsyncAttachment = {
|
||||||
|
id: string;
|
||||||
|
type: "unknown" | "image" | "gifv" | "video" | "audio";
|
||||||
|
url: string | null;
|
||||||
|
remote_url: string | null;
|
||||||
|
preview_url: string;
|
||||||
|
text_url: string | null;
|
||||||
|
meta: Meta | null;
|
||||||
|
description: string | null;
|
||||||
|
blurhash: string | null;
|
||||||
|
};
|
||||||
|
}
|
49
packages/megalodon/src/entities/attachment.ts
Normal file
49
packages/megalodon/src/entities/attachment.ts
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Sub = {
|
||||||
|
// For Image, Gifv, and Video
|
||||||
|
width?: number;
|
||||||
|
height?: number;
|
||||||
|
size?: string;
|
||||||
|
aspect?: number;
|
||||||
|
|
||||||
|
// For Gifv and Video
|
||||||
|
frame_rate?: string;
|
||||||
|
|
||||||
|
// For Audio, Gifv, and Video
|
||||||
|
duration?: number;
|
||||||
|
bitrate?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Focus = {
|
||||||
|
x: number;
|
||||||
|
y: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Meta = {
|
||||||
|
original?: Sub;
|
||||||
|
small?: Sub;
|
||||||
|
focus?: Focus;
|
||||||
|
length?: string;
|
||||||
|
duration?: number;
|
||||||
|
fps?: number;
|
||||||
|
size?: string;
|
||||||
|
width?: number;
|
||||||
|
height?: number;
|
||||||
|
aspect?: number;
|
||||||
|
audio_encode?: string;
|
||||||
|
audio_bitrate?: string;
|
||||||
|
audio_channel?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Attachment = {
|
||||||
|
id: string;
|
||||||
|
type: "unknown" | "image" | "gifv" | "video" | "audio";
|
||||||
|
url: string;
|
||||||
|
remote_url: string | null;
|
||||||
|
preview_url: string | null;
|
||||||
|
text_url: string | null;
|
||||||
|
meta: Meta | null;
|
||||||
|
description: string | null;
|
||||||
|
blurhash: string | null;
|
||||||
|
};
|
||||||
|
}
|
16
packages/megalodon/src/entities/card.ts
Normal file
16
packages/megalodon/src/entities/card.ts
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Card = {
|
||||||
|
url: string;
|
||||||
|
title: string;
|
||||||
|
description: string;
|
||||||
|
type: "link" | "photo" | "video" | "rich";
|
||||||
|
image?: string;
|
||||||
|
author_name?: string;
|
||||||
|
author_url?: string;
|
||||||
|
provider_name?: string;
|
||||||
|
provider_url?: string;
|
||||||
|
html?: string;
|
||||||
|
width?: number;
|
||||||
|
height?: number;
|
||||||
|
};
|
||||||
|
}
|
8
packages/megalodon/src/entities/context.ts
Normal file
8
packages/megalodon/src/entities/context.ts
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
/// <reference path="status.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Context = {
|
||||||
|
ancestors: Array<Status>;
|
||||||
|
descendants: Array<Status>;
|
||||||
|
};
|
||||||
|
}
|
11
packages/megalodon/src/entities/conversation.ts
Normal file
11
packages/megalodon/src/entities/conversation.ts
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
/// <reference path="account.ts" />
|
||||||
|
/// <reference path="status.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Conversation = {
|
||||||
|
id: string;
|
||||||
|
accounts: Array<Account>;
|
||||||
|
last_status: Status | null;
|
||||||
|
unread: boolean;
|
||||||
|
};
|
||||||
|
}
|
9
packages/megalodon/src/entities/emoji.ts
Normal file
9
packages/megalodon/src/entities/emoji.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Emoji = {
|
||||||
|
shortcode: string;
|
||||||
|
static_url: string;
|
||||||
|
url: string;
|
||||||
|
visible_in_picker: boolean;
|
||||||
|
category: string;
|
||||||
|
};
|
||||||
|
}
|
8
packages/megalodon/src/entities/featured_tag.ts
Normal file
8
packages/megalodon/src/entities/featured_tag.ts
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type FeaturedTag = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
statuses_count: number;
|
||||||
|
last_status_at: string;
|
||||||
|
};
|
||||||
|
}
|
7
packages/megalodon/src/entities/field.ts
Normal file
7
packages/megalodon/src/entities/field.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Field = {
|
||||||
|
name: string;
|
||||||
|
value: string;
|
||||||
|
verified_at: string | null;
|
||||||
|
};
|
||||||
|
}
|
12
packages/megalodon/src/entities/filter.ts
Normal file
12
packages/megalodon/src/entities/filter.ts
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Filter = {
|
||||||
|
id: string;
|
||||||
|
phrase: string;
|
||||||
|
context: Array<FilterContext>;
|
||||||
|
expires_at: string | null;
|
||||||
|
irreversible: boolean;
|
||||||
|
whole_word: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type FilterContext = string;
|
||||||
|
}
|
7
packages/megalodon/src/entities/history.ts
Normal file
7
packages/megalodon/src/entities/history.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type History = {
|
||||||
|
day: string;
|
||||||
|
uses: number;
|
||||||
|
accounts: number;
|
||||||
|
};
|
||||||
|
}
|
9
packages/megalodon/src/entities/identity_proof.ts
Normal file
9
packages/megalodon/src/entities/identity_proof.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type IdentityProof = {
|
||||||
|
provider: string;
|
||||||
|
provider_username: string;
|
||||||
|
updated_at: string;
|
||||||
|
proof_url: string;
|
||||||
|
profile_url: string;
|
||||||
|
};
|
||||||
|
}
|
41
packages/megalodon/src/entities/instance.ts
Normal file
41
packages/megalodon/src/entities/instance.ts
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
/// <reference path="account.ts" />
|
||||||
|
/// <reference path="urls.ts" />
|
||||||
|
/// <reference path="stats.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Instance = {
|
||||||
|
uri: string;
|
||||||
|
title: string;
|
||||||
|
description: string;
|
||||||
|
email: string;
|
||||||
|
version: string;
|
||||||
|
thumbnail: string | null;
|
||||||
|
urls: URLs;
|
||||||
|
stats: Stats;
|
||||||
|
languages: Array<string>;
|
||||||
|
contact_account: Account | null;
|
||||||
|
max_toot_chars?: number;
|
||||||
|
registrations?: boolean;
|
||||||
|
configuration?: {
|
||||||
|
statuses: {
|
||||||
|
max_characters: number;
|
||||||
|
max_media_attachments: number;
|
||||||
|
characters_reserved_per_url: number;
|
||||||
|
};
|
||||||
|
media_attachments: {
|
||||||
|
supported_mime_types: Array<string>;
|
||||||
|
image_size_limit: number;
|
||||||
|
image_matrix_limit: number;
|
||||||
|
video_size_limit: number;
|
||||||
|
video_frame_limit: number;
|
||||||
|
video_matrix_limit: number;
|
||||||
|
};
|
||||||
|
polls: {
|
||||||
|
max_options: number;
|
||||||
|
max_characters_per_option: number;
|
||||||
|
min_expiration: number;
|
||||||
|
max_expiration: number;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
6
packages/megalodon/src/entities/list.ts
Normal file
6
packages/megalodon/src/entities/list.ts
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type List = {
|
||||||
|
id: string;
|
||||||
|
title: string;
|
||||||
|
};
|
||||||
|
}
|
15
packages/megalodon/src/entities/marker.ts
Normal file
15
packages/megalodon/src/entities/marker.ts
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Marker = {
|
||||||
|
home?: {
|
||||||
|
last_read_id: string;
|
||||||
|
version: number;
|
||||||
|
updated_at: string;
|
||||||
|
};
|
||||||
|
notifications?: {
|
||||||
|
last_read_id: string;
|
||||||
|
version: number;
|
||||||
|
updated_at: string;
|
||||||
|
unread_count?: number;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
8
packages/megalodon/src/entities/mention.ts
Normal file
8
packages/megalodon/src/entities/mention.ts
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Mention = {
|
||||||
|
id: string;
|
||||||
|
username: string;
|
||||||
|
url: string;
|
||||||
|
acct: string;
|
||||||
|
};
|
||||||
|
}
|
15
packages/megalodon/src/entities/notification.ts
Normal file
15
packages/megalodon/src/entities/notification.ts
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
/// <reference path="account.ts" />
|
||||||
|
/// <reference path="status.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Notification = {
|
||||||
|
account: Account;
|
||||||
|
created_at: string;
|
||||||
|
id: string;
|
||||||
|
status?: Status;
|
||||||
|
reaction?: Reaction;
|
||||||
|
type: NotificationType;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type NotificationType = string;
|
||||||
|
}
|
14
packages/megalodon/src/entities/poll.ts
Normal file
14
packages/megalodon/src/entities/poll.ts
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
/// <reference path="poll_option.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Poll = {
|
||||||
|
id: string;
|
||||||
|
expires_at: string | null;
|
||||||
|
expired: boolean;
|
||||||
|
multiple: boolean;
|
||||||
|
votes_count: number;
|
||||||
|
options: Array<PollOption>;
|
||||||
|
voted: boolean;
|
||||||
|
own_votes: Array<number>;
|
||||||
|
};
|
||||||
|
}
|
6
packages/megalodon/src/entities/poll_option.ts
Normal file
6
packages/megalodon/src/entities/poll_option.ts
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type PollOption = {
|
||||||
|
title: string;
|
||||||
|
votes_count: number | null;
|
||||||
|
};
|
||||||
|
}
|
9
packages/megalodon/src/entities/preferences.ts
Normal file
9
packages/megalodon/src/entities/preferences.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Preferences = {
|
||||||
|
"posting:default:visibility": "public" | "unlisted" | "private" | "direct";
|
||||||
|
"posting:default:sensitive": boolean;
|
||||||
|
"posting:default:language": string | null;
|
||||||
|
"reading:expand:media": "default" | "show_all" | "hide_all";
|
||||||
|
"reading:expand:spoilers": boolean;
|
||||||
|
};
|
||||||
|
}
|
16
packages/megalodon/src/entities/push_subscription.ts
Normal file
16
packages/megalodon/src/entities/push_subscription.ts
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Alerts = {
|
||||||
|
follow: boolean;
|
||||||
|
favourite: boolean;
|
||||||
|
mention: boolean;
|
||||||
|
reblog: boolean;
|
||||||
|
poll: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type PushSubscription = {
|
||||||
|
id: string;
|
||||||
|
endpoint: string;
|
||||||
|
server_key: string;
|
||||||
|
alerts: Alerts;
|
||||||
|
};
|
||||||
|
}
|
12
packages/megalodon/src/entities/reaction.ts
Normal file
12
packages/megalodon/src/entities/reaction.ts
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
/// <reference path="account.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Reaction = {
|
||||||
|
count: number;
|
||||||
|
me: boolean;
|
||||||
|
name: string;
|
||||||
|
url?: string;
|
||||||
|
static_url?: string;
|
||||||
|
accounts?: Array<Account>;
|
||||||
|
};
|
||||||
|
}
|
17
packages/megalodon/src/entities/relationship.ts
Normal file
17
packages/megalodon/src/entities/relationship.ts
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Relationship = {
|
||||||
|
id: string;
|
||||||
|
following: boolean;
|
||||||
|
followed_by: boolean;
|
||||||
|
delivery_following?: boolean;
|
||||||
|
blocking: boolean;
|
||||||
|
blocked_by: boolean;
|
||||||
|
muting: boolean;
|
||||||
|
muting_notifications: boolean;
|
||||||
|
requested: boolean;
|
||||||
|
domain_blocking: boolean;
|
||||||
|
showing_reblogs: boolean;
|
||||||
|
endorsed: boolean;
|
||||||
|
notifying: boolean;
|
||||||
|
};
|
||||||
|
}
|
9
packages/megalodon/src/entities/report.ts
Normal file
9
packages/megalodon/src/entities/report.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Report = {
|
||||||
|
id: string;
|
||||||
|
action_taken: string;
|
||||||
|
comment: string;
|
||||||
|
account_id: string;
|
||||||
|
status_ids: Array<string>;
|
||||||
|
};
|
||||||
|
}
|
11
packages/megalodon/src/entities/results.ts
Normal file
11
packages/megalodon/src/entities/results.ts
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
/// <reference path="account.ts" />
|
||||||
|
/// <reference path="status.ts" />
|
||||||
|
/// <reference path="tag.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Results = {
|
||||||
|
accounts: Array<Account>;
|
||||||
|
statuses: Array<Status>;
|
||||||
|
hashtags: Array<Tag>;
|
||||||
|
};
|
||||||
|
}
|
10
packages/megalodon/src/entities/scheduled_status.ts
Normal file
10
packages/megalodon/src/entities/scheduled_status.ts
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
/// <reference path="attachment.ts" />
|
||||||
|
/// <reference path="status_params.ts" />
|
||||||
|
namespace Entity {
|
||||||
|
export type ScheduledStatus = {
|
||||||
|
id: string;
|
||||||
|
scheduled_at: string;
|
||||||
|
params: StatusParams;
|
||||||
|
media_attachments: Array<Attachment>;
|
||||||
|
};
|
||||||
|
}
|
10
packages/megalodon/src/entities/source.ts
Normal file
10
packages/megalodon/src/entities/source.ts
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
/// <reference path="field.ts" />
|
||||||
|
namespace Entity {
|
||||||
|
export type Source = {
|
||||||
|
privacy: string | null;
|
||||||
|
sensitive: boolean | null;
|
||||||
|
language: string | null;
|
||||||
|
note: string;
|
||||||
|
fields: Array<Field>;
|
||||||
|
};
|
||||||
|
}
|
7
packages/megalodon/src/entities/stats.ts
Normal file
7
packages/megalodon/src/entities/stats.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Stats = {
|
||||||
|
user_count: number;
|
||||||
|
status_count: number;
|
||||||
|
domain_count: number;
|
||||||
|
};
|
||||||
|
}
|
45
packages/megalodon/src/entities/status.ts
Normal file
45
packages/megalodon/src/entities/status.ts
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/// <reference path="account.ts" />
|
||||||
|
/// <reference path="application.ts" />
|
||||||
|
/// <reference path="mention.ts" />
|
||||||
|
/// <reference path="tag.ts" />
|
||||||
|
/// <reference path="attachment.ts" />
|
||||||
|
/// <reference path="emoji.ts" />
|
||||||
|
/// <reference path="card.ts" />
|
||||||
|
/// <reference path="poll.ts" />
|
||||||
|
/// <reference path="reaction.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Status = {
|
||||||
|
id: string;
|
||||||
|
uri: string;
|
||||||
|
url: string;
|
||||||
|
account: Account;
|
||||||
|
in_reply_to_id: string | null;
|
||||||
|
in_reply_to_account_id: string | null;
|
||||||
|
reblog: Status | null;
|
||||||
|
content: string;
|
||||||
|
plain_content: string | null;
|
||||||
|
created_at: string;
|
||||||
|
emojis: Emoji[];
|
||||||
|
replies_count: number;
|
||||||
|
reblogs_count: number;
|
||||||
|
favourites_count: number;
|
||||||
|
reblogged: boolean | null;
|
||||||
|
favourited: boolean | null;
|
||||||
|
muted: boolean | null;
|
||||||
|
sensitive: boolean;
|
||||||
|
spoiler_text: string;
|
||||||
|
visibility: "public" | "unlisted" | "private" | "direct";
|
||||||
|
media_attachments: Array<Attachment>;
|
||||||
|
mentions: Array<Mention>;
|
||||||
|
tags: Array<Tag>;
|
||||||
|
card: Card | null;
|
||||||
|
poll: Poll | null;
|
||||||
|
application: Application | null;
|
||||||
|
language: string | null;
|
||||||
|
pinned: boolean | null;
|
||||||
|
reactions: Array<Reaction>;
|
||||||
|
quote: Status | null;
|
||||||
|
bookmarked: boolean;
|
||||||
|
};
|
||||||
|
}
|
23
packages/megalodon/src/entities/status_edit.ts
Normal file
23
packages/megalodon/src/entities/status_edit.ts
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
/// <reference path="account.ts" />
|
||||||
|
/// <reference path="application.ts" />
|
||||||
|
/// <reference path="mention.ts" />
|
||||||
|
/// <reference path="tag.ts" />
|
||||||
|
/// <reference path="attachment.ts" />
|
||||||
|
/// <reference path="emoji.ts" />
|
||||||
|
/// <reference path="card.ts" />
|
||||||
|
/// <reference path="poll.ts" />
|
||||||
|
/// <reference path="reaction.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type StatusEdit = {
|
||||||
|
account: Account;
|
||||||
|
content: string;
|
||||||
|
plain_content: string | null;
|
||||||
|
created_at: string;
|
||||||
|
emojis: Emoji[];
|
||||||
|
sensitive: boolean;
|
||||||
|
spoiler_text: string;
|
||||||
|
media_attachments: Array<Attachment>;
|
||||||
|
poll: Poll | null;
|
||||||
|
};
|
||||||
|
}
|
12
packages/megalodon/src/entities/status_params.ts
Normal file
12
packages/megalodon/src/entities/status_params.ts
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type StatusParams = {
|
||||||
|
text: string;
|
||||||
|
in_reply_to_id: string | null;
|
||||||
|
media_ids: Array<string> | null;
|
||||||
|
sensitive: boolean | null;
|
||||||
|
spoiler_text: string | null;
|
||||||
|
visibility: "public" | "unlisted" | "private" | "direct";
|
||||||
|
scheduled_at: string | null;
|
||||||
|
application_id: string;
|
||||||
|
};
|
||||||
|
}
|
10
packages/megalodon/src/entities/tag.ts
Normal file
10
packages/megalodon/src/entities/tag.ts
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
/// <reference path="history.ts" />
|
||||||
|
|
||||||
|
namespace Entity {
|
||||||
|
export type Tag = {
|
||||||
|
name: string;
|
||||||
|
url: string;
|
||||||
|
history: Array<History> | null;
|
||||||
|
following?: boolean;
|
||||||
|
};
|
||||||
|
}
|
8
packages/megalodon/src/entities/token.ts
Normal file
8
packages/megalodon/src/entities/token.ts
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type Token = {
|
||||||
|
access_token: string;
|
||||||
|
token_type: string;
|
||||||
|
scope: string;
|
||||||
|
created_at: number;
|
||||||
|
};
|
||||||
|
}
|
5
packages/megalodon/src/entities/urls.ts
Normal file
5
packages/megalodon/src/entities/urls.ts
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
namespace Entity {
|
||||||
|
export type URLs = {
|
||||||
|
streaming_api: string;
|
||||||
|
};
|
||||||
|
}
|
38
packages/megalodon/src/entity.ts
Normal file
38
packages/megalodon/src/entity.ts
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
/// <reference path="./entities/account.ts" />
|
||||||
|
/// <reference path="./entities/activity.ts" />
|
||||||
|
/// <reference path="./entities/announcement.ts" />
|
||||||
|
/// <reference path="./entities/application.ts" />
|
||||||
|
/// <reference path="./entities/async_attachment.ts" />
|
||||||
|
/// <reference path="./entities/attachment.ts" />
|
||||||
|
/// <reference path="./entities/card.ts" />
|
||||||
|
/// <reference path="./entities/context.ts" />
|
||||||
|
/// <reference path="./entities/conversation.ts" />
|
||||||
|
/// <reference path="./entities/emoji.ts" />
|
||||||
|
/// <reference path="./entities/featured_tag.ts" />
|
||||||
|
/// <reference path="./entities/field.ts" />
|
||||||
|
/// <reference path="./entities/filter.ts" />
|
||||||
|
/// <reference path="./entities/history.ts" />
|
||||||
|
/// <reference path="./entities/identity_proof.ts" />
|
||||||
|
/// <reference path="./entities/instance.ts" />
|
||||||
|
/// <reference path="./entities/list.ts" />
|
||||||
|
/// <reference path="./entities/marker.ts" />
|
||||||
|
/// <reference path="./entities/mention.ts" />
|
||||||
|
/// <reference path="./entities/notification.ts" />
|
||||||
|
/// <reference path="./entities/poll.ts" />
|
||||||
|
/// <reference path="./entities/poll_option.ts" />
|
||||||
|
/// <reference path="./entities/preferences.ts" />
|
||||||
|
/// <reference path="./entities/push_subscription.ts" />
|
||||||
|
/// <reference path="./entities/reaction.ts" />
|
||||||
|
/// <reference path="./entities/relationship.ts" />
|
||||||
|
/// <reference path="./entities/report.ts" />
|
||||||
|
/// <reference path="./entities/results.ts" />
|
||||||
|
/// <reference path="./entities/scheduled_status.ts" />
|
||||||
|
/// <reference path="./entities/source.ts" />
|
||||||
|
/// <reference path="./entities/stats.ts" />
|
||||||
|
/// <reference path="./entities/status.ts" />
|
||||||
|
/// <reference path="./entities/status_params.ts" />
|
||||||
|
/// <reference path="./entities/tag.ts" />
|
||||||
|
/// <reference path="./entities/token.ts" />
|
||||||
|
/// <reference path="./entities/urls.ts" />
|
||||||
|
|
||||||
|
export default Entity;
|
11
packages/megalodon/src/filter_context.ts
Normal file
11
packages/megalodon/src/filter_context.ts
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
import Entity from "./entity";
|
||||||
|
|
||||||
|
namespace FilterContext {
|
||||||
|
export const Home: Entity.FilterContext = "home";
|
||||||
|
export const Notifications: Entity.FilterContext = "notifications";
|
||||||
|
export const Public: Entity.FilterContext = "public";
|
||||||
|
export const Thread: Entity.FilterContext = "thread";
|
||||||
|
export const Account: Entity.FilterContext = "account";
|
||||||
|
}
|
||||||
|
|
||||||
|
export default FilterContext;
|
32
packages/megalodon/src/index.ts
Normal file
32
packages/megalodon/src/index.ts
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
import Response from "./response";
|
||||||
|
import OAuth from "./oauth";
|
||||||
|
import { isCancel, RequestCanceledError } from "./cancel";
|
||||||
|
import { ProxyConfig } from "./proxy_config";
|
||||||
|
import generator, {
|
||||||
|
detector,
|
||||||
|
MegalodonInterface,
|
||||||
|
WebSocketInterface,
|
||||||
|
} from "./megalodon";
|
||||||
|
import Misskey from "./misskey";
|
||||||
|
import Entity from "./entity";
|
||||||
|
import NotificationType from "./notification";
|
||||||
|
import FilterContext from "./filter_context";
|
||||||
|
import Converter from "./converter";
|
||||||
|
|
||||||
|
export {
|
||||||
|
Response,
|
||||||
|
OAuth,
|
||||||
|
RequestCanceledError,
|
||||||
|
isCancel,
|
||||||
|
ProxyConfig,
|
||||||
|
detector,
|
||||||
|
MegalodonInterface,
|
||||||
|
WebSocketInterface,
|
||||||
|
NotificationType,
|
||||||
|
FilterContext,
|
||||||
|
Misskey,
|
||||||
|
Entity,
|
||||||
|
Converter,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default generator;
|
1532
packages/megalodon/src/megalodon.ts
Normal file
1532
packages/megalodon/src/megalodon.ts
Normal file
File diff suppressed because it is too large
Load diff
3436
packages/megalodon/src/misskey.ts
Normal file
3436
packages/megalodon/src/misskey.ts
Normal file
File diff suppressed because it is too large
Load diff
727
packages/megalodon/src/misskey/api_client.ts
Normal file
727
packages/megalodon/src/misskey/api_client.ts
Normal file
|
@ -0,0 +1,727 @@
|
||||||
|
import axios, { AxiosResponse, AxiosRequestConfig } from "axios";
|
||||||
|
import dayjs from "dayjs";
|
||||||
|
import FormData from "form-data";
|
||||||
|
|
||||||
|
import { DEFAULT_UA } from "../default";
|
||||||
|
import proxyAgent, { ProxyConfig } from "../proxy_config";
|
||||||
|
import Response from "../response";
|
||||||
|
import MisskeyEntity from "./entity";
|
||||||
|
import MegalodonEntity from "../entity";
|
||||||
|
import WebSocket from "./web_socket";
|
||||||
|
import MisskeyNotificationType from "./notification";
|
||||||
|
import NotificationType from "../notification";
|
||||||
|
|
||||||
|
namespace MisskeyAPI {
|
||||||
|
export namespace Entity {
|
||||||
|
export type App = MisskeyEntity.App;
|
||||||
|
export type Announcement = MisskeyEntity.Announcement;
|
||||||
|
export type Blocking = MisskeyEntity.Blocking;
|
||||||
|
export type Choice = MisskeyEntity.Choice;
|
||||||
|
export type CreatedNote = MisskeyEntity.CreatedNote;
|
||||||
|
export type Emoji = MisskeyEntity.Emoji;
|
||||||
|
export type Favorite = MisskeyEntity.Favorite;
|
||||||
|
export type Field = MisskeyEntity.Field;
|
||||||
|
export type File = MisskeyEntity.File;
|
||||||
|
export type Follower = MisskeyEntity.Follower;
|
||||||
|
export type Following = MisskeyEntity.Following;
|
||||||
|
export type FollowRequest = MisskeyEntity.FollowRequest;
|
||||||
|
export type Hashtag = MisskeyEntity.Hashtag;
|
||||||
|
export type List = MisskeyEntity.List;
|
||||||
|
export type Meta = MisskeyEntity.Meta;
|
||||||
|
export type Mute = MisskeyEntity.Mute;
|
||||||
|
export type Note = MisskeyEntity.Note;
|
||||||
|
export type Notification = MisskeyEntity.Notification;
|
||||||
|
export type Poll = MisskeyEntity.Poll;
|
||||||
|
export type Reaction = MisskeyEntity.Reaction;
|
||||||
|
export type Relation = MisskeyEntity.Relation;
|
||||||
|
export type User = MisskeyEntity.User;
|
||||||
|
export type UserDetail = MisskeyEntity.UserDetail;
|
||||||
|
export type UserDetailMe = MisskeyEntity.UserDetailMe;
|
||||||
|
export type GetAll = MisskeyEntity.GetAll;
|
||||||
|
export type UserKey = MisskeyEntity.UserKey;
|
||||||
|
export type Session = MisskeyEntity.Session;
|
||||||
|
export type Stats = MisskeyEntity.Stats;
|
||||||
|
export type State = MisskeyEntity.State;
|
||||||
|
export type APIEmoji = { emojis: Emoji[] };
|
||||||
|
}
|
||||||
|
|
||||||
|
export class Converter {
|
||||||
|
private baseUrl: string;
|
||||||
|
private instanceHost: string;
|
||||||
|
private plcUrl: string;
|
||||||
|
private modelOfAcct = {
|
||||||
|
id: "1",
|
||||||
|
username: "none",
|
||||||
|
acct: "none",
|
||||||
|
display_name: "none",
|
||||||
|
locked: true,
|
||||||
|
bot: true,
|
||||||
|
discoverable: false,
|
||||||
|
group: false,
|
||||||
|
created_at: "1971-01-01T00:00:00.000Z",
|
||||||
|
note: "",
|
||||||
|
url: "plc",
|
||||||
|
avatar: "plc",
|
||||||
|
avatar_static: "plc",
|
||||||
|
header: "plc",
|
||||||
|
header_static: "plc",
|
||||||
|
followers_count: -1,
|
||||||
|
following_count: 0,
|
||||||
|
statuses_count: 0,
|
||||||
|
last_status_at: "1971-01-01T00:00:00.000Z",
|
||||||
|
noindex: true,
|
||||||
|
emojis: [],
|
||||||
|
fields: [],
|
||||||
|
moved: null,
|
||||||
|
};
|
||||||
|
|
||||||
|
constructor(baseUrl: string) {
|
||||||
|
this.baseUrl = baseUrl;
|
||||||
|
this.instanceHost = baseUrl.substring(baseUrl.indexOf("//") + 2);
|
||||||
|
this.plcUrl = `${baseUrl}/static-assets/transparent.png`;
|
||||||
|
this.modelOfAcct.url = this.plcUrl;
|
||||||
|
this.modelOfAcct.avatar = this.plcUrl;
|
||||||
|
this.modelOfAcct.avatar_static = this.plcUrl;
|
||||||
|
this.modelOfAcct.header = this.plcUrl;
|
||||||
|
this.modelOfAcct.header_static = this.plcUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME: Properly render MFM instead of just escaping HTML characters.
|
||||||
|
escapeMFM = (text: string): string =>
|
||||||
|
text
|
||||||
|
.replace(/&/g, "&")
|
||||||
|
.replace(/</g, "<")
|
||||||
|
.replace(/>/g, ">")
|
||||||
|
.replace(/"/g, """)
|
||||||
|
.replace(/'/g, "'")
|
||||||
|
.replace(/`/g, "`")
|
||||||
|
.replace(/\r?\n/g, "<br>");
|
||||||
|
|
||||||
|
emoji = (e: Entity.Emoji): MegalodonEntity.Emoji => {
|
||||||
|
return {
|
||||||
|
shortcode: e.name,
|
||||||
|
static_url: e.url,
|
||||||
|
url: e.url,
|
||||||
|
visible_in_picker: true,
|
||||||
|
category: e.category,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
field = (f: Entity.Field): MegalodonEntity.Field => ({
|
||||||
|
name: f.name,
|
||||||
|
value: this.escapeMFM(f.value),
|
||||||
|
verified_at: null,
|
||||||
|
});
|
||||||
|
|
||||||
|
user = (u: Entity.User): MegalodonEntity.Account => {
|
||||||
|
let acct = u.username;
|
||||||
|
let acctUrl = `https://${u.host || this.instanceHost}/@${u.username}`;
|
||||||
|
if (u.host) {
|
||||||
|
acct = `${u.username}@${u.host}`;
|
||||||
|
acctUrl = `https://${u.host}/@${u.username}`;
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
id: u.id,
|
||||||
|
username: u.username,
|
||||||
|
acct: acct,
|
||||||
|
display_name: u.name || u.username,
|
||||||
|
locked: false,
|
||||||
|
created_at: new Date().toISOString(),
|
||||||
|
followers_count: 0,
|
||||||
|
following_count: 0,
|
||||||
|
statuses_count: 0,
|
||||||
|
note: "",
|
||||||
|
url: acctUrl,
|
||||||
|
avatar: u.avatarUrl,
|
||||||
|
avatar_static: u.avatarUrl,
|
||||||
|
header: this.plcUrl,
|
||||||
|
header_static: this.plcUrl,
|
||||||
|
emojis: u.emojis.map((e) => this.emoji(e)),
|
||||||
|
moved: null,
|
||||||
|
fields: [],
|
||||||
|
bot: false,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
userDetail = (
|
||||||
|
u: Entity.UserDetail,
|
||||||
|
host: string,
|
||||||
|
): MegalodonEntity.Account => {
|
||||||
|
let acct = u.username;
|
||||||
|
host = host.replace("https://", "");
|
||||||
|
let acctUrl = `https://${host || u.host || this.instanceHost}/@${
|
||||||
|
u.username
|
||||||
|
}`;
|
||||||
|
if (u.host) {
|
||||||
|
acct = `${u.username}@${u.host}`;
|
||||||
|
acctUrl = `https://${u.host}/@${u.username}`;
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
id: u.id,
|
||||||
|
username: u.username,
|
||||||
|
acct: acct,
|
||||||
|
display_name: u.name || u.username,
|
||||||
|
locked: u.isLocked,
|
||||||
|
created_at: u.createdAt,
|
||||||
|
followers_count: u.followersCount,
|
||||||
|
following_count: u.followingCount,
|
||||||
|
statuses_count: u.notesCount,
|
||||||
|
note: u.description?.replace(/\n|\\n/g, "<br>") ?? "",
|
||||||
|
url: acctUrl,
|
||||||
|
avatar: u.avatarUrl,
|
||||||
|
avatar_static: u.avatarUrl,
|
||||||
|
header: u.bannerUrl ?? this.plcUrl,
|
||||||
|
header_static: u.bannerUrl ?? this.plcUrl,
|
||||||
|
emojis: u.emojis.map((e) => this.emoji(e)),
|
||||||
|
moved: null,
|
||||||
|
fields: u.fields.map((f) => this.field(f)),
|
||||||
|
bot: u.isBot,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
userPreferences = (
|
||||||
|
u: MisskeyAPI.Entity.UserDetailMe,
|
||||||
|
v: "public" | "unlisted" | "private" | "direct",
|
||||||
|
): MegalodonEntity.Preferences => {
|
||||||
|
return {
|
||||||
|
"reading:expand:media": "default",
|
||||||
|
"reading:expand:spoilers": false,
|
||||||
|
"posting:default:language": u.lang,
|
||||||
|
"posting:default:sensitive": u.alwaysMarkNsfw,
|
||||||
|
"posting:default:visibility": v,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
visibility = (
|
||||||
|
v: "public" | "home" | "followers" | "specified",
|
||||||
|
): "public" | "unlisted" | "private" | "direct" => {
|
||||||
|
switch (v) {
|
||||||
|
case "public":
|
||||||
|
return v;
|
||||||
|
case "home":
|
||||||
|
return "unlisted";
|
||||||
|
case "followers":
|
||||||
|
return "private";
|
||||||
|
case "specified":
|
||||||
|
return "direct";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
encodeVisibility = (
|
||||||
|
v: "public" | "unlisted" | "private" | "direct",
|
||||||
|
): "public" | "home" | "followers" | "specified" => {
|
||||||
|
switch (v) {
|
||||||
|
case "public":
|
||||||
|
return v;
|
||||||
|
case "unlisted":
|
||||||
|
return "home";
|
||||||
|
case "private":
|
||||||
|
return "followers";
|
||||||
|
case "direct":
|
||||||
|
return "specified";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
fileType = (
|
||||||
|
s: string,
|
||||||
|
): "unknown" | "image" | "gifv" | "video" | "audio" => {
|
||||||
|
if (s === "image/gif") {
|
||||||
|
return "gifv";
|
||||||
|
}
|
||||||
|
if (s.includes("image")) {
|
||||||
|
return "image";
|
||||||
|
}
|
||||||
|
if (s.includes("video")) {
|
||||||
|
return "video";
|
||||||
|
}
|
||||||
|
if (s.includes("audio")) {
|
||||||
|
return "audio";
|
||||||
|
}
|
||||||
|
return "unknown";
|
||||||
|
};
|
||||||
|
|
||||||
|
file = (f: Entity.File): MegalodonEntity.Attachment => {
|
||||||
|
return {
|
||||||
|
id: f.id,
|
||||||
|
type: this.fileType(f.type),
|
||||||
|
url: f.url,
|
||||||
|
remote_url: f.url,
|
||||||
|
preview_url: f.thumbnailUrl,
|
||||||
|
text_url: f.url,
|
||||||
|
meta: {
|
||||||
|
width: f.properties.width,
|
||||||
|
height: f.properties.height,
|
||||||
|
},
|
||||||
|
description: f.comment,
|
||||||
|
blurhash: f.blurhash,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
follower = (f: Entity.Follower): MegalodonEntity.Account => {
|
||||||
|
return this.user(f.follower);
|
||||||
|
};
|
||||||
|
|
||||||
|
following = (f: Entity.Following): MegalodonEntity.Account => {
|
||||||
|
return this.user(f.followee);
|
||||||
|
};
|
||||||
|
|
||||||
|
relation = (r: Entity.Relation): MegalodonEntity.Relationship => {
|
||||||
|
return {
|
||||||
|
id: r.id,
|
||||||
|
following: r.isFollowing,
|
||||||
|
followed_by: r.isFollowed,
|
||||||
|
blocking: r.isBlocking,
|
||||||
|
blocked_by: r.isBlocked,
|
||||||
|
muting: r.isMuted,
|
||||||
|
muting_notifications: false,
|
||||||
|
requested: r.hasPendingFollowRequestFromYou,
|
||||||
|
domain_blocking: false,
|
||||||
|
showing_reblogs: true,
|
||||||
|
endorsed: false,
|
||||||
|
notifying: false,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
choice = (c: Entity.Choice): MegalodonEntity.PollOption => {
|
||||||
|
return {
|
||||||
|
title: c.text,
|
||||||
|
votes_count: c.votes,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
poll = (p: Entity.Poll, id: string): MegalodonEntity.Poll => {
|
||||||
|
const now = dayjs();
|
||||||
|
const expire = dayjs(p.expiresAt);
|
||||||
|
const count = p.choices.reduce((sum, choice) => sum + choice.votes, 0);
|
||||||
|
return {
|
||||||
|
id: id,
|
||||||
|
expires_at: p.expiresAt,
|
||||||
|
expired: now.isAfter(expire),
|
||||||
|
multiple: p.multiple,
|
||||||
|
votes_count: count,
|
||||||
|
options: p.choices.map((c) => this.choice(c)),
|
||||||
|
voted: p.choices.some((c) => c.isVoted),
|
||||||
|
own_votes: p.choices
|
||||||
|
.filter((c) => c.isVoted)
|
||||||
|
.map((c) => p.choices.indexOf(c)),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
note = (n: Entity.Note, host: string): MegalodonEntity.Status => {
|
||||||
|
host = host.replace("https://", "");
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: n.id,
|
||||||
|
uri: n.uri ? n.uri : `https://${host}/notes/${n.id}`,
|
||||||
|
url: n.uri ? n.uri : `https://${host}/notes/${n.id}`,
|
||||||
|
account: this.user(n.user),
|
||||||
|
in_reply_to_id: n.replyId,
|
||||||
|
in_reply_to_account_id: n.reply?.userId ?? null,
|
||||||
|
reblog: n.renote ? this.note(n.renote, host) : null,
|
||||||
|
content: n.text ? this.escapeMFM(n.text) : "",
|
||||||
|
plain_content: n.text ? n.text : null,
|
||||||
|
created_at: n.createdAt,
|
||||||
|
// Remove reaction emojis with names containing @ from the emojis list.
|
||||||
|
emojis: n.emojis
|
||||||
|
.filter((e) => e.name.indexOf("@") === -1)
|
||||||
|
.map((e) => this.emoji(e)),
|
||||||
|
replies_count: n.repliesCount,
|
||||||
|
reblogs_count: n.renoteCount,
|
||||||
|
favourites_count: this.getTotalReactions(n.reactions),
|
||||||
|
reblogged: false,
|
||||||
|
favourited: !!n.myReaction,
|
||||||
|
muted: false,
|
||||||
|
sensitive: n.files ? n.files.some((f) => f.isSensitive) : false,
|
||||||
|
spoiler_text: n.cw ? n.cw : "",
|
||||||
|
visibility: this.visibility(n.visibility),
|
||||||
|
media_attachments: n.files ? n.files.map((f) => this.file(f)) : [],
|
||||||
|
mentions: [],
|
||||||
|
tags: [],
|
||||||
|
card: null,
|
||||||
|
poll: n.poll ? this.poll(n.poll, n.id) : null,
|
||||||
|
application: null,
|
||||||
|
language: null,
|
||||||
|
pinned: null,
|
||||||
|
// Use emojis list to provide URLs for emoji reactions.
|
||||||
|
reactions: this.mapReactions(n.emojis, n.reactions, n.myReaction),
|
||||||
|
bookmarked: false,
|
||||||
|
quote: n.renote && n.text ? this.note(n.renote, host) : null,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
mapReactions = (
|
||||||
|
emojis: Array<MisskeyEntity.Emoji>,
|
||||||
|
r: { [key: string]: number },
|
||||||
|
myReaction?: string,
|
||||||
|
): Array<MegalodonEntity.Reaction> => {
|
||||||
|
// Map of emoji shortcodes to image URLs.
|
||||||
|
const emojiUrls = new Map<string, string>(
|
||||||
|
emojis.map((e) => [e.name, e.url]),
|
||||||
|
);
|
||||||
|
return Object.keys(r).map((key) => {
|
||||||
|
// Strip colons from custom emoji reaction names to match emoji shortcodes.
|
||||||
|
const shortcode = key.replaceAll(":", "");
|
||||||
|
// If this is a custom emoji (vs. a Unicode emoji), find its image URL.
|
||||||
|
const url = emojiUrls.get(shortcode);
|
||||||
|
// Finally, remove trailing @. from local custom emoji reaction names.
|
||||||
|
const name = shortcode.replace("@.", "");
|
||||||
|
return {
|
||||||
|
count: r[key],
|
||||||
|
me: key === myReaction,
|
||||||
|
name,
|
||||||
|
url,
|
||||||
|
// We don't actually have a static version of the asset, but clients expect one anyway.
|
||||||
|
static_url: url,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
getTotalReactions = (r: { [key: string]: number }): number => {
|
||||||
|
return Object.values(r).length > 0
|
||||||
|
? Object.values(r).reduce(
|
||||||
|
(previousValue, currentValue) => previousValue + currentValue,
|
||||||
|
)
|
||||||
|
: 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
reactions = (
|
||||||
|
r: Array<Entity.Reaction>,
|
||||||
|
): Array<MegalodonEntity.Reaction> => {
|
||||||
|
const result: Array<MegalodonEntity.Reaction> = [];
|
||||||
|
for (const e of r) {
|
||||||
|
const i = result.findIndex((res) => res.name === e.type);
|
||||||
|
if (i >= 0) {
|
||||||
|
result[i].count++;
|
||||||
|
} else {
|
||||||
|
result.push({
|
||||||
|
count: 1,
|
||||||
|
me: false,
|
||||||
|
name: e.type,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
noteToConversation = (
|
||||||
|
n: Entity.Note,
|
||||||
|
host: string,
|
||||||
|
): MegalodonEntity.Conversation => {
|
||||||
|
const accounts: Array<MegalodonEntity.Account> = [this.user(n.user)];
|
||||||
|
if (n.reply) {
|
||||||
|
accounts.push(this.user(n.reply.user));
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
id: n.id,
|
||||||
|
accounts: accounts,
|
||||||
|
last_status: this.note(n, host),
|
||||||
|
unread: false,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
list = (l: Entity.List): MegalodonEntity.List => ({
|
||||||
|
id: l.id,
|
||||||
|
title: l.name,
|
||||||
|
});
|
||||||
|
|
||||||
|
encodeNotificationType = (
|
||||||
|
e: MegalodonEntity.NotificationType,
|
||||||
|
): MisskeyEntity.NotificationType => {
|
||||||
|
switch (e) {
|
||||||
|
case NotificationType.Follow:
|
||||||
|
return MisskeyNotificationType.Follow;
|
||||||
|
case NotificationType.Mention:
|
||||||
|
return MisskeyNotificationType.Reply;
|
||||||
|
case NotificationType.Favourite:
|
||||||
|
case NotificationType.Reaction:
|
||||||
|
return MisskeyNotificationType.Reaction;
|
||||||
|
case NotificationType.Reblog:
|
||||||
|
return MisskeyNotificationType.Renote;
|
||||||
|
case NotificationType.Poll:
|
||||||
|
return MisskeyNotificationType.PollEnded;
|
||||||
|
case NotificationType.FollowRequest:
|
||||||
|
return MisskeyNotificationType.ReceiveFollowRequest;
|
||||||
|
default:
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
decodeNotificationType = (
|
||||||
|
e: MisskeyEntity.NotificationType,
|
||||||
|
): MegalodonEntity.NotificationType => {
|
||||||
|
switch (e) {
|
||||||
|
case MisskeyNotificationType.Follow:
|
||||||
|
return NotificationType.Follow;
|
||||||
|
case MisskeyNotificationType.Mention:
|
||||||
|
case MisskeyNotificationType.Reply:
|
||||||
|
return NotificationType.Mention;
|
||||||
|
case MisskeyNotificationType.Renote:
|
||||||
|
case MisskeyNotificationType.Quote:
|
||||||
|
return NotificationType.Reblog;
|
||||||
|
case MisskeyNotificationType.Reaction:
|
||||||
|
return NotificationType.Reaction;
|
||||||
|
case MisskeyNotificationType.PollEnded:
|
||||||
|
return NotificationType.Poll;
|
||||||
|
case MisskeyNotificationType.ReceiveFollowRequest:
|
||||||
|
return NotificationType.FollowRequest;
|
||||||
|
case MisskeyNotificationType.FollowRequestAccepted:
|
||||||
|
return NotificationType.Follow;
|
||||||
|
default:
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
announcement = (a: Entity.Announcement): MegalodonEntity.Announcement => ({
|
||||||
|
id: a.id,
|
||||||
|
content: `<h1>${this.escapeMFM(a.title)}</h1>${this.escapeMFM(a.text)}`,
|
||||||
|
starts_at: null,
|
||||||
|
ends_at: null,
|
||||||
|
published: true,
|
||||||
|
all_day: false,
|
||||||
|
published_at: a.createdAt,
|
||||||
|
updated_at: a.updatedAt,
|
||||||
|
read: a.isRead,
|
||||||
|
mentions: [],
|
||||||
|
statuses: [],
|
||||||
|
tags: [],
|
||||||
|
emojis: [],
|
||||||
|
reactions: [],
|
||||||
|
});
|
||||||
|
|
||||||
|
notification = (
|
||||||
|
n: Entity.Notification,
|
||||||
|
host: string,
|
||||||
|
): MegalodonEntity.Notification => {
|
||||||
|
let notification = {
|
||||||
|
id: n.id,
|
||||||
|
account: n.user ? this.user(n.user) : this.modelOfAcct,
|
||||||
|
created_at: n.createdAt,
|
||||||
|
type: this.decodeNotificationType(n.type),
|
||||||
|
};
|
||||||
|
if (n.note) {
|
||||||
|
notification = Object.assign(notification, {
|
||||||
|
status: this.note(n.note, host),
|
||||||
|
});
|
||||||
|
if (notification.type === NotificationType.Poll) {
|
||||||
|
notification = Object.assign(notification, {
|
||||||
|
account: this.note(n.note, host).account,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (n.reaction) {
|
||||||
|
notification = Object.assign(notification, {
|
||||||
|
reaction: this.mapReactions(n.note.emojis, { [n.reaction]: 1 })[0],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return notification;
|
||||||
|
};
|
||||||
|
|
||||||
|
stats = (s: Entity.Stats): MegalodonEntity.Stats => {
|
||||||
|
return {
|
||||||
|
user_count: s.usersCount,
|
||||||
|
status_count: s.notesCount,
|
||||||
|
domain_count: s.instances,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
meta = (m: Entity.Meta, s: Entity.Stats): MegalodonEntity.Instance => {
|
||||||
|
const wss = m.uri.replace(/^https:\/\//, "wss://");
|
||||||
|
return {
|
||||||
|
uri: m.uri,
|
||||||
|
title: m.name,
|
||||||
|
description: m.description,
|
||||||
|
email: m.maintainerEmail,
|
||||||
|
version: m.version,
|
||||||
|
thumbnail: m.bannerUrl,
|
||||||
|
urls: {
|
||||||
|
streaming_api: `${wss}/streaming`,
|
||||||
|
},
|
||||||
|
stats: this.stats(s),
|
||||||
|
languages: m.langs,
|
||||||
|
contact_account: null,
|
||||||
|
max_toot_chars: m.maxNoteTextLength,
|
||||||
|
registrations: !m.disableRegistration,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
hashtag = (h: Entity.Hashtag): MegalodonEntity.Tag => {
|
||||||
|
return {
|
||||||
|
name: h.tag,
|
||||||
|
url: h.tag,
|
||||||
|
history: null,
|
||||||
|
following: false,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export const DEFAULT_SCOPE = [
|
||||||
|
"read:account",
|
||||||
|
"write:account",
|
||||||
|
"read:blocks",
|
||||||
|
"write:blocks",
|
||||||
|
"read:drive",
|
||||||
|
"write:drive",
|
||||||
|
"read:favorites",
|
||||||
|
"write:favorites",
|
||||||
|
"read:following",
|
||||||
|
"write:following",
|
||||||
|
"read:mutes",
|
||||||
|
"write:mutes",
|
||||||
|
"write:notes",
|
||||||
|
"read:notifications",
|
||||||
|
"write:notifications",
|
||||||
|
"read:reactions",
|
||||||
|
"write:reactions",
|
||||||
|
"write:votes",
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface
|
||||||
|
*/
|
||||||
|
export interface Interface {
|
||||||
|
post<T = any>(
|
||||||
|
path: string,
|
||||||
|
params?: any,
|
||||||
|
headers?: { [key: string]: string },
|
||||||
|
): Promise<Response<T>>;
|
||||||
|
cancel(): void;
|
||||||
|
socket(
|
||||||
|
channel:
|
||||||
|
| "user"
|
||||||
|
| "localTimeline"
|
||||||
|
| "hybridTimeline"
|
||||||
|
| "globalTimeline"
|
||||||
|
| "conversation"
|
||||||
|
| "list",
|
||||||
|
listId?: string,
|
||||||
|
): WebSocket;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Misskey API client.
|
||||||
|
*
|
||||||
|
* Usign axios for request, you will handle promises.
|
||||||
|
*/
|
||||||
|
export class Client implements Interface {
|
||||||
|
private accessToken: string | null;
|
||||||
|
private baseUrl: string;
|
||||||
|
private userAgent: string;
|
||||||
|
private abortController: AbortController;
|
||||||
|
private proxyConfig: ProxyConfig | false = false;
|
||||||
|
private converter: Converter;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param baseUrl hostname or base URL
|
||||||
|
* @param accessToken access token from OAuth2 authorization
|
||||||
|
* @param userAgent UserAgent is specified in header on request.
|
||||||
|
* @param proxyConfig Proxy setting, or set false if don't use proxy.
|
||||||
|
* @param converter Converter instance.
|
||||||
|
*/
|
||||||
|
constructor(
|
||||||
|
baseUrl: string,
|
||||||
|
accessToken: string | null,
|
||||||
|
userAgent: string = DEFAULT_UA,
|
||||||
|
proxyConfig: ProxyConfig | false = false,
|
||||||
|
converter: Converter,
|
||||||
|
) {
|
||||||
|
this.accessToken = accessToken;
|
||||||
|
this.baseUrl = baseUrl;
|
||||||
|
this.userAgent = userAgent;
|
||||||
|
this.proxyConfig = proxyConfig;
|
||||||
|
this.abortController = new AbortController();
|
||||||
|
this.converter = converter;
|
||||||
|
axios.defaults.signal = this.abortController.signal;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* POST request to mastodon REST API.
|
||||||
|
* @param path relative path from baseUrl
|
||||||
|
* @param params Form data
|
||||||
|
* @param headers Request header object
|
||||||
|
*/
|
||||||
|
public async post<T>(
|
||||||
|
path: string,
|
||||||
|
params: any = {},
|
||||||
|
headers: { [key: string]: string } = {},
|
||||||
|
): Promise<Response<T>> {
|
||||||
|
let options: AxiosRequestConfig = {
|
||||||
|
headers: headers,
|
||||||
|
maxContentLength: Infinity,
|
||||||
|
maxBodyLength: Infinity,
|
||||||
|
};
|
||||||
|
if (this.proxyConfig) {
|
||||||
|
options = Object.assign(options, {
|
||||||
|
httpAgent: proxyAgent(this.proxyConfig),
|
||||||
|
httpsAgent: proxyAgent(this.proxyConfig),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
let bodyParams = params;
|
||||||
|
if (this.accessToken) {
|
||||||
|
if (params instanceof FormData) {
|
||||||
|
bodyParams.append("i", this.accessToken);
|
||||||
|
} else {
|
||||||
|
bodyParams = Object.assign(params, {
|
||||||
|
i: this.accessToken,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return axios
|
||||||
|
.post<T>(this.baseUrl + path, bodyParams, options)
|
||||||
|
.then((resp: AxiosResponse<T>) => {
|
||||||
|
const res: Response<T> = {
|
||||||
|
data: resp.data,
|
||||||
|
status: resp.status,
|
||||||
|
statusText: resp.statusText,
|
||||||
|
headers: resp.headers,
|
||||||
|
};
|
||||||
|
return res;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cancel all requests in this instance.
|
||||||
|
* @returns void
|
||||||
|
*/
|
||||||
|
public cancel(): void {
|
||||||
|
return this.abortController.abort();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get connection and receive websocket connection for Misskey API.
|
||||||
|
*
|
||||||
|
* @param channel Channel name is user, localTimeline, hybridTimeline, globalTimeline, conversation or list.
|
||||||
|
* @param listId This parameter is required only list channel.
|
||||||
|
*/
|
||||||
|
public socket(
|
||||||
|
channel:
|
||||||
|
| "user"
|
||||||
|
| "localTimeline"
|
||||||
|
| "hybridTimeline"
|
||||||
|
| "globalTimeline"
|
||||||
|
| "conversation"
|
||||||
|
| "list",
|
||||||
|
listId?: string,
|
||||||
|
): WebSocket {
|
||||||
|
if (!this.accessToken) {
|
||||||
|
throw new Error("accessToken is required");
|
||||||
|
}
|
||||||
|
const url = `${this.baseUrl}/streaming`;
|
||||||
|
const streaming = new WebSocket(
|
||||||
|
url,
|
||||||
|
channel,
|
||||||
|
this.accessToken,
|
||||||
|
listId,
|
||||||
|
this.userAgent,
|
||||||
|
this.proxyConfig,
|
||||||
|
this.converter,
|
||||||
|
);
|
||||||
|
process.nextTick(() => {
|
||||||
|
streaming.start();
|
||||||
|
});
|
||||||
|
return streaming;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default MisskeyAPI;
|
6
packages/megalodon/src/misskey/entities/GetAll.ts
Normal file
6
packages/megalodon/src/misskey/entities/GetAll.ts
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type GetAll = {
|
||||||
|
tutorial: number;
|
||||||
|
defaultNoteVisibility: "public" | "home" | "followers" | "specified";
|
||||||
|
};
|
||||||
|
}
|
10
packages/megalodon/src/misskey/entities/announcement.ts
Normal file
10
packages/megalodon/src/misskey/entities/announcement.ts
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Announcement = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
updatedAt: string;
|
||||||
|
text: string;
|
||||||
|
title: string;
|
||||||
|
isRead?: boolean;
|
||||||
|
};
|
||||||
|
}
|
9
packages/megalodon/src/misskey/entities/app.ts
Normal file
9
packages/megalodon/src/misskey/entities/app.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type App = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
callbackUrl: string;
|
||||||
|
permission: Array<string>;
|
||||||
|
secret: string;
|
||||||
|
};
|
||||||
|
}
|
10
packages/megalodon/src/misskey/entities/blocking.ts
Normal file
10
packages/megalodon/src/misskey/entities/blocking.ts
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
/// <reference path="userDetail.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Blocking = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
blockeeId: string;
|
||||||
|
blockee: UserDetail;
|
||||||
|
};
|
||||||
|
}
|
7
packages/megalodon/src/misskey/entities/createdNote.ts
Normal file
7
packages/megalodon/src/misskey/entities/createdNote.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
/// <reference path="note.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type CreatedNote = {
|
||||||
|
createdNote: Note;
|
||||||
|
};
|
||||||
|
}
|
9
packages/megalodon/src/misskey/entities/emoji.ts
Normal file
9
packages/megalodon/src/misskey/entities/emoji.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Emoji = {
|
||||||
|
name: string;
|
||||||
|
host: string | null;
|
||||||
|
url: string;
|
||||||
|
aliases: Array<string>;
|
||||||
|
category: string;
|
||||||
|
};
|
||||||
|
}
|
10
packages/megalodon/src/misskey/entities/favorite.ts
Normal file
10
packages/megalodon/src/misskey/entities/favorite.ts
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
/// <reference path="note.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Favorite = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
noteId: string;
|
||||||
|
note: Note;
|
||||||
|
};
|
||||||
|
}
|
7
packages/megalodon/src/misskey/entities/field.ts
Normal file
7
packages/megalodon/src/misskey/entities/field.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Field = {
|
||||||
|
name: string;
|
||||||
|
value: string;
|
||||||
|
verified?: string;
|
||||||
|
};
|
||||||
|
}
|
20
packages/megalodon/src/misskey/entities/file.ts
Normal file
20
packages/megalodon/src/misskey/entities/file.ts
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type File = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
name: string;
|
||||||
|
type: string;
|
||||||
|
md5: string;
|
||||||
|
size: number;
|
||||||
|
isSensitive: boolean;
|
||||||
|
properties: {
|
||||||
|
width: number;
|
||||||
|
height: number;
|
||||||
|
avgColor: string;
|
||||||
|
};
|
||||||
|
url: string;
|
||||||
|
thumbnailUrl: string;
|
||||||
|
comment: string;
|
||||||
|
blurhash: string;
|
||||||
|
};
|
||||||
|
}
|
9
packages/megalodon/src/misskey/entities/followRequest.ts
Normal file
9
packages/megalodon/src/misskey/entities/followRequest.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
/// <reference path="user.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type FollowRequest = {
|
||||||
|
id: string;
|
||||||
|
follower: User;
|
||||||
|
followee: User;
|
||||||
|
};
|
||||||
|
}
|
11
packages/megalodon/src/misskey/entities/follower.ts
Normal file
11
packages/megalodon/src/misskey/entities/follower.ts
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
/// <reference path="userDetail.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Follower = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
followeeId: string;
|
||||||
|
followerId: string;
|
||||||
|
follower: UserDetail;
|
||||||
|
};
|
||||||
|
}
|
11
packages/megalodon/src/misskey/entities/following.ts
Normal file
11
packages/megalodon/src/misskey/entities/following.ts
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
/// <reference path="userDetail.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Following = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
followeeId: string;
|
||||||
|
followerId: string;
|
||||||
|
followee: UserDetail;
|
||||||
|
};
|
||||||
|
}
|
7
packages/megalodon/src/misskey/entities/hashtag.ts
Normal file
7
packages/megalodon/src/misskey/entities/hashtag.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Hashtag = {
|
||||||
|
tag: string;
|
||||||
|
chart: Array<number>;
|
||||||
|
usersCount: number;
|
||||||
|
};
|
||||||
|
}
|
8
packages/megalodon/src/misskey/entities/list.ts
Normal file
8
packages/megalodon/src/misskey/entities/list.ts
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type List = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
name: string;
|
||||||
|
userIds: Array<string>;
|
||||||
|
};
|
||||||
|
}
|
18
packages/megalodon/src/misskey/entities/meta.ts
Normal file
18
packages/megalodon/src/misskey/entities/meta.ts
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
/// <reference path="emoji.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Meta = {
|
||||||
|
maintainerName: string;
|
||||||
|
maintainerEmail: string;
|
||||||
|
name: string;
|
||||||
|
version: string;
|
||||||
|
uri: string;
|
||||||
|
description: string;
|
||||||
|
langs: Array<string>;
|
||||||
|
disableRegistration: boolean;
|
||||||
|
disableLocalTimeline: boolean;
|
||||||
|
bannerUrl: string;
|
||||||
|
maxNoteTextLength: 3000;
|
||||||
|
emojis: Array<Emoji>;
|
||||||
|
};
|
||||||
|
}
|
10
packages/megalodon/src/misskey/entities/mute.ts
Normal file
10
packages/megalodon/src/misskey/entities/mute.ts
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
/// <reference path="userDetail.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Mute = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
muteeId: string;
|
||||||
|
mutee: UserDetail;
|
||||||
|
};
|
||||||
|
}
|
32
packages/megalodon/src/misskey/entities/note.ts
Normal file
32
packages/megalodon/src/misskey/entities/note.ts
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
/// <reference path="user.ts" />
|
||||||
|
/// <reference path="emoji.ts" />
|
||||||
|
/// <reference path="file.ts" />
|
||||||
|
/// <reference path="poll.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Note = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
userId: string;
|
||||||
|
user: User;
|
||||||
|
text: string | null;
|
||||||
|
cw: string | null;
|
||||||
|
visibility: "public" | "home" | "followers" | "specified";
|
||||||
|
renoteCount: number;
|
||||||
|
repliesCount: number;
|
||||||
|
reactions: { [key: string]: number };
|
||||||
|
emojis: Array<Emoji>;
|
||||||
|
fileIds: Array<string>;
|
||||||
|
files: Array<File>;
|
||||||
|
replyId: string | null;
|
||||||
|
renoteId: string | null;
|
||||||
|
uri?: string;
|
||||||
|
reply?: Note;
|
||||||
|
renote?: Note;
|
||||||
|
viaMobile?: boolean;
|
||||||
|
tags?: Array<string>;
|
||||||
|
poll?: Poll;
|
||||||
|
mentions?: Array<string>;
|
||||||
|
myReaction?: string;
|
||||||
|
};
|
||||||
|
}
|
17
packages/megalodon/src/misskey/entities/notification.ts
Normal file
17
packages/megalodon/src/misskey/entities/notification.ts
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
/// <reference path="user.ts" />
|
||||||
|
/// <reference path="note.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Notification = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
// https://github.com/syuilo/misskey/blob/056942391aee135eb6c77aaa63f6ed5741d701a6/src/models/entities/notification.ts#L50-L62
|
||||||
|
type: NotificationType;
|
||||||
|
userId: string;
|
||||||
|
user: User;
|
||||||
|
note?: Note;
|
||||||
|
reaction?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type NotificationType = string;
|
||||||
|
}
|
13
packages/megalodon/src/misskey/entities/poll.ts
Normal file
13
packages/megalodon/src/misskey/entities/poll.ts
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Choice = {
|
||||||
|
text: string;
|
||||||
|
votes: number;
|
||||||
|
isVoted: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Poll = {
|
||||||
|
multiple: boolean;
|
||||||
|
expiresAt: string;
|
||||||
|
choices: Array<Choice>;
|
||||||
|
};
|
||||||
|
}
|
11
packages/megalodon/src/misskey/entities/reaction.ts
Normal file
11
packages/megalodon/src/misskey/entities/reaction.ts
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
/// <reference path="user.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Reaction = {
|
||||||
|
id: string;
|
||||||
|
createdAt: string;
|
||||||
|
user: User;
|
||||||
|
url?: string;
|
||||||
|
type: string;
|
||||||
|
};
|
||||||
|
}
|
12
packages/megalodon/src/misskey/entities/relation.ts
Normal file
12
packages/megalodon/src/misskey/entities/relation.ts
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Relation = {
|
||||||
|
id: string;
|
||||||
|
isFollowing: boolean;
|
||||||
|
hasPendingFollowRequestFromYou: boolean;
|
||||||
|
hasPendingFollowRequestToYou: boolean;
|
||||||
|
isFollowed: boolean;
|
||||||
|
isBlocking: boolean;
|
||||||
|
isBlocked: boolean;
|
||||||
|
isMuted: boolean;
|
||||||
|
};
|
||||||
|
}
|
6
packages/megalodon/src/misskey/entities/session.ts
Normal file
6
packages/megalodon/src/misskey/entities/session.ts
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Session = {
|
||||||
|
token: string;
|
||||||
|
url: string;
|
||||||
|
};
|
||||||
|
}
|
7
packages/megalodon/src/misskey/entities/state.ts
Normal file
7
packages/megalodon/src/misskey/entities/state.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type State = {
|
||||||
|
isFavorited: boolean;
|
||||||
|
isMutedThread: boolean;
|
||||||
|
isWatching: boolean;
|
||||||
|
};
|
||||||
|
}
|
9
packages/megalodon/src/misskey/entities/stats.ts
Normal file
9
packages/megalodon/src/misskey/entities/stats.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type Stats = {
|
||||||
|
notesCount: number;
|
||||||
|
originalNotesCount: number;
|
||||||
|
usersCount: number;
|
||||||
|
originalUsersCount: number;
|
||||||
|
instances: number;
|
||||||
|
};
|
||||||
|
}
|
13
packages/megalodon/src/misskey/entities/user.ts
Normal file
13
packages/megalodon/src/misskey/entities/user.ts
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
/// <reference path="emoji.ts" />
|
||||||
|
|
||||||
|
namespace MisskeyEntity {
|
||||||
|
export type User = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
username: string;
|
||||||
|
host: string | null;
|
||||||
|
avatarUrl: string;
|
||||||
|
avatarColor: string;
|
||||||
|
emojis: Array<Emoji>;
|
||||||
|
};
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue