-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(strapi-admin-extensions): enable to import VAC as CSV
- Loading branch information
1 parent
434a366
commit 5959439
Showing
26 changed files
with
826 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,53 @@ | ||
# Strapi Admin Extensions | ||
|
||
This project contains custom extensions for the Strapi admin panel. It depends on another app called Strapi dashboard. | ||
|
||
## Prerequisites | ||
|
||
- Ensure `pdc-dashboard` is installed and set up properly before using `strapi-admin-extensions`. | ||
|
||
## Installation | ||
|
||
1. **Clone the repository:** | ||
|
||
```bash | ||
git clone git@github.com:frameless/strapi.git | ||
``` | ||
|
||
2. **Install dependencies:** | ||
Make sure you are in the project root: | ||
|
||
```bash | ||
yarn install | ||
``` | ||
|
||
## Usage | ||
|
||
1. Ensure the `pdc-dashboard` app is running: | ||
|
||
```bash | ||
yarn workspace @frameless/pdc-dashboard dev | ||
``` | ||
|
||
2. Copy the environment configuration file to the `strapi-admin-extensions` folder: | ||
|
||
```bash | ||
cp .env.example .env | ||
``` | ||
|
||
3. Run the development server for `strapi-admin-extensions`: | ||
|
||
```bash | ||
yarn workspace @frameless/strapi-admin-extensions dev | ||
``` | ||
|
||
## Contributing | ||
|
||
We welcome contributions! Feel free to: | ||
|
||
- Open an issue to report bugs or suggest new features. | ||
- Submit a pull request with improvements or fixes. | ||
|
||
## License | ||
|
||
This project is licensed under the EUPL-1.2 License. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
import type { Config } from 'jest'; | ||
|
||
const config: Config = { | ||
preset: 'ts-jest', | ||
// to obtain access to the matchers. | ||
moduleFileExtensions: ['ts', 'tsx', 'js', 'json', 'node'], | ||
setupFilesAfterEnv: ['<rootDir>/src/tests/jest.setup.ts'], | ||
modulePaths: ['<rootDir>'], | ||
testEnvironment: 'node', | ||
roots: ['<rootDir>/src'], | ||
transform: { | ||
'^.+\\.(ts)$': [ | ||
'ts-jest', | ||
{ | ||
tsconfig: 'tsconfig.test.json', | ||
}, | ||
], | ||
}, | ||
}; | ||
|
||
export default config; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
{ | ||
"name": "@frameless/strapi-admin-extensions", | ||
"version": "0.0.0", | ||
"private": true, | ||
"author": "@frameless", | ||
"description": "Strapi Admin Extensions", | ||
"license": "EUPL-1.2", | ||
"keywords": [], | ||
"scripts": { | ||
"prebuild": "yarn clean", | ||
"build": "npm-run-all --parallel build:*", | ||
"build:server": "tsc -p ./tsconfig.json", | ||
"watch": "tsc -p ./tsconfig.json -w", | ||
"start": "NODE_ENV=production node ./dist/src/server.js", | ||
"dev": "NODE_ENV=development nodemon src/server.ts", | ||
"clean": "rimraf dist src/types tmp", | ||
"test": "OVERIGE_OBJECTEN_API_PORT=3000 jest --coverage --forceExit --verbose", | ||
"test:watch": "OVERIGE_OBJECTEN_API_PORT=3000 jest --watch" | ||
}, | ||
"dependencies": { | ||
"cors": "2.8.5", | ||
"csv-parser": "3.0.0", | ||
"dompurify": "3.2.1", | ||
"dotenv": "16.4.5", | ||
"express": "4.21.0", | ||
"lodash.memoize": "4.1.2", | ||
"lodash.merge": "4.6.2", | ||
"lodash.snakecase": "4.1.1", | ||
"p-limit": "3.0.0", | ||
"morgan": "1.10.0" | ||
}, | ||
"devDependencies": { | ||
"@types/cors": "2.8.17", | ||
"@types/dompurify": "3.2.0", | ||
"@types/jest": "29.5.12", | ||
"@types/lodash.memoize": "4.1.9", | ||
"@types/lodash.merge": "4.6.9", | ||
"@types/lodash.snakecase": "4.1.9", | ||
"@types/supertest": "6.0.2", | ||
"jest": "29.7.0", | ||
"jest-fetch-mock": "3.0.3", | ||
"nodemon": "3.1.7", | ||
"rimraf": "6.0.1", | ||
"supertest": "7.0.0", | ||
"ts-jest": "29.2.3", | ||
"ts-node": "10.9.2", | ||
"typescript": "5.0.4", | ||
"@types/morgan": "1.9.9" | ||
}, | ||
"repository": { | ||
"type": "git+ssh", | ||
"url": "git@github.com:frameless/strapi.git", | ||
"directory": "apps/strapi-admin-extensio0s" | ||
} | ||
} |
64 changes: 64 additions & 0 deletions
64
apps/strapi-admin-extensions/src/controllers/import/index.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,64 @@ | ||
import type { NextFunction, Request, Response } from 'express'; | ||
import fs from 'node:fs'; | ||
import pLimit from 'p-limit'; | ||
import { CREATE_VAC } from '../../queries'; | ||
// import { CreateVacResponse } from '../../strapi-product-type'; | ||
import { fetchData, processCsvFile } from '../../utils'; | ||
|
||
const limit = pLimit(5); // Limit the number of concurrent file uploads | ||
export const importController = async (req: Request, res: Response, next: NextFunction) => { | ||
const type = req.body.type; | ||
if (req.file) { | ||
const filePath = req.file.path; | ||
const requiredColumns = ['vraag', 'antwoord']; | ||
|
||
try { | ||
// Process the CSV file and sanitize results | ||
const authorizationHeader = req.headers?.authorization || ''; | ||
const [authType, authToken] = authorizationHeader.split(/\s+/); | ||
const tokenAuth = authType === 'Token' ? authToken : authorizationHeader; | ||
const graphqlURL = new URL('/graphql', process.env.STRAPI_PRIVATE_URL); | ||
const sanitizedResults = await processCsvFile(filePath, requiredColumns); | ||
const locale = req.query?.locale || 'nl'; | ||
|
||
if (type === 'vac') { | ||
// Loop through the sanitized results and create entries one by one | ||
const results = await Promise.all( | ||
sanitizedResults.map((entry) => | ||
limit(async () => { | ||
try { | ||
const { data: responseData } = await fetchData<any>({ | ||
url: graphqlURL.href, | ||
query: CREATE_VAC, | ||
variables: { locale, data: entry }, | ||
headers: { | ||
Authorization: `Bearer ${tokenAuth}`, | ||
}, | ||
}); | ||
return responseData; | ||
} catch (error: any) { | ||
next(error); | ||
// eslint-disable-next-line no-console | ||
console.error('Error processing entry:', error); | ||
return { error: error.message, entry }; | ||
} | ||
}), | ||
), | ||
); | ||
res.json({ message: 'CSV converted to JSON', data: results }); | ||
// Delete temporary file after processing | ||
await fs.promises.unlink(filePath); | ||
} else { | ||
res.status(400).send('Invalid import type.'); | ||
} | ||
} catch (error) { | ||
await fs.promises.unlink(filePath); // Delete the temporary file in case of error | ||
// Forward any errors to the error handler middleware | ||
next(error); | ||
return null; | ||
} | ||
} else { | ||
res.status(400).send('No file uploaded.'); | ||
} | ||
return null; | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export { importController } from './import'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
const gql = (query: any) => query; | ||
|
||
export const CREATE_VAC = gql(` | ||
mutation createVac($data: VacInput!) { | ||
createVac(data: $data){ | ||
data { | ||
id | ||
attributes { | ||
createdAt | ||
publishedAt | ||
vac { | ||
id | ||
vraag | ||
antwoord(pagination: { start: 0, limit: -1 }) { | ||
content | ||
kennisartikelCategorie | ||
} | ||
status | ||
doelgroep | ||
uuid | ||
toelichting | ||
afdelingen { | ||
afdelingId | ||
afdelingNaam | ||
} | ||
trefwoorden { | ||
id | ||
trefwoord | ||
} | ||
} | ||
} | ||
} | ||
} | ||
} | ||
`); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
import express from 'express'; | ||
import multer from 'multer'; | ||
import { importController } from '../../controllers'; | ||
const upload = multer({ dest: 'tmp/uploads/' }); | ||
const router = express.Router({ mergeParams: true }); | ||
|
||
router.post('/import', upload.single('file'), importController); | ||
export default router; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
import importRoute from './import'; | ||
export { importRoute }; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,82 @@ | ||
import type { CorsOptions } from 'cors'; | ||
import cors from 'cors'; | ||
import { config } from 'dotenv'; | ||
import express from 'express'; | ||
import { NextFunction, Request, Response } from 'express'; | ||
import morgan from 'morgan'; | ||
import { importRoute } from './routers'; | ||
import { envAvailability, ErrorHandler } from './utils'; | ||
config(); | ||
|
||
// Validate environment variables | ||
envAvailability({ | ||
env: process.env, | ||
keys: ['STRAPI_PRIVATE_URL', 'STRAPI_ADMIN_EXTENSIONS_PORT'], | ||
}); | ||
|
||
const whitelist = process.env.STRAPI_ADMIN_EXTENSIONS_CORS?.split(', ') || []; | ||
const corsOption: CorsOptions = { | ||
origin: (origin, callback) => { | ||
if (!origin || whitelist.indexOf(origin) !== -1) { | ||
callback(null, true); | ||
} else { | ||
callback( | ||
new ErrorHandler('Not allowed by CORS', { | ||
statusCode: 403, | ||
}), | ||
); | ||
} | ||
}, | ||
optionsSuccessStatus: 200, | ||
}; | ||
const app = express(); | ||
// Multer file upload middleware. | ||
// The order is important, so this should be before the express.json() middleware to parse the file. | ||
app.use('/api/v2', importRoute); | ||
// parse application/json | ||
app.use(express.json()); | ||
// parse application/x-www-form-urlencoded | ||
app.use(express.urlencoded({ extended: true })); | ||
// log HTTP requests | ||
app.use(morgan('dev')); | ||
|
||
const port = process.env.STRAPI_ADMIN_EXTENSIONS_PORT; | ||
// Centralized error handler middleware | ||
const globalErrorHandler = (err: ErrorHandler, _req: Request, res: Response, _next: NextFunction) => { | ||
if (err instanceof ErrorHandler || (err as ErrorHandler)?.isOperational) { | ||
// Send the proper error response with status code and message | ||
return res.status(err?.options?.statusCode || 500).json({ | ||
message: err.message, | ||
}); | ||
} | ||
|
||
// If it's an unknown error (not an operational error), log it and send a generic response | ||
// eslint-disable-next-line no-console | ||
console.error('Unexpected error:', err); | ||
return res.status(500).json({ | ||
message: 'An unexpected error occurred.', | ||
}); | ||
}; | ||
|
||
/** | ||
* CORS | ||
* Enable CORS with a whitelist of allowed origins | ||
*/ | ||
app.use(cors(corsOption)); | ||
// handle non existing routes | ||
app.use((_req, res) => { | ||
res.status(404).send('Route not found'); | ||
}); | ||
// Use global error handler middleware | ||
app.use(globalErrorHandler); | ||
/** | ||
* Start the server | ||
*/ | ||
if (process.env.NODE_ENV !== 'test') { | ||
app.listen(port, () => { | ||
// eslint-disable-next-line no-console | ||
console.log(`Overige Objecten app listening on port ${port}!`); | ||
}); | ||
} | ||
|
||
export default app; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
import fetchMock from 'jest-fetch-mock'; | ||
|
||
fetchMock.enableMocks(); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
interface EnvValidator { | ||
env: any; | ||
keys: string[]; | ||
} | ||
export const envAvailability = ({ env, keys }: EnvValidator) => { | ||
keys?.forEach((key: string) => { | ||
if (!env[key]) { | ||
throw new Error(`Missing required environment variable: ${key}`); | ||
} | ||
}); | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
export type Options = { | ||
statusCode: number; | ||
}; | ||
export class ErrorHandler extends Error { | ||
isOperational: boolean; // this flag for custom error identification | ||
|
||
constructor( | ||
message?: string, | ||
public options?: Options, | ||
) { | ||
super(message); | ||
this.name = 'ErrorHandler'; | ||
this.options = options; | ||
this.isOperational = true; // Operational errors should be marked | ||
} | ||
} |
Oops, something went wrong.