mirror of
https://github.com/misskey-dev/misskey.git
synced 2025-01-04 23:32:04 +01:00
Resolve #1736
This commit is contained in:
parent
a7e60f80bd
commit
c932f7a25b
5 changed files with 140 additions and 1 deletions
|
@ -101,7 +101,7 @@
|
||||||
</ui-select>
|
</ui-select>
|
||||||
<ui-horizon-group class="fit-bottom">
|
<ui-horizon-group class="fit-bottom">
|
||||||
<ui-button @click="doExport()"><fa :icon="faDownload"/> {{ $t('export') }}</ui-button>
|
<ui-button @click="doExport()"><fa :icon="faDownload"/> {{ $t('export') }}</ui-button>
|
||||||
<ui-button @click="doImport()" :disabled="!['user-lists'].includes(exportTarget)"><fa :icon="faUpload"/> {{ $t('import') }}</ui-button>
|
<ui-button @click="doImport()" :disabled="!['following', 'user-lists'].includes(exportTarget)"><fa :icon="faUpload"/> {{ $t('import') }}</ui-button>
|
||||||
</ui-horizon-group>
|
</ui-horizon-group>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
@ -301,6 +301,7 @@ export default Vue.extend({
|
||||||
doImport() {
|
doImport() {
|
||||||
this.$chooseDriveFile().then(file => {
|
this.$chooseDriveFile().then(file => {
|
||||||
this.$root.api(
|
this.$root.api(
|
||||||
|
this.exportTarget == 'following' ? 'i/import-following' :
|
||||||
this.exportTarget == 'user-lists' ? 'i/import-user-lists' :
|
this.exportTarget == 'user-lists' ? 'i/import-user-lists' :
|
||||||
null, {
|
null, {
|
||||||
fileId: file.id
|
fileId: file.id
|
||||||
|
|
|
@ -146,6 +146,16 @@ export function createExportUserListsJob(user: ILocalUser) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function createImportFollowingJob(user: ILocalUser, fileId: IDriveFile['_id']) {
|
||||||
|
return dbQueue.add('importFollowing', {
|
||||||
|
user: user,
|
||||||
|
fileId: fileId
|
||||||
|
}, {
|
||||||
|
removeOnComplete: true,
|
||||||
|
removeOnFail: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
export function createImportUserListsJob(user: ILocalUser, fileId: IDriveFile['_id']) {
|
export function createImportUserListsJob(user: ILocalUser, fileId: IDriveFile['_id']) {
|
||||||
return dbQueue.add('importUserLists', {
|
return dbQueue.add('importUserLists', {
|
||||||
user: user,
|
user: user,
|
||||||
|
|
62
src/queue/processors/db/import-following.ts
Normal file
62
src/queue/processors/db/import-following.ts
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
import * as Bull from 'bull';
|
||||||
|
import * as mongo from 'mongodb';
|
||||||
|
|
||||||
|
import { queueLogger } from '../../logger';
|
||||||
|
import User from '../../../models/user';
|
||||||
|
import config from '../../../config';
|
||||||
|
import follow from '../../../services/following/create';
|
||||||
|
import DriveFile from '../../../models/drive-file';
|
||||||
|
import { getOriginalUrl } from '../../../misc/get-drive-file-url';
|
||||||
|
import parseAcct from '../../../misc/acct/parse';
|
||||||
|
import resolveUser from '../../../remote/resolve-user';
|
||||||
|
import { downloadTextFile } from '../../../misc/download-text-file';
|
||||||
|
import Following from '../../../models/following';
|
||||||
|
|
||||||
|
const logger = queueLogger.createSubLogger('import-following');
|
||||||
|
|
||||||
|
export async function importFollowing(job: Bull.Job, done: any): Promise<void> {
|
||||||
|
logger.info(`Importing following of ${job.data.user._id} ...`);
|
||||||
|
|
||||||
|
const user = await User.findOne({
|
||||||
|
_id: new mongo.ObjectID(job.data.user._id.toString())
|
||||||
|
});
|
||||||
|
|
||||||
|
const file = await DriveFile.findOne({
|
||||||
|
_id: new mongo.ObjectID(job.data.fileId.toString())
|
||||||
|
});
|
||||||
|
|
||||||
|
const url = getOriginalUrl(file);
|
||||||
|
|
||||||
|
const csv = await downloadTextFile(url);
|
||||||
|
|
||||||
|
for (const line of csv.trim().split('\n')) {
|
||||||
|
const { username, host } = parseAcct(line.trim());
|
||||||
|
|
||||||
|
let target = host === config.host ? await User.findOne({
|
||||||
|
host: null,
|
||||||
|
usernameLower: username.toLowerCase()
|
||||||
|
}) : await User.findOne({
|
||||||
|
host: host,
|
||||||
|
usernameLower: username.toLowerCase()
|
||||||
|
});
|
||||||
|
|
||||||
|
if (host == null && target == null) continue;
|
||||||
|
|
||||||
|
if (target == null) {
|
||||||
|
target = await resolveUser(username, host);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if already following
|
||||||
|
const exist = await Following.findOne({
|
||||||
|
followerId: user._id,
|
||||||
|
followeeId: target._id
|
||||||
|
});
|
||||||
|
|
||||||
|
if (exist) continue;
|
||||||
|
|
||||||
|
follow(user, target);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.succ('Imported');
|
||||||
|
done();
|
||||||
|
}
|
|
@ -6,6 +6,7 @@ import { exportFollowing } from './export-following';
|
||||||
import { exportMute } from './export-mute';
|
import { exportMute } from './export-mute';
|
||||||
import { exportBlocking } from './export-blocking';
|
import { exportBlocking } from './export-blocking';
|
||||||
import { exportUserLists } from './export-user-lists';
|
import { exportUserLists } from './export-user-lists';
|
||||||
|
import { importFollowing } from './import-following';
|
||||||
import { importUserLists } from './import-user-lists';
|
import { importUserLists } from './import-user-lists';
|
||||||
|
|
||||||
const jobs = {
|
const jobs = {
|
||||||
|
@ -16,6 +17,7 @@ const jobs = {
|
||||||
exportMute,
|
exportMute,
|
||||||
exportBlocking,
|
exportBlocking,
|
||||||
exportUserLists,
|
exportUserLists,
|
||||||
|
importFollowing,
|
||||||
importUserLists
|
importUserLists
|
||||||
} as any;
|
} as any;
|
||||||
|
|
||||||
|
|
64
src/server/api/endpoints/i/import-following.ts
Normal file
64
src/server/api/endpoints/i/import-following.ts
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
import $ from 'cafy';
|
||||||
|
import ID, { transform } from '../../../../misc/cafy-id';
|
||||||
|
import define from '../../define';
|
||||||
|
import { createImportFollowingJob } from '../../../../queue';
|
||||||
|
import ms = require('ms');
|
||||||
|
import DriveFile from '../../../../models/drive-file';
|
||||||
|
import { ApiError } from '../../error';
|
||||||
|
|
||||||
|
export const meta = {
|
||||||
|
secure: true,
|
||||||
|
requireCredential: true,
|
||||||
|
limit: {
|
||||||
|
duration: ms('1hour'),
|
||||||
|
max: 1,
|
||||||
|
},
|
||||||
|
|
||||||
|
params: {
|
||||||
|
fileId: {
|
||||||
|
validator: $.type(ID),
|
||||||
|
transform: transform,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
errors: {
|
||||||
|
noSuchFile: {
|
||||||
|
message: 'No such file.',
|
||||||
|
code: 'NO_SUCH_FILE',
|
||||||
|
id: 'b98644cf-a5ac-4277-a502-0b8054a709a3'
|
||||||
|
},
|
||||||
|
|
||||||
|
unexpectedFileType: {
|
||||||
|
message: 'We need csv file.',
|
||||||
|
code: 'UNEXPECTED_FILE_TYPE',
|
||||||
|
id: '660f3599-bce0-4f95-9dde-311fd841c183'
|
||||||
|
},
|
||||||
|
|
||||||
|
tooBigFile: {
|
||||||
|
message: 'That file is too big.',
|
||||||
|
code: 'TOO_BIG_FILE',
|
||||||
|
id: 'dee9d4ed-ad07-43ed-8b34-b2856398bc60'
|
||||||
|
},
|
||||||
|
|
||||||
|
emptyFile: {
|
||||||
|
message: 'That file is empty.',
|
||||||
|
code: 'EMPTY_FILE',
|
||||||
|
id: '31a1b42c-06f7-42ae-8a38-a661c5c9f691'
|
||||||
|
},
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export default define(meta, async (ps, user) => {
|
||||||
|
const file = await DriveFile.findOne({
|
||||||
|
_id: ps.fileId
|
||||||
|
});
|
||||||
|
|
||||||
|
if (file == null) throw new ApiError(meta.errors.noSuchFile);
|
||||||
|
//if (!file.contentType.endsWith('/csv')) throw new ApiError(meta.errors.unexpectedFileType);
|
||||||
|
if (file.length > 50000) throw new ApiError(meta.errors.tooBigFile);
|
||||||
|
if (file.length === 0) throw new ApiError(meta.errors.emptyFile);
|
||||||
|
|
||||||
|
createImportFollowingJob(user, file._id);
|
||||||
|
|
||||||
|
return;
|
||||||
|
});
|
Loading…
Reference in a new issue