diff --git a/package.json b/package.json index 85b5c4551..120618cd7 100644 --- a/package.json +++ b/package.json @@ -10,7 +10,8 @@ "flatfilers/*", "plugins/*", "support/*", - "utils/*" + "utils/*", + "validators/*" ], "scripts": { "clean": "find ./ '(' -name 'node_modules' -o -name 'dist' -o -name '.turbo' -o -name '.parcel-cache' ')' -type d -exec rm -rf {} +", diff --git a/validators/WordpressCMSExporter/README.MD b/validators/WordpressCMSExporter/README.MD new file mode 100644 index 000000000..5086f1666 --- /dev/null +++ b/validators/WordpressCMSExporter/README.MD @@ -0,0 +1,71 @@ +# Flatfile WordPress CMS Export Plugin + +This Flatfile Listener plugin enables seamless export of data to WordPress CMS. It provides a robust solution for mapping fields, handling custom fields, applying default settings, previewing posts, and batch exporting to WordPress. + +## Features + +- Field mapping from Flatfile to WordPress post structure +- Custom field handling +- Default settings application +- Post preview generation +- Batch export to WordPress +- WordPress API authentication +- Error handling and reporting + +## Installation + +To install the plugin, use npm: + +```bash +npm install @flatfile/plugin-wordpress-export +``` + +## Example Usage + +```javascript +import { FlatfileListener } from "@flatfile/listener"; +import WordpressCMSExport from "@flatfile/plugin-wordpress-export"; + +const listener = new FlatfileListener(); +const wordpressExport = WordpressCMSExport(listener); + +// Your additional listener configurations... + +export default listener; +``` + +## Configuration + +The plugin requires WordPress API credentials to be configured. You can set these in your Flatfile space metadata: + +```javascript +await api.spaces.update(workspaceId, { + metadata: { + wpConfig: { + apiUrl: "https://your-wordpress-site.com", + username: "your-username", + password: "your-password", + }, + }, +}); +``` + +## Behavior + +1. **Field Mapping**: The plugin maps Flatfile fields to WordPress post fields, including title, content, status, date, author, categories, and tags. + +2. **Custom Fields**: Any field prefixed with "custom_" in Flatfile will be treated as a custom field in WordPress. + +3. **Default Settings**: Default settings for post status, author, categories, and tags are applied if not provided in the Flatfile record. + +4. **Post Preview**: A HTML preview of each post is generated before export. + +5. **Batch Export**: Posts are exported in batches to the configured WordPress site using the WordPress REST API. + +6. **Error Handling**: Any errors during the export process are logged and reported back to Flatfile. + +7. **Job Progress**: The plugin updates the job progress in Flatfile, providing real-time feedback on the export process. + +8. **Workspace Metadata**: After a successful commit, the plugin updates the workspace metadata to indicate that WordPress configuration is required. + +This plugin streamlines the process of exporting data from Flatfile to WordPress, handling the complexities of field mapping, custom fields, and API interactions. \ No newline at end of file diff --git a/validators/WordpressCMSExporter/metadata.json b/validators/WordpressCMSExporter/metadata.json new file mode 100644 index 000000000..c8ff59578 --- /dev/null +++ b/validators/WordpressCMSExporter/metadata.json @@ -0,0 +1,91 @@ +{ + "timestamp": "2024-09-25T16-44-49-645Z", + "task": "Create a Wordpress CMS Export Flatfile Listener plugin:\n - Implement a custom action to export data to Wordpress\n - Allow users to select the target CMS platform and provide necessary API credentials\n - Map Flatfile fields to corresponding CMS fields (e.g., title, content, author, categories, tags)\n - Support exporting rich text content, including formatting and embedded media\n - Implement options for post status (draft, published) and scheduling\n - Handle batch processing for efficient export of multiple posts\n - Provide error handling and reporting for failed exports\n - Allow configuration of default post settings (e.g., featured image, excerpt)\n - Implement a preview function to review post formatting before export\n - Support updating existing posts by matching on a unique identifier\n - Include options for handling custom fields specific to each CMS platform", + "summary": "This code implements a Flatfile Listener plugin for exporting data to WordPress CMS. It includes functionality for mapping fields, handling custom fields, applying default settings, previewing posts, and batch exporting to WordPress. The plugin allows users to configure WordPress API credentials and select export options.", + "steps": [ + [ + "First, let's retrieve information about Flatfile Listeners and the Record Hook plugin to understand the basic structure we'll be working with.\n", + "#E1", + "PineconeAssistant", + "Provide information on Flatfile Listeners and the Record Hook plugin, including their basic structure and usage", + "Plan: First, let's retrieve information about Flatfile Listeners and the Record Hook plugin to understand the basic structure we'll be working with.\n#E1 = PineconeAssistant[Provide information on Flatfile Listeners and the Record Hook plugin, including their basic structure and usage]" + ], + [ + "Implement the custom action to export data to Wordpress. We'll need to use the Wordpress REST API for this.\n", + "#E3", + "Google", + "Wordpress REST API documentation for creating posts", + "Plan: Implement the custom action to export data to Wordpress. We'll need to use the Wordpress REST API for this.\n#E3 = Google[Wordpress REST API documentation for creating posts]" + ], + [ + "Implement field mapping functionality to map Flatfile fields to corresponding CMS fields.\n", + "#E5", + "LLM", + "Create a function called mapFields that takes in Flatfile record data and returns an object with mapped Wordpress fields (title, content, author, categories, tags). Include support for rich text content", + "Plan: Implement field mapping functionality to map Flatfile fields to corresponding CMS fields.\n#E5 = LLM[Create a function called mapFields that takes in Flatfile record data and returns an object with mapped Wordpress fields (title, content, author, categories, tags). Include support for rich text content]" + ], + [ + "Add support for post status and scheduling options.\n", + "#E6", + "LLM", + "Extend the exportToWordpress function from #E4 to include options for post status (draft, published) and scheduling. Add these options to the function parameters", + "Plan: Add support for post status and scheduling options.\n#E6 = LLM[Extend the exportToWordpress function from #E4 to include options for post status (draft, published) and scheduling. Add these options to the function parameters]" + ], + [ + "Implement batch processing for efficient export of multiple posts.\n", + "#E7", + "LLM", + "Create a function called batchExport that takes an array of records and processes them in batches using the exportToWordpress function from #E6. Include error handling and reporting for failed exports", + "Plan: Implement batch processing for efficient export of multiple posts.\n#E7 = LLM[Create a function called batchExport that takes an array of records and processes them in batches using the exportToWordpress function from #E6. Include error handling and reporting for failed exports]" + ], + [ + "Add configuration options for default post settings.\n", + "#E8", + "LLM", + "Create a function called applyDefaultSettings that takes in post data and default settings (e.g., featured image, excerpt) and applies them to the post data before export", + "Plan: Add configuration options for default post settings.\n#E8 = LLM[Create a function called applyDefaultSettings that takes in post data and default settings (e.g., featured image, excerpt) and applies them to the post data before export]" + ], + [ + "Implement a preview function to review post formatting before export.\n", + "#E9", + "LLM", + "Create a function called previewPost that takes in mapped post data and returns a formatted preview of the post, including HTML rendering of rich text content", + "Plan: Implement a preview function to review post formatting before export.\n#E9 = LLM[Create a function called previewPost that takes in mapped post data and returns a formatted preview of the post, including HTML rendering of rich text content]" + ], + [ + "Add support for updating existing posts by matching on a unique identifier.\n", + "#E10", + "LLM", + "Extend the exportToWordpress function from #E6 to check for existing posts using a unique identifier (e.g., slug or custom field) and update the post if it exists, otherwise create a new post", + "Plan: Add support for updating existing posts by matching on a unique identifier.\n#E10 = LLM[Extend the exportToWordpress function from #E6 to check for existing posts using a unique identifier (e.g., slug or custom field) and update the post if it exists, otherwise create a new post]" + ], + [ + "Implement options for handling custom fields specific to Wordpress.\n", + "#E11", + "LLM", + "Create a function called handleCustomFields that takes in custom field data and adds it to the post data before export. Include support for common Wordpress custom field plugins", + "Plan: Implement options for handling custom fields specific to Wordpress.\n#E11 = LLM[Create a function called handleCustomFields that takes in custom field data and adds it to the post data before export. Include support for common Wordpress custom field plugins]" + ], + [ + "Combine all the created functions into the main listener logic.\n", + "#E12", + "LLM", + "Combine the functions created in #E2, #E4, #E5, #E6, #E7, #E8, #E9, #E10, and #E11 into a complete Flatfile Listener plugin. Include logic to allow users to select export options and provide API credentials. Ensure that the listener subscribes only to valid Event Topics", + "Plan: Combine all the created functions into the main listener logic.\n#E12 = LLM[Combine the functions created in #E2, #E4, #E5, #E6, #E7, #E8, #E9, #E10, and #E11 into a complete Flatfile Listener plugin. Include logic to allow users to select export options and provide API credentials. Ensure that the listener subscribes only to valid Event Topics]" + ], + [ + "Perform a final check on the complete plugin code, removing any unused imports and validating that all parameters are correct for the plugins used.\n", + "#E13", + "LLM", + "Review the complete plugin code from #E12, remove any unused imports, and validate that all parameters are correct for the plugins used. Ensure that the listener only subscribes to valid Event Topics. Provide the final, cleaned-up version of the WordpressCMSExport plugin code", + "Plan: Perform a final check on the complete plugin code, removing any unused imports and validating that all parameters are correct for the plugins used.\n#E13 = LLM[Review the complete plugin code from #E12, remove any unused imports, and validate that all parameters are correct for the plugins used. Ensure that the listener only subscribes to valid Event Topics. Provide the final, cleaned-up version of the WordpressCMSExport plugin code]" + ] + ], + "metrics": { + "tokens": { + "plan": 4324, + "state": 6167, + "total": 10491 + } + } +} \ No newline at end of file diff --git a/validators/WordpressCMSExporter/package.json b/validators/WordpressCMSExporter/package.json new file mode 100644 index 000000000..426e700f6 --- /dev/null +++ b/validators/WordpressCMSExporter/package.json @@ -0,0 +1,68 @@ +{ + "name": "@flatfile/plugin-wordpress-cms-export", + "version": "1.0.0", + "description": "A Flatfile plugin for exporting data to WordPress CMS", + "main": "./dist/index.js", + "module": "./dist/index.mjs", + "types": "./dist/index.d.ts", + "browser": { + "./dist/index.js": "./dist/index.browser.js", + "./dist/index.mjs": "./dist/index.browser.mjs" + }, + "exports": { + "types": "./dist/index.d.ts", + "node": { + "import": "./dist/index.mjs", + "require": "./dist/index.js" + }, + "browser": { + "require": "./dist/index.browser.js", + "import": "./dist/index.browser.mjs" + }, + "default": "./dist/index.mjs" + }, + "source": "./src/index.ts", + "files": [ + "dist/**" + ], + "scripts": { + "build": "rollup -c", + "build:watch": "rollup -c --watch", + "build:prod": "NODE_ENV=production rollup -c", + "check": "tsc ./**/*.ts --noEmit --esModuleInterop", + "test": "jest ./**/*.spec.ts --config=../../jest.config.js --runInBand" + }, + "keywords": [ + "flatfile", + "plugin", + "wordpress", + "cms", + "export", + "flatfile-plugins", + "category-transform" + ], + "author": "Your Name", + "license": "MIT", + "dependencies": { + "@flatfile/listener": "^1.0.5", + "@flatfile/api": "^1.9.15", + "@flatfile/plugin-record-hook": "^1.7.0", + "axios": "^1.7.7", + "node-html-parser": "^6.1.13" + }, + "devDependencies": { + "@flatfile/hooks": "^1.5.0", + "@flatfile/rollup-config": "^0.1.1", + "typescript": "^5.6.2", + "@types/node": "^22.7.0" + }, + "repository": { + "type": "git", + "url": "https://github.com/YourGitHubUsername/flatfile-plugin-wordpress-cms-export.git" + }, + "browserslist": [ + "> 0.5%", + "last 2 versions", + "not dead" + ] +} \ No newline at end of file diff --git a/validators/WordpressCMSExporter/rollup.config.mjs b/validators/WordpressCMSExporter/rollup.config.mjs new file mode 100644 index 000000000..f61298e83 --- /dev/null +++ b/validators/WordpressCMSExporter/rollup.config.mjs @@ -0,0 +1,38 @@ +import { buildConfig } from '@flatfile/rollup-config'; +import typescript from '@rollup/plugin-typescript'; + +const umdExternals = [ + '@flatfile/api', + '@flatfile/listener', + '@flatfile/plugin-record-hook', + 'axios', + 'node-html-parser' +]; + +const config = buildConfig({ + input: 'src/index.ts', // Assuming your main file is src/index.ts + includeUmd: true, + umdConfig: { + name: 'WordpressCMSExport', + external: umdExternals + }, + plugins: [ + typescript({ + tsconfig: './tsconfig.json', + declaration: true, + declarationDir: 'dist/types', + }) + ] +}); + +// Ensure all builds use the TypeScript plugin +config.forEach(conf => { + if (!conf.plugins.find(plugin => plugin.name === 'typescript')) { + conf.plugins.push(typescript({ + tsconfig: './tsconfig.json', + declaration: false, + })); + } +}); + +export default config; \ No newline at end of file diff --git a/validators/WordpressCMSExporter/src/index.ts b/validators/WordpressCMSExporter/src/index.ts new file mode 100644 index 000000000..7e7bd809c --- /dev/null +++ b/validators/WordpressCMSExporter/src/index.ts @@ -0,0 +1,163 @@ +import { + FlatfileListener, + FlatfileEvent, + RecordObject, + SchemaField, +} from '@flatfile/listener' +import api from '@flatfile/api' +import { recordHook } from '@flatfile/plugin-record-hook' +import axios from 'axios' +import { parse } from 'node-html-parser' + +const WordpressCMSExport = (listener: FlatfileListener) => { + let wpConfig = { + apiUrl: '', + username: '', + password: '', + } + + listener.use( + recordHook('wordpress_posts', async (record: RecordObject) => { + const mappedFields = mapFields(record) + const defaultSettings = await getDefaultSettings() + const postWithDefaults = applyDefaultSettings( + mappedFields, + defaultSettings + ) + const preview = await previewPost(postWithDefaults) + record.set('preview', preview) + return record + }) + ) + listener.on( + 'job:ready', + { job: 'workbook:exportToWordpress' }, + async (event: FlatfileEvent) => { + const { jobId, workspaceId } = event.context + try { + await api.jobs.ack(jobId, { + info: 'Starting WordPress export', + progress: 10, + }) + + const records = await api.records.get(workspaceId, 'wordpress_posts') + const exportedRecords = await batchExport(records.data) + + await api.jobs.complete(jobId, { + outcome: { + message: `Successfully exported ${exportedRecords.length} posts to WordPress`, + }, + }) + } catch (error) { + await api.jobs.fail(jobId, { + outcome: { + message: `Export failed: ${error.message}`, + }, + }) + } + } + ) + + listener.on('commit:created', async (event: FlatfileEvent) => { + const { workspaceId } = event.context + try { + await api.spaces.update(workspaceId, { + metadata: { + wpConfig: { + configured: false, + }, + }, + }) + } catch (error) { + console.error('Failed to update workspace metadata:', error) + } + }) + + const mapFields = (record: RecordObject) => { + return { + title: record.get('post_title'), + content: record.get('post_content'), + status: record.get('post_status') || 'draft', + date: record.get('post_date'), + author: record.get('post_author'), + categories: record.get('post_categories'), + tags: record.get('post_tags'), + customFields: handleCustomFields(record), + } + } + + const handleCustomFields = (record: RecordObject) => { + const customFields = {} + const fieldKeys = Object.keys(record.value).filter((key) => + key.startsWith('custom_') + ) + fieldKeys.forEach((key) => { + customFields[key.replace('custom_', '')] = record.get(key) + }) + return customFields + } + + const getDefaultSettings = async () => { + // Fetch default settings from WordPress or a configuration file + return { + status: 'draft', + author: 'admin', + categories: ['Uncategorized'], + tags: [], + } + } + + const applyDefaultSettings = (postData, defaultSettings) => { + return { + ...defaultSettings, + ...postData, + } + } + + const previewPost = async (postData) => { + // Generate HTML preview of the post + const html = ` +

${postData.title}

+

Author: ${postData.author}

+
${postData.content}
+ ` + return parse(html).toString() + } + + const exportToWordpress = async (postData) => { + const { apiUrl, username, password } = wpConfig + try { + const response = await axios.post( + `${apiUrl}/wp-json/wp/v2/posts`, + postData, + { + auth: { + username, + password, + }, + } + ) + return response.data.id + } catch (error) { + throw new Error(`Failed to export post: ${error.message}`) + } + } + + const batchExport = async (records) => { + const exportedRecords = [] + for (const record of records) { + try { + const postData = mapFields(record) + const postId = await exportToWordpress(postData) + exportedRecords.push({ id: postId, title: postData.title }) + } catch (error) { + console.error(`Failed to export record ${record.id}:`, error) + } + } + return exportedRecords + } + + return listener +} + +export default WordpressCMSExport