249 lines
7.8 KiB
JavaScript
249 lines
7.8 KiB
JavaScript
// Roo Code conversion profile for rule-transformer
|
|
import { fileURLToPath } from 'url';
|
|
import path from 'path';
|
|
import fs from 'fs';
|
|
import { isSilentMode, log } from '../modules/utils.js';
|
|
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = path.dirname(__filename);
|
|
|
|
const brandName = 'Roo';
|
|
const brandDir = '.roo';
|
|
const rulesDir = '.roo/rules';
|
|
const mcpConfig = true;
|
|
const mcpConfigName = 'mcp.json';
|
|
|
|
// File name mapping (specific files with naming changes)
|
|
const fileMap = {
|
|
'cursor_rules.mdc': 'roo_rules.md',
|
|
'dev_workflow.mdc': 'dev_workflow.md',
|
|
'self_improve.mdc': 'self_improve.md',
|
|
'taskmaster.mdc': 'taskmaster.md'
|
|
// Add other mappings as needed
|
|
};
|
|
|
|
const globalReplacements = [
|
|
// 1. Handle cursor.so in any possible context
|
|
{ from: /cursor\.so/gi, to: 'roocode.com' },
|
|
// Edge case: URL with different formatting
|
|
{ from: /cursor\s*\.\s*so/gi, to: 'roocode.com' },
|
|
{ from: /https?:\/\/cursor\.so/gi, to: 'https://roocode.com' },
|
|
{ from: /https?:\/\/www\.cursor\.so/gi, to: 'https://www.roocode.com' },
|
|
// 2. Handle tool references - even partial ones
|
|
{ from: /\bedit_file\b/gi, to: 'apply_diff' },
|
|
{ from: /\bsearch tool\b/gi, to: 'search_files tool' },
|
|
{ from: /\bSearch Tool\b/g, to: 'Search_Files Tool' },
|
|
// 3. Handle basic terms (with case handling)
|
|
{
|
|
from: /\bcursor\b/gi,
|
|
to: (match) => (match.charAt(0) === 'C' ? 'Roo Code' : 'roo')
|
|
},
|
|
{ from: /Cursor/g, to: 'Roo Code' },
|
|
{ from: /CURSOR/g, to: 'ROO CODE' },
|
|
// 4. Handle file extensions
|
|
{ from: /\.mdc\b/g, to: '.md' },
|
|
// 5. Handle any missed URL patterns
|
|
{ from: /docs\.cursor\.com/gi, to: 'docs.roocode.com' },
|
|
{ from: /docs\.roo\.com/gi, to: 'docs.roocode.com' }
|
|
];
|
|
|
|
const conversionConfig = {
|
|
// Product and brand name replacements
|
|
brandTerms: [
|
|
{ from: /cursor\.so/g, to: 'roocode.com' },
|
|
{ from: /\[cursor\.so\]/g, to: '[roocode.com]' },
|
|
{ from: /href="https:\/\/cursor\.so/g, to: 'href="https://roocode.com' },
|
|
{ from: /\(https:\/\/cursor\.so/g, to: '(https://roocode.com' },
|
|
{
|
|
from: /\bcursor\b/gi,
|
|
to: (match) => (match === 'Cursor' ? 'Roo Code' : 'roo')
|
|
},
|
|
{ from: /Cursor/g, to: 'Roo Code' }
|
|
],
|
|
|
|
// File extension replacements
|
|
fileExtensions: [{ from: /\.mdc\b/g, to: '.md' }],
|
|
|
|
// Documentation URL replacements
|
|
docUrls: [
|
|
{
|
|
from: /https:\/\/docs\.cursor\.com\/[^\s)'\"]+/g,
|
|
to: (match) => match.replace('docs.cursor.com', 'docs.roocode.com')
|
|
},
|
|
{ from: /https:\/\/docs\.roo\.com\//g, to: 'https://docs.roocode.com/' }
|
|
],
|
|
|
|
// Tool references - direct replacements
|
|
toolNames: {
|
|
search: 'search_files',
|
|
read_file: 'read_file',
|
|
edit_file: 'apply_diff',
|
|
create_file: 'write_to_file',
|
|
run_command: 'execute_command',
|
|
terminal_command: 'execute_command',
|
|
use_mcp: 'use_mcp_tool',
|
|
switch_mode: 'switch_mode'
|
|
},
|
|
|
|
// Tool references in context - more specific replacements
|
|
toolContexts: [
|
|
{ from: /\bsearch tool\b/g, to: 'search_files tool' },
|
|
{ from: /\bedit_file tool\b/g, to: 'apply_diff tool' },
|
|
{ from: /\buse the search\b/g, to: 'use the search_files' },
|
|
{ from: /\bThe edit_file\b/g, to: 'The apply_diff' },
|
|
{ from: /\brun_command executes\b/g, to: 'execute_command executes' },
|
|
{ from: /\buse_mcp connects\b/g, to: 'use_mcp_tool connects' },
|
|
{ from: /\bCursor search\b/g, to: 'Roo Code search_files' },
|
|
{ from: /\bCursor edit\b/g, to: 'Roo Code apply_diff' },
|
|
{ from: /\bCursor create\b/g, to: 'Roo Code write_to_file' },
|
|
{ from: /\bCursor run\b/g, to: 'Roo Code execute_command' }
|
|
],
|
|
|
|
// Tool group and category names
|
|
toolGroups: [
|
|
{ from: /\bSearch tools\b/g, to: 'Read Group tools' },
|
|
{ from: /\bEdit tools\b/g, to: 'Edit Group tools' },
|
|
{ from: /\bRun tools\b/g, to: 'Command Group tools' },
|
|
{ from: /\bMCP servers\b/g, to: 'MCP Group tools' },
|
|
{ from: /\bSearch Group\b/g, to: 'Read Group' },
|
|
{ from: /\bEdit Group\b/g, to: 'Edit Group' },
|
|
{ from: /\bRun Group\b/g, to: 'Command Group' }
|
|
],
|
|
|
|
// File references in markdown links
|
|
fileReferences: {
|
|
pathPattern: /\[(.+?)\]\(mdc:\.cursor\/rules\/(.+?)\.mdc\)/g,
|
|
replacement: (match, text, filePath) => {
|
|
// Get the base filename
|
|
const baseName = path.basename(filePath, '.mdc');
|
|
// Get the new filename (either from mapping or by replacing extension)
|
|
const newFileName = fileMap[`${baseName}.mdc`] || `${baseName}.md`;
|
|
// Return the updated link
|
|
return `[${text}](mdc:.roo/rules/${newFileName})`;
|
|
}
|
|
}
|
|
};
|
|
|
|
// Recursively copy everything from assets/roocode to the project root
|
|
|
|
export function onAddBrandRules(targetDir) {
|
|
const sourceDir = path.resolve(__dirname, '../../assets/roocode');
|
|
copyRecursiveSync(sourceDir, targetDir);
|
|
|
|
const rooModesDir = path.join(sourceDir, '.roo');
|
|
const rooModes = ['architect', 'ask', 'boomerang', 'code', 'debug', 'test'];
|
|
|
|
// Copy .roomodes to project root
|
|
const roomodesSrc = path.join(sourceDir, '.roomodes');
|
|
const roomodesDest = path.join(targetDir, '.roomodes');
|
|
if (fs.existsSync(roomodesSrc)) {
|
|
try {
|
|
fs.copyFileSync(roomodesSrc, roomodesDest);
|
|
log('debug', `[Roo] Copied .roomodes to ${roomodesDest}`);
|
|
} catch (err) {
|
|
log('debug', `[Roo] Failed to copy .roomodes: ${err.message}`);
|
|
}
|
|
} else {
|
|
log('debug', `[Roo] .roomodes not found at ${roomodesSrc}`);
|
|
}
|
|
|
|
for (const mode of rooModes) {
|
|
const src = path.join(rooModesDir, `rules-${mode}`, `${mode}-rules`);
|
|
const dest = path.join(targetDir, '.roo', `rules-${mode}`, `${mode}-rules`);
|
|
if (fs.existsSync(src)) {
|
|
try {
|
|
const destDir = path.dirname(dest);
|
|
if (!fs.existsSync(destDir)) fs.mkdirSync(destDir, { recursive: true });
|
|
fs.copyFileSync(src, dest);
|
|
log('debug', `[Roo] Copied ${src} to ${dest}`);
|
|
} catch (err) {
|
|
log('debug', `[Roo] Failed to copy ${src} to ${dest}: ${err.message}`);
|
|
}
|
|
} else {
|
|
log('debug', `[Roo] Roo rule file not found for mode '${mode}': ${src}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
function copyRecursiveSync(src, dest) {
|
|
const exists = fs.existsSync(src);
|
|
const stats = exists && fs.statSync(src);
|
|
const isDirectory = exists && stats.isDirectory();
|
|
if (isDirectory) {
|
|
if (!fs.existsSync(dest)) fs.mkdirSync(dest, { recursive: true });
|
|
fs.readdirSync(src).forEach((childItemName) => {
|
|
copyRecursiveSync(
|
|
path.join(src, childItemName),
|
|
path.join(dest, childItemName)
|
|
);
|
|
});
|
|
} else {
|
|
fs.copyFileSync(src, dest);
|
|
}
|
|
}
|
|
|
|
export function onRemoveBrandRules(targetDir) {
|
|
log('debug', `[Roo] onRemoveBrandRules called for ${targetDir}`);
|
|
const roomodesPath = path.join(targetDir, '.roomodes');
|
|
if (fs.existsSync(roomodesPath)) {
|
|
try {
|
|
fs.rmSync(roomodesPath, { force: true });
|
|
log('debug', `[Roo] Removed .roomodes from ${targetDir}`);
|
|
} catch (err) {
|
|
log('debug', `[Roo] Failed to remove .roomodes: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
const rooDir = path.join(targetDir, '.roo');
|
|
if (fs.existsSync(rooDir)) {
|
|
fs.readdirSync(rooDir).forEach((entry) => {
|
|
if (entry.startsWith('rules-')) {
|
|
const modeDir = path.join(rooDir, entry);
|
|
try {
|
|
fs.rmSync(modeDir, { recursive: true, force: true });
|
|
log('debug', `[Roo] Removed ${modeDir}`);
|
|
} catch (err) {
|
|
log('debug', `[Roo] Failed to remove ${modeDir}: ${err.message}`);
|
|
}
|
|
}
|
|
});
|
|
if (fs.readdirSync(rooDir).length === 0) {
|
|
try {
|
|
fs.rmSync(rooDir, { recursive: true, force: true });
|
|
log('debug', `[Roo] Removed empty .roo directory`);
|
|
} catch (err) {
|
|
log('debug', `[Roo] Failed to remove .roo directory: ${err.message}`);
|
|
}
|
|
}
|
|
}
|
|
log('debug', `[Roo] onRemoveBrandRules completed for ${targetDir}`);
|
|
}
|
|
|
|
function isDirectoryEmpty(dirPath) {
|
|
return fs.readdirSync(dirPath).length === 0;
|
|
}
|
|
|
|
function onPostConvertBrandRules(targetDir) {
|
|
onAddBrandRules(targetDir);
|
|
}
|
|
|
|
function getTargetRuleFilename(sourceFilename) {
|
|
if (fileMap[sourceFilename]) {
|
|
return fileMap[sourceFilename];
|
|
}
|
|
return sourceFilename.replace(/\.mdc$/, '.md');
|
|
}
|
|
|
|
export {
|
|
conversionConfig,
|
|
fileMap,
|
|
globalReplacements,
|
|
brandName,
|
|
brandDir,
|
|
rulesDir,
|
|
getTargetRuleFilename,
|
|
mcpConfig,
|
|
mcpConfigName,
|
|
onPostConvertBrandRules
|
|
};
|