1
0
Fork 0
mirror of https://github.com/plankanban/planka.git synced 2025-07-18 20:59:44 +02:00

feat: add routes for creating, cycling and deleting apiKeys

This commit is contained in:
Samuel 2025-07-13 01:56:44 +02:00
parent d6cbb889fb
commit 846b0579b3
7 changed files with 243 additions and 0 deletions

View file

@ -0,0 +1,51 @@
/*!
* Copyright (c) 2025 PLANKA Software GmbH
* Licensed under the Fair Use License: https://github.com/plankanban/planka/blob/master/LICENSE.md
*/
const { idInput } = require('../../../utils/inputs');
const Errors = {
USER_NOT_FOUND: {
userNotFound: 'User not found',
},
ALREADY_EXISTS: {
alreadyExists: 'API key already exists',
},
};
module.exports = {
inputs: {
id: {
...idInput,
required: true,
},
},
exits: {
userNotFound: {
responseType: 'notFound',
},
alreadyExists: {
responseType: 'conflict',
},
},
async fn(inputs) {
const { id } = inputs;
const { apiKey } = await sails.helpers.apiKey.createAndStore
.with({
id,
cycle: false,
})
.intercept('alreadyExists', () => Errors.ALREADY_EXISTS)
.intercept('userNotFound', () => Errors.USER_NOT_FOUND);
return {
item: {
apiKey,
},
};
},
};

View file

@ -0,0 +1,51 @@
/*!
* Copyright (c) 2025 PLANKA Software GmbH
* Licensed under the Fair Use License: https://github.com/plankanban/planka/blob/master/LICENSE.md
*/
const { idInput } = require('../../../utils/inputs');
const Errors = {
USER_NOT_FOUND: {
userNotFound: 'User not found',
},
DOES_NOT_EXIST: {
doesNotExist: 'User does not have an API key',
},
};
module.exports = {
inputs: {
id: {
...idInput,
required: true,
},
},
exits: {
userNotFound: {
responseType: 'notFound',
},
doesNotExist: {
responseType: 'notFound',
},
},
async fn(inputs) {
const { id } = inputs;
const { apiKey } = await sails.helpers.apiKey.createAndStore
.with({
id,
cycle: true,
})
.intercept('doesNotExist', () => Errors.DOES_NOT_EXIST)
.intercept('userNotFound', () => Errors.USER_NOT_FOUND);
return {
item: {
apiKey,
},
};
},
};

View file

@ -0,0 +1,50 @@
/*!
* Copyright (c) 2025 PLANKA Software GmbH
* Licensed under the Fair Use License: https://github.com/plankanban/planka/blob/master/LICENSE.md
*/
const { idInput } = require('../../../utils/inputs');
const Errors = {
USER_NOT_FOUND: {
userNotFound: 'User not found',
},
DOES_NOT_EXIST: {
doesNotExist: 'User does not have an API key',
},
};
module.exports = {
inputs: {
id: {
...idInput,
required: true,
},
},
exits: {
userNotFound: {
responseType: 'notFound',
},
doesNotExist: {
responseType: 'notFound',
},
},
async fn(inputs) {
const { id } = inputs;
await sails.helpers.apiKey.deleteOne
.with({
id,
})
.intercept('doesNotExist', () => Errors.DOES_NOT_EXIST)
.intercept('userNotFound', () => Errors.USER_NOT_FOUND);
return {
item: {
success: true,
},
};
},
};

View file

@ -0,0 +1,48 @@
const bcrypt = require('bcrypt');
const { v4: uuidv4 } = require('uuid');
const crypto = require('crypto');
const { idInput } = require('../../../utils/inputs');
module.exports = {
inputs: {
id: {
...idInput,
required: true,
},
cycle: { type: 'boolean', defaultsTo: false },
},
exits: {
userNotFound: {},
alreadyExists: {},
doesNotExist: {},
},
async fn(inputs) {
const { id, cycle } = inputs;
const user = await User.findOne({ id });
if (!user) throw 'userNotFound';
if (user.apiKeyHash && !cycle) throw 'alreadyExists';
if (!user.apiKeyHash && cycle) throw 'doesNotExist';
const prefix = `${Number(id).toString(36).padStart(8, '0')}${crypto
.randomBytes(4)
.toString('hex')}`;
const rawKey = `${prefix}.${uuidv4().replace(
/-/g,
'',
)}${crypto.randomBytes(16).toString('hex')}`;
const hash = await bcrypt.hash(rawKey, 12);
await User.updateOne({ id }).set({
apiKeyPrefix: prefix,
apiKeyHash: hash,
});
return { apiKey: rawKey };
},
};

View file

@ -0,0 +1,35 @@
/*!
* Copyright (c) 2025 PLANKA Software GmbH
* Licensed under the Fair Use License: https://github.com/plankanban/planka/blob/master/LICENSE.md
*/
const { idInput } = require('../../../utils/inputs');
module.exports = {
inputs: {
id: {
...idInput,
required: true,
},
},
exits: {
userNotFound: {},
doesNotExist: {},
},
async fn(inputs) {
const { id } = inputs;
const user = await User.findOne({ id });
if (!user) throw 'userNotFound';
if (!user.apiKeyHash) throw 'doesNotExist';
await User.updateOne({ id }).set({
apiKeyPrefix: null,
apiKeyHash: null,
});
return { success: true };
},
};

View file

@ -33,6 +33,10 @@ module.exports.policies = {
'users/update-avatar': 'is-authenticated', 'users/update-avatar': 'is-authenticated',
'users/delete': ['is-authenticated', 'is-admin'], 'users/delete': ['is-authenticated', 'is-admin'],
'api-keys/create': ['is-authenticated', 'is-admin'],
'api-keys/cycle': ['is-authenticated', 'is-admin'],
'api-keys/delete': ['is-authenticated', 'is-admin'],
'projects/create': ['is-authenticated', 'is-external', 'is-admin-or-project-owner'], 'projects/create': ['is-authenticated', 'is-external', 'is-admin-or-project-owner'],
'config/show': true, 'config/show': true,

View file

@ -83,6 +83,10 @@ module.exports.routes = {
'POST /api/users/:id/avatar': 'users/update-avatar', 'POST /api/users/:id/avatar': 'users/update-avatar',
'DELETE /api/users/:id': 'users/delete', 'DELETE /api/users/:id': 'users/delete',
'POST /api/users/:id/api-key': 'api-key/create',
'PATCH /api/users/:id/api-key': 'api-key/cycle',
'DELETE /api/users/:id/api-key': 'api-key/delete',
'GET /api/projects': 'projects/index', 'GET /api/projects': 'projects/index',
'POST /api/projects': 'projects/create', 'POST /api/projects': 'projects/create',
'GET /api/projects/:id': 'projects/show', 'GET /api/projects/:id': 'projects/show',