๐Ÿ“… let's chat! explore the endless possibilities creating industries that don't exist. click here

evernote-data-handling

Best practices for handling Evernote data. Use when implementing data storage, processing notes, handling attachments, or ensuring data integrity. Trigger with phrases like "evernote data", "handle evernote notes", "evernote storage", "process evernote content". allowed-tools: Read, Write, Edit, Grep version: 1.0.0 license: MIT author: Jeremy Longshore <jeremy@intentsolutions.io>

Allowed Tools

No tools specified

Provided by Plugin

evernote-pack

Claude Code skill pack for Evernote (24 skills)

saas packs v1.0.0
View Plugin

Installation

This skill is included in the evernote-pack plugin:

/plugin install evernote-pack@claude-code-plugins-plus

Click to copy

Instructions

# Evernote Data Handling ## Overview Best practices for handling Evernote data including notes, attachments, ENML content, and synchronization data. ## Prerequisites - Understanding of Evernote data model - Database for local storage - File storage for attachments ## Evernote Data Model ``` User โ”œโ”€โ”€ Notebooks (max 250-1000) โ”‚ โ””โ”€โ”€ Notes (max 100,000) โ”‚ โ”œโ”€โ”€ Content (ENML format) โ”‚ โ”œโ”€โ”€ Resources (attachments) โ”‚ โ”œโ”€โ”€ Tags (many-to-many) โ”‚ โ””โ”€โ”€ Attributes (metadata) โ”œโ”€โ”€ Tags (max 100,000) โ””โ”€โ”€ Saved Searches (max 100) ``` ## Instructions ### Step 1: Data Schema Design ```sql -- PostgreSQL schema for Evernote data -- Users CREATE TABLE evernote_users ( id SERIAL PRIMARY KEY, evernote_user_id BIGINT UNIQUE NOT NULL, username VARCHAR(255), email VARCHAR(255), privilege INTEGER, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ); -- Notebooks CREATE TABLE notebooks ( guid UUID PRIMARY KEY, user_id INTEGER REFERENCES evernote_users(id), name VARCHAR(255) NOT NULL, stack VARCHAR(255), default_notebook BOOLEAN DEFAULT FALSE, created_at BIGINT, updated_at BIGINT, synced_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, INDEX idx_notebooks_user (user_id), INDEX idx_notebooks_stack (stack) ); -- Tags CREATE TABLE tags ( guid UUID PRIMARY KEY, user_id INTEGER REFERENCES evernote_users(id), name VARCHAR(255) NOT NULL, parent_guid UUID REFERENCES tags(guid), update_sequence_num INTEGER, synced_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, INDEX idx_tags_user (user_id), INDEX idx_tags_parent (parent_guid), UNIQUE(user_id, name) ); -- Notes CREATE TABLE notes ( guid UUID PRIMARY KEY, user_id INTEGER REFERENCES evernote_users(id), notebook_guid UUID REFERENCES notebooks(guid), title VARCHAR(255) NOT NULL, content TEXT, -- ENML content content_hash BYTEA, content_length INTEGER, created_at BIGINT, updated_at BIGINT, deleted_at BIGINT, active BOOLEAN DEFAULT TRUE, update_sequence_num INTEGER, synced_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, INDEX idx_notes_user (user_id), INDEX idx_notes_notebook (notebook_guid), INDEX idx_notes_updated (updated_at), INDEX idx_notes_active (active) ); -- Note-Tag relationship CREATE TABLE note_tags ( note_guid UUID REFERENCES notes(guid), tag_guid UUID REFERENCES tags(guid), PRIMARY KEY (note_guid, tag_guid) ); -- Resources (attachments) CREATE TABLE resources ( guid UUID PRIMARY KEY, note_guid UUID REFERENCES notes(guid), mime_type VARCHAR(100) NOT NULL, width INTEGER, height INTEGER, duration INTEGER, data_hash BYTEA NOT NULL, data_size INTEGER NOT NULL, file_name VARCHAR(255), source_url TEXT, storage_path VARCHAR(500), -- Path to stored file synced_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, INDEX idx_resources_note (note_guid), INDEX idx_resources_hash (data_hash) ); -- Sync state CREATE TABLE sync_state ( user_id INTEGER PRIMARY KEY REFERENCES evernote_users(id), last_update_count INTEGER DEFAULT 0, last_sync_time BIGINT, full_sync_required BOOLEAN DEFAULT TRUE ); ``` ### Step 2: ENML Content Processing ```javascript // services/enml-processor.js const cheerio = require('cheerio'); const crypto = require('crypto'); class ENMLProcessor { /** * Parse ENML content */ parse(enml) { const $ = cheerio.load(enml, { xmlMode: true, decodeEntities: true }); return { $, title: $('title').text(), text: this.extractText($), todos: this.extractTodos($), links: this.extractLinks($), media: this.extractMedia($), checksum: this.computeChecksum(enml) }; } /** * Extract plain text from ENML */ extractText($) { // Remove en-media and en-crypt elements $('en-media, en-crypt').remove(); // Get text content return $('en-note') .text() .replace(/\s+/g, ' ') .trim(); } /** * Extract todos */ extractTodos($) { const todos = []; $('en-todo').each((i, el) => { const checked = $(el).attr('checked') === 'true'; // Get text after the todo element const text = $(el).next().text().trim() || $(el).parent().text().trim(); todos.push({ index: i, checked, text: text.slice(0, 200) // Limit length }); }); return todos; } /** * Extract links */ extractLinks($) { const links = []; $('a[href]').each((i, el) => { const href = $(el).attr('href'); const text = $(el).text(); // Skip javascript: and mailto: links if (href && !href.startsWith('javascript:') && !href.startsWith('mailto:')) { links.push({ href, text }); } }); return links; } /** * Extract media references */ extractMedia($) { const media = []; $('en-media').each((i, el) => { media.push({ type: $(el).attr('type'), hash: $(el).attr('hash'), width: parseInt($(el).attr('width')) || null, height: parseInt($(el).attr('height')) || null }); }); return media; } /** * Compute content checksum */ computeChecksum(content) { return crypto.createHash('md5').update(content).digest('hex'); } /** * Validate ENML structure */ validate(enml) { const errors = []; if (!enml.includes('') || !enml.includes('')) { errors.push('Missing en-note root element'); } return { valid: errors.length === 0, errors }; } /** * Sanitize content for storage */ sanitize(enml) { const $ = cheerio.load(enml, { xmlMode: true }); // Remove potentially sensitive data $('en-crypt').each((i, el) => { $(el).replaceWith('[Encrypted content]'); }); return $.html(); } } module.exports = ENMLProcessor; ``` ### Step 3: Resource (Attachment) Handling ```javascript // services/resource-handler.js const fs = require('fs').promises; const path = require('path'); const crypto = require('crypto'); class ResourceHandler { constructor(options) { this.storageRoot = options.storageRoot || './storage/resources'; this.maxSizeBytes = options.maxSizeBytes || 100 * 1024 * 1024; // 100MB } /** * Store resource from Evernote */ async storeResource(resource, userId) { const hash = Buffer.from(resource.data.bodyHash).toString('hex'); const storagePath = this.getStoragePath(userId, hash, resource.mime); // Ensure directory exists await fs.mkdir(path.dirname(storagePath), { recursive: true }); // Write file await fs.writeFile(storagePath, resource.data.body); return { storagePath, hash, size: resource.data.size, mime: resource.mime }; } /** * Get storage path for resource */ getStoragePath(userId, hash, mimeType) { const ext = this.getExtension(mimeType); // Shard by first 2 chars of hash for performance const shard = hash.substring(0, 2); return path.join(this.storageRoot, userId.toString(), shard, `${hash}${ext}`); } /** * Get file extension from MIME type */ getExtension(mimeType) { const map = { 'image/png': '.png', 'image/jpeg': '.jpg', 'image/gif': '.gif', 'application/pdf': '.pdf', 'audio/wav': '.wav', 'audio/mpeg': '.mp3' }; return map[mimeType] || ''; } /** * Retrieve stored resource */ async getResource(storagePath) { try { const data = await fs.readFile(storagePath); return { data, exists: true }; } catch (error) { if (error.code === 'ENOENT') { return { data: null, exists: false }; } throw error; } } /** * Delete resource */ async deleteResource(storagePath) { try { await fs.unlink(storagePath); return { deleted: true }; } catch (error) { if (error.code === 'ENOENT') { return { deleted: false, reason: 'not_found' }; } throw error; } } /** * Calculate storage usage for user */ async getUserStorageUsage(userId) { const userDir = path.join(this.storageRoot, userId.toString()); let totalSize = 0; let fileCount = 0; async function walkDir(dir) { try { const entries = await fs.readdir(dir, { withFileTypes: true }); for (const entry of entries) { const fullPath = path.join(dir, entry.name); if (entry.isDirectory()) { await walkDir(fullPath); } else { const stat = await fs.stat(fullPath); totalSize += stat.size; fileCount++; } } } catch (error) { if (error.code !== 'ENOENT') throw error; } } await walkDir(userDir); return { totalBytes: totalSize, totalMB: (totalSize / (1024 * 1024)).toFixed(2), fileCount }; } } module.exports = ResourceHandler; ``` ### Step 4: Sync Data Manager ```javascript // services/sync-manager.js class SyncManager { constructor(db, noteStore) { this.db = db; this.noteStore = noteStore; } /** * Process sync chunk and update local data */ async processSyncChunk(userId, chunk) { const results = { notes: { created: 0, updated: 0, deleted: 0 }, notebooks: { created: 0, updated: 0, deleted: 0 }, tags: { created: 0, updated: 0, deleted: 0 } }; // Process notebooks if (chunk.notebooks) { for (const notebook of chunk.notebooks) { await this.upsertNotebook(userId, notebook); results.notebooks.updated++; } } // Process expunged notebooks if (chunk.expungedNotebooks) { for (const guid of chunk.expungedNotebooks) { await this.deleteNotebook(guid); results.notebooks.deleted++; } } // Process tags if (chunk.tags) { for (const tag of chunk.tags) { await this.upsertTag(userId, tag); results.tags.updated++; } } // Process expunged tags if (chunk.expungedTags) { for (const guid of chunk.expungedTags) { await this.deleteTag(guid); results.tags.deleted++; } } // Process notes if (chunk.notes) { for (const note of chunk.notes) { const isNew = await this.upsertNote(userId, note); if (isNew) results.notes.created++; else results.notes.updated++; } } // Process expunged notes if (chunk.expungedNotes) { for (const guid of chunk.expungedNotes) { await this.deleteNote(guid); results.notes.deleted++; } } return results; } /** * Upsert notebook */ async upsertNotebook(userId, notebook) { return this.db.query(` INSERT INTO notebooks (guid, user_id, name, stack, default_notebook, created_at, updated_at, synced_at) VALUES ($1, $2, $3, $4, $5, $6, $7, NOW()) ON CONFLICT (guid) DO UPDATE SET name = EXCLUDED.name, stack = EXCLUDED.stack, default_notebook = EXCLUDED.default_notebook, updated_at = EXCLUDED.updated_at, synced_at = NOW() `, [notebook.guid, userId, notebook.name, notebook.stack, notebook.defaultNotebook, notebook.serviceCreated, notebook.serviceUpdated]); } /** * Upsert note (without content) */ async upsertNote(userId, note) { const existing = await this.db.query( 'SELECT guid FROM notes WHERE guid = $1', [note.guid] ); await this.db.query(` INSERT INTO notes (guid, user_id, notebook_guid, title, content_hash, content_length, created_at, updated_at, active, update_sequence_num, synced_at) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, NOW()) ON CONFLICT (guid) DO UPDATE SET notebook_guid = EXCLUDED.notebook_guid, title = EXCLUDED.title, content_hash = EXCLUDED.content_hash, content_length = EXCLUDED.content_length, updated_at = EXCLUDED.updated_at, active = EXCLUDED.active, update_sequence_num = EXCLUDED.update_sequence_num, synced_at = NOW() `, [note.guid, userId, note.notebookGuid, note.title, note.contentHash, note.contentLength, note.created, note.updated, note.active !== false, note.updateSequenceNum]); // Update tags if (note.tagGuids) { await this.updateNoteTags(note.guid, note.tagGuids); } return existing.rows.length === 0; // Return true if new } /** * Update note tags */ async updateNoteTags(noteGuid, tagGuids) { await this.db.query('DELETE FROM note_tags WHERE note_guid = $1', [noteGuid]); if (tagGuids.length > 0) { const values = tagGuids.map((tagGuid, i) => `($1, $${i + 2})` ).join(', '); await this.db.query( `INSERT INTO note_tags (note_guid, tag_guid) VALUES ${values}`, [noteGuid, ...tagGuids] ); } } /** * Delete note (soft delete) */ async deleteNote(guid) { return this.db.query( 'UPDATE notes SET active = FALSE, deleted_at = $1, synced_at = NOW() WHERE guid = $2', [Date.now(), guid] ); } /** * Update sync state */ async updateSyncState(userId, updateCount) { return this.db.query(` INSERT INTO sync_state (user_id, last_update_count, last_sync_time, full_sync_required) VALUES ($1, $2, $3, FALSE) ON CONFLICT (user_id) DO UPDATE SET last_update_count = EXCLUDED.last_update_count, last_sync_time = EXCLUDED.last_sync_time, full_sync_required = FALSE `, [userId, updateCount, Date.now()]); } } module.exports = SyncManager; ``` ### Step 5: Data Export ```javascript // services/data-export.js const archiver = require('archiver'); class DataExporter { constructor(db, resourceHandler) { this.db = db; this.resourceHandler = resourceHandler; } /** * Export all user data */ async exportUserData(userId, outputStream) { const archive = archiver('zip', { zlib: { level: 9 } }); archive.pipe(outputStream); // Export notes const notes = await this.db.query( 'SELECT * FROM notes WHERE user_id = $1 AND active = TRUE', [userId] ); // Add notes as JSON archive.append(JSON.stringify(notes.rows, null, 2), { name: 'notes.json' }); // Export notebooks const notebooks = await this.db.query( 'SELECT * FROM notebooks WHERE user_id = $1', [userId] ); archive.append(JSON.stringify(notebooks.rows, null, 2), { name: 'notebooks.json' }); // Export tags const tags = await this.db.query( 'SELECT * FROM tags WHERE user_id = $1', [userId] ); archive.append(JSON.stringify(tags.rows, null, 2), { name: 'tags.json' }); // Export resources const resources = await this.db.query(` SELECT r.* FROM resources r JOIN notes n ON r.note_guid = n.guid WHERE n.user_id = $1 `, [userId]); for (const resource of resources.rows) { if (resource.storage_path) { const { data, exists } = await this.resourceHandler.getResource( resource.storage_path ); if (exists) { archive.append(data, { name: `resources/${resource.guid}${path.extname(resource.storage_path)}` }); } } } await archive.finalize(); } /** * Delete all user data (GDPR right to erasure) */ async deleteUserData(userId) { // Delete in order of dependencies await this.db.query('DELETE FROM note_tags WHERE note_guid IN (SELECT guid FROM notes WHERE user_id = $1)', [userId]); await this.db.query('DELETE FROM resources WHERE note_guid IN (SELECT guid FROM notes WHERE user_id = $1)', [userId]); await this.db.query('DELETE FROM notes WHERE user_id = $1', [userId]); await this.db.query('DELETE FROM tags WHERE user_id = $1', [userId]); await this.db.query('DELETE FROM notebooks WHERE user_id = $1', [userId]); await this.db.query('DELETE FROM sync_state WHERE user_id = $1', [userId]); await this.db.query('DELETE FROM evernote_users WHERE id = $1', [userId]); // Delete stored resources const userStoragePath = path.join(this.resourceHandler.storageRoot, userId.toString()); await fs.rm(userStoragePath, { recursive: true, force: true }); return { deleted: true }; } } module.exports = DataExporter; ``` ## Output - Database schema for Evernote data - ENML content processor - Resource/attachment handler - Sync data manager - Data export utility ## Data Handling Checklist ```markdown - [ ] Database schema created with proper indexes - [ ] ENML content validation implemented - [ ] Resource storage configured - [ ] Sync state management implemented - [ ] Data export capability - [ ] Data deletion capability (GDPR) - [ ] Backup procedures documented ``` ## Resources - [Evernote Data Model](https://dev.evernote.com/doc/articles/data_model.php) - [ENML Reference](https://dev.evernote.com/doc/articles/enml.php) - [Synchronization](https://dev.evernote.com/doc/articles/synchronization.php) ## Next Steps For enterprise features, see `evernote-enterprise-rbac`.

Skill file: plugins/saas-packs/evernote-pack/skills/evernote-data-handling/SKILL.md