Add an import feature for import Trello board as JSON to Planka
parent
a3d1a8a09a
commit
a729e11636
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="JavaScriptLibraryMappings">
|
||||
<includedPredefinedLibrary name="Node.js Core" />
|
||||
</component>
|
||||
</project>
|
||||
@ -0,0 +1,164 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="AutoImportSettings">
|
||||
<option name="autoReloadType" value="SELECTIVE" />
|
||||
</component>
|
||||
<component name="ChangeListManager">
|
||||
<list default="true" id="1caf0577-5a29-440e-83bc-e061a28fd9bd" name="Default Changelist" comment="">
|
||||
<change afterPath="$PROJECT_DIR$/client/src/assets/images/import-icon.svg" afterDir="false" />
|
||||
<change afterPath="$PROJECT_DIR$/server/api/controllers/projects/import.js" afterDir="false" />
|
||||
<change afterPath="$PROJECT_DIR$/server/api/helpers/download-import-data.js" afterDir="false" />
|
||||
<change afterPath="$PROJECT_DIR$/server/api/helpers/get-color-from-trello.js" afterDir="false" />
|
||||
<change afterPath="$PROJECT_DIR$/server/api/helpers/import-attachment-receiver.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/package-lock.json" beforeDir="false" afterPath="$PROJECT_DIR$/client/package-lock.json" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/actions/entry/project.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/actions/entry/project.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/actions/project.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/actions/project.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/api/projects.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/api/projects.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/components/Projects/Projects.jsx" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/components/Projects/Projects.jsx" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/constants/ActionTypes.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/constants/ActionTypes.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/constants/EntryActionTypes.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/constants/EntryActionTypes.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/containers/ProjectsContainer.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/containers/ProjectsContainer.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/locales/en/core.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/locales/en/core.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/sagas/core/requests/project.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/sagas/core/requests/project.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/sagas/core/services/project.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/sagas/core/services/project.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/client/src/sagas/core/watchers/project.js" beforeDir="false" afterPath="$PROJECT_DIR$/client/src/sagas/core/watchers/project.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/server/api/helpers/create-project.js" beforeDir="false" afterPath="$PROJECT_DIR$/server/api/helpers/create-project.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/server/config/policies.js" beforeDir="false" afterPath="$PROJECT_DIR$/server/config/policies.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/server/config/routes.js" beforeDir="false" afterPath="$PROJECT_DIR$/server/config/routes.js" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/server/package-lock.json" beforeDir="false" afterPath="$PROJECT_DIR$/server/package-lock.json" afterDir="false" />
|
||||
<change beforePath="$PROJECT_DIR$/server/package.json" beforeDir="false" afterPath="$PROJECT_DIR$/server/package.json" afterDir="false" />
|
||||
</list>
|
||||
<option name="SHOW_DIALOG" value="false" />
|
||||
<option name="HIGHLIGHT_CONFLICTS" value="true" />
|
||||
<option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
|
||||
<option name="LAST_RESOLUTION" value="IGNORE" />
|
||||
</component>
|
||||
<component name="FileTemplateManagerImpl">
|
||||
<option name="RECENT_TEMPLATES">
|
||||
<list>
|
||||
<option value="JavaScript File" />
|
||||
</list>
|
||||
</option>
|
||||
</component>
|
||||
<component name="Git.Settings">
|
||||
<option name="RECENT_GIT_ROOT_PATH" value="$PROJECT_DIR$" />
|
||||
</component>
|
||||
<component name="GitSEFilterConfiguration">
|
||||
<file-type-list>
|
||||
<filtered-out-file-type name="LOCAL_BRANCH" />
|
||||
<filtered-out-file-type name="REMOTE_BRANCH" />
|
||||
<filtered-out-file-type name="TAG" />
|
||||
<filtered-out-file-type name="COMMIT_BY_MESSAGE" />
|
||||
</file-type-list>
|
||||
</component>
|
||||
<component name="ProjectId" id="1nKo8aaBkVIUrr6v0k4kbcQNnFj" />
|
||||
<component name="ProjectViewState">
|
||||
<option name="hideEmptyMiddlePackages" value="true" />
|
||||
<option name="showLibraryContents" value="true" />
|
||||
</component>
|
||||
<component name="PropertiesComponent">
|
||||
<property name="RunOnceActivity.OpenProjectViewOnStart" value="true" />
|
||||
<property name="RunOnceActivity.ShowReadmeOnStart" value="true" />
|
||||
<property name="WebServerToolWindowFactoryState" value="true" />
|
||||
<property name="javascript.nodejs.core.library.configured.version" value="14.15.0" />
|
||||
<property name="javascript.nodejs.core.library.typings.version" value="14.14.22" />
|
||||
<property name="last_opened_file_path" value="$USER_HOME$/Documents/fr.dic" />
|
||||
<property name="nodejs_interpreter_path.stuck_in_default_project" value="undefined stuck path" />
|
||||
<property name="nodejs_npm_path_reset_for_default_project" value="true" />
|
||||
<property name="nodejs_package_manager_path" value="npm" />
|
||||
<property name="settings.editor.selected.configurable" value="web.server" />
|
||||
<property name="vue.rearranger.settings.migration" value="true" />
|
||||
</component>
|
||||
<component name="RunManager" selected="npm.start">
|
||||
<configuration name="postinstall" type="js.build_tools.npm" temporary="true" nameIsGenerated="true">
|
||||
<package-json value="$PROJECT_DIR$/package.json" />
|
||||
<command value="run" />
|
||||
<scripts>
|
||||
<script value="postinstall" />
|
||||
</scripts>
|
||||
<node-interpreter value="project" />
|
||||
<envs />
|
||||
<method v="2" />
|
||||
</configuration>
|
||||
<configuration name="server:db:init" type="js.build_tools.npm" temporary="true" nameIsGenerated="true">
|
||||
<package-json value="$PROJECT_DIR$/package.json" />
|
||||
<command value="run" />
|
||||
<scripts>
|
||||
<script value="server:db:init" />
|
||||
</scripts>
|
||||
<node-interpreter value="project" />
|
||||
<envs />
|
||||
<method v="2" />
|
||||
</configuration>
|
||||
<configuration name="server:db:migrate" type="js.build_tools.npm" temporary="true" nameIsGenerated="true">
|
||||
<package-json value="$PROJECT_DIR$/package.json" />
|
||||
<command value="run" />
|
||||
<scripts>
|
||||
<script value="server:db:migrate" />
|
||||
</scripts>
|
||||
<node-interpreter value="project" />
|
||||
<envs />
|
||||
<method v="2" />
|
||||
</configuration>
|
||||
<configuration name="server:db:seed" type="js.build_tools.npm" temporary="true" nameIsGenerated="true">
|
||||
<package-json value="$PROJECT_DIR$/package.json" />
|
||||
<command value="run" />
|
||||
<scripts>
|
||||
<script value="server:db:seed" />
|
||||
</scripts>
|
||||
<node-interpreter value="project" />
|
||||
<envs />
|
||||
<method v="2" />
|
||||
</configuration>
|
||||
<configuration name="start" type="js.build_tools.npm" temporary="true" nameIsGenerated="true">
|
||||
<package-json value="$PROJECT_DIR$/package.json" />
|
||||
<command value="run" />
|
||||
<scripts>
|
||||
<script value="start" />
|
||||
</scripts>
|
||||
<node-interpreter value="project" />
|
||||
<envs />
|
||||
<method v="2" />
|
||||
</configuration>
|
||||
<recent_temporary>
|
||||
<list>
|
||||
<item itemvalue="npm.start" />
|
||||
<item itemvalue="npm.server:db:seed" />
|
||||
<item itemvalue="npm.server:db:migrate" />
|
||||
<item itemvalue="npm.server:db:init" />
|
||||
<item itemvalue="npm.postinstall" />
|
||||
</list>
|
||||
</recent_temporary>
|
||||
</component>
|
||||
<component name="SpellCheckerSettings" RuntimeDictionaries="0" Folders="1" Folder0="$USER_HOME$/Documents" CustomDictionaries="1" CustomDictionary0="$USER_HOME$/Documents/fr.dic" DefaultDictionary="application-level" UseSingleDictionary="true" transferred="true" />
|
||||
<component name="TaskManager">
|
||||
<task active="true" id="Default" summary="Default task">
|
||||
<changelist id="1caf0577-5a29-440e-83bc-e061a28fd9bd" name="Default Changelist" comment="" />
|
||||
<created>1611147987113</created>
|
||||
<option name="number" value="Default" />
|
||||
<option name="presentableId" value="Default" />
|
||||
<updated>1611147987113</updated>
|
||||
<workItem from="1611147990942" duration="24000" />
|
||||
<workItem from="1611148027002" duration="59000" />
|
||||
<workItem from="1611148112754" duration="1608000" />
|
||||
<workItem from="1611152887492" duration="1828000" />
|
||||
<workItem from="1616688042421" duration="3772000" />
|
||||
</task>
|
||||
<servers />
|
||||
</component>
|
||||
<component name="TypeScriptGeneratedFilesManager">
|
||||
<option name="version" value="3" />
|
||||
</component>
|
||||
<component name="Vcs.Log.Tabs.Properties">
|
||||
<option name="TAB_STATES">
|
||||
<map>
|
||||
<entry key="MAIN">
|
||||
<value>
|
||||
<State />
|
||||
</value>
|
||||
</entry>
|
||||
</map>
|
||||
</option>
|
||||
<option name="oldMeFiltersMigrated" value="true" />
|
||||
</component>
|
||||
</project>
|
||||
@ -0,0 +1,5 @@
|
||||
<svg viewBox="0 0 24 24">
|
||||
<path d="M11 5C11 4.44772 11.4477 4 12 4C12.5523 4 13 4.44772 13 5V12.1578L16.2428 8.91501L17.657 10.3292L12.0001 15.9861L6.34326 10.3292L7.75748 8.91501L11 12.1575V5Z"
|
||||
fill="currentColor"/>
|
||||
<path d="M4 14H6V18H18V14H20V18C20 19.1046 19.1046 20 18 20H6C4.89543 20 4 19.1046 4 18V14Z" fill="currentColor"/>
|
||||
</svg>
|
||||
|
After Width: | Height: | Size: 357 B |
@ -0,0 +1,157 @@
|
||||
const got = require('got');
|
||||
|
||||
const Errors = {
|
||||
INVALID_IMPORT_FILE: {
|
||||
invalidImport: 'Invalid Import',
|
||||
},
|
||||
};
|
||||
|
||||
module.exports = {
|
||||
exits: {
|
||||
cardNotFound: {
|
||||
responseType: 'notFound',
|
||||
},
|
||||
uploadError: {
|
||||
responseType: 'unprocessableEntity',
|
||||
},
|
||||
},
|
||||
|
||||
async fn(inputs, exits) {
|
||||
const { currentUser } = this.req;
|
||||
|
||||
this.req.file('file').upload(sails.helpers.downloadImportData(), async (error, files) => {
|
||||
try {
|
||||
const data = JSON.parse(files[0].extra);
|
||||
const { project, projectMembership } = await sails.helpers.createProject(
|
||||
currentUser,
|
||||
{ name: data.name },
|
||||
this.req,
|
||||
true,
|
||||
false,
|
||||
);
|
||||
|
||||
// Création du tableau
|
||||
const board = await sails.helpers.createBoard(
|
||||
project,
|
||||
{ name: 'Premier tableau', position: 0, type: 'kanban' },
|
||||
this.req,
|
||||
);
|
||||
|
||||
// Création des labels
|
||||
const labels = new Map();
|
||||
/* eslint-disable no-await-in-loop */
|
||||
// eslint-disable-next-line no-restricted-syntax
|
||||
for (const l of data.labels) {
|
||||
labels.set(
|
||||
l.id,
|
||||
await sails.helpers.createLabel(board, {
|
||||
name: l.name || '?',
|
||||
color: await sails.helpers.getColorFromTrello(l.color),
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
// Création des listes
|
||||
const lists = new Map();
|
||||
// eslint-disable-next-line no-restricted-syntax
|
||||
for (const l of data.lists) {
|
||||
lists.set(
|
||||
l.id,
|
||||
await sails.helpers.createList(board, {
|
||||
name: l.name || '?',
|
||||
position: l.pos,
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
// Création des cartes
|
||||
const cards = new Map();
|
||||
// eslint-disable-next-line no-restricted-syntax
|
||||
for (const c of data.cards) {
|
||||
const card = await sails.helpers.createCard(
|
||||
board,
|
||||
lists.get(c.idList),
|
||||
{
|
||||
position: Math.round(c.pos),
|
||||
name: c.name,
|
||||
description: c.desc || null,
|
||||
},
|
||||
currentUser,
|
||||
this.req,
|
||||
);
|
||||
cards.set(c.id, card);
|
||||
c.idLabels.forEach(async (l) => {
|
||||
await sails.helpers.createCardLabel(card, labels.get(l), this.req);
|
||||
});
|
||||
|
||||
// eslint-disable-next-line no-restricted-syntax
|
||||
for (const a of c.attachments) {
|
||||
if (a.url.startsWith('https://trello-attachments.s3.amazonaws.com/'))
|
||||
got
|
||||
.stream(a.url)
|
||||
.pipe(sails.helpers.importAttachmentReceiver(a, card, currentUser, {}, this.req));
|
||||
}
|
||||
}
|
||||
|
||||
// Création des commentaires
|
||||
const listOfComments = data.actions.filter((a) => a.type === 'commentCard');
|
||||
// eslint-disable-next-line no-restricted-syntax
|
||||
for (const comment of listOfComments) {
|
||||
const userName =
|
||||
data.members.find((u) => u.id === comment.idMemberCreator).fullName || 'Inconnu';
|
||||
await sails.helpers.createAction(
|
||||
cards.get(comment.data.card.id),
|
||||
currentUser,
|
||||
{
|
||||
type: 'commentCard',
|
||||
data: { text: `${userName} - ${comment.data.text}` },
|
||||
},
|
||||
this.req,
|
||||
false,
|
||||
);
|
||||
}
|
||||
|
||||
// eslint-disable-next-line no-restricted-syntax
|
||||
for (const listOfTask of data.checklists) {
|
||||
const card = cards.get(listOfTask.idCard);
|
||||
// eslint-disable-next-line no-restricted-syntax
|
||||
for (const task of listOfTask.checkItems) {
|
||||
await sails.helpers.createTask(
|
||||
card,
|
||||
{
|
||||
name: task.name,
|
||||
isCompleted: task.state === 'complete',
|
||||
},
|
||||
this.req,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
sails.sockets.broadcast(
|
||||
`user:${projectMembership.userId}`,
|
||||
'projectCreate',
|
||||
{
|
||||
item: project,
|
||||
included: {
|
||||
users: [currentUser],
|
||||
projectMemberships: [projectMembership],
|
||||
boards: await sails.helpers.getBoardsForProject(project.id),
|
||||
},
|
||||
},
|
||||
this.req,
|
||||
);
|
||||
|
||||
return exits.success({
|
||||
item: project,
|
||||
included: {
|
||||
users: [currentUser],
|
||||
projectMemberships: [projectMembership],
|
||||
boards: await sails.helpers.getBoardsForProject(project.id),
|
||||
},
|
||||
});
|
||||
} catch (e) {
|
||||
throw Errors.INVALID_IMPORT_FILE;
|
||||
}
|
||||
});
|
||||
},
|
||||
};
|
||||
@ -0,0 +1,33 @@
|
||||
const util = require('util');
|
||||
const stream = require('stream');
|
||||
const streamToArray = require('stream-to-array');
|
||||
|
||||
module.exports = {
|
||||
sync: true,
|
||||
|
||||
fn(inputs, exits) {
|
||||
const receiver = stream.Writable({
|
||||
objectMode: true,
|
||||
});
|
||||
|
||||
let firstFileHandled = false;
|
||||
// eslint-disable-next-line no-underscore-dangle
|
||||
receiver._write = async (file, receiverEncoding, done) => {
|
||||
if (firstFileHandled) {
|
||||
file.pipe(new stream.Writable());
|
||||
|
||||
return done();
|
||||
}
|
||||
firstFileHandled = true;
|
||||
|
||||
const buffer = await streamToArray(file).then((parts) =>
|
||||
Buffer.concat(parts.map((part) => (util.isBuffer(part) ? part : Buffer.from(part)))),
|
||||
);
|
||||
// eslint-disable-next-line no-param-reassign
|
||||
file.extra = buffer.toString('UTF-8');
|
||||
return done();
|
||||
};
|
||||
|
||||
return exits.success(receiver);
|
||||
},
|
||||
};
|
||||
@ -0,0 +1,35 @@
|
||||
module.exports = {
|
||||
inputs: {
|
||||
color: {
|
||||
type: 'string',
|
||||
required: true,
|
||||
},
|
||||
},
|
||||
|
||||
async fn(inputs, exits) {
|
||||
switch (inputs.color) {
|
||||
case 'green':
|
||||
return exits.success('bright-moss');
|
||||
case 'yellow':
|
||||
return exits.success('egg-yellow');
|
||||
case 'orange':
|
||||
return exits.success('pumpkin-orange');
|
||||
case 'red':
|
||||
return exits.success('berry-red');
|
||||
case 'purple':
|
||||
return exits.success('red-burgundy');
|
||||
case 'blue':
|
||||
return exits.success('lagoon-blue');
|
||||
case 'sky':
|
||||
return exits.success('morning-sky');
|
||||
case 'lime':
|
||||
return exits.success('sunny-grass');
|
||||
case 'pink':
|
||||
return exits.success('pink-tulip');
|
||||
case 'black':
|
||||
return exits.success('dark-granite');
|
||||
default:
|
||||
return exits.success('berry-red');
|
||||
}
|
||||
},
|
||||
};
|
||||
@ -0,0 +1,81 @@
|
||||
const fs = require('fs');
|
||||
const path = require('path');
|
||||
const util = require('util');
|
||||
const { v4: uuid } = require('uuid');
|
||||
const sharp = require('sharp');
|
||||
|
||||
const writeFile = util.promisify(fs.writeFile);
|
||||
|
||||
module.exports = {
|
||||
sync: true,
|
||||
inputs: {
|
||||
attachment: {
|
||||
type: 'ref',
|
||||
required: true,
|
||||
},
|
||||
card: {
|
||||
type: 'ref',
|
||||
required: true,
|
||||
},
|
||||
user: {
|
||||
type: 'ref',
|
||||
required: true,
|
||||
},
|
||||
values: {
|
||||
type: 'json',
|
||||
required: true,
|
||||
},
|
||||
request: {
|
||||
type: 'ref',
|
||||
},
|
||||
},
|
||||
|
||||
fn(inputs, exits) {
|
||||
const dirname = uuid();
|
||||
const filename = inputs.attachment.fileName;
|
||||
const rootPath = path.join(sails.config.custom.attachmentsPath, dirname);
|
||||
fs.mkdirSync(rootPath);
|
||||
const writeStream = fs.createWriteStream(path.join(rootPath, filename));
|
||||
writeStream.on('finish', async () => {
|
||||
const image = sharp(fs.readFileSync(path.join(rootPath, filename)));
|
||||
let imageMetadata;
|
||||
|
||||
try {
|
||||
imageMetadata = await image.metadata();
|
||||
} catch (error) {} // eslint-disable-line no-empty
|
||||
|
||||
if (imageMetadata) {
|
||||
let cover256Buffer;
|
||||
if (imageMetadata.height > imageMetadata.width) {
|
||||
cover256Buffer = await image.resize(256, 320).jpeg().toBuffer();
|
||||
} else {
|
||||
cover256Buffer = await image
|
||||
.resize({
|
||||
width: 256,
|
||||
})
|
||||
.jpeg()
|
||||
.toBuffer();
|
||||
}
|
||||
|
||||
const thumbnailsPath = path.join(rootPath, 'thumbnails');
|
||||
fs.mkdirSync(thumbnailsPath);
|
||||
|
||||
await writeFile(path.join(thumbnailsPath, 'cover-256.jpg'), cover256Buffer);
|
||||
|
||||
await sails.helpers.createAttachment(
|
||||
inputs.card,
|
||||
inputs.user,
|
||||
{
|
||||
dirname,
|
||||
filename: inputs.attachment.name,
|
||||
isImage: !!imageMetadata,
|
||||
name: inputs.attachment.name,
|
||||
},
|
||||
null,
|
||||
inputs.request,
|
||||
);
|
||||
}
|
||||
});
|
||||
return exits.success(writeStream);
|
||||
},
|
||||
};
|
||||
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue