MCP Server
by mokemoke0821
Verified
/**
* cross-platform-integration.js
* MCPサーバーにクロスプラットフォーム対応機能を統合するモジュール
*/
const platformInfo = require('./platform-detect');
const crossPlatformPath = require('./cross-platform-path');
const configTransformer = require('./config-transform');
const fs = require('fs-extra');
const path = require('path');
/**
* クロスプラットフォーム対応機能の統合クラス
*/
class CrossPlatformIntegration {
/**
* MCPサーバーにクロスプラットフォーム対応機能を統合
* @param {object} server - MCPサーバーインスタンス
* @param {object} config - 設定オブジェクト
*/
constructor(server, config = {}) {
this.server = server;
this.config = config;
this.platformInfo = platformInfo;
this.path = crossPlatformPath;
this.configTransformer = configTransformer;
// ロガーの設定
this.logger = this.setupLogger();
}
/**
* ロガーのセットアップ
* @returns {object} - ロガーオブジェクト
*/
setupLogger() {
const logDir = this.config.monitoring?.logPath || './logs';
const normalizedLogDir = this.path.normalize(logDir);
// ディレクトリが存在しなければ作成
if (!fs.existsSync(normalizedLogDir)) {
fs.mkdirSync(normalizedLogDir, { recursive: true });
}
const logFile = path.join(normalizedLogDir, 'cross-platform.log');
// シンプルなロガーを実装
return {
info: (context, message) => {
if (!this.config.monitoring?.errorLogging) return;
const logEntry = `[${new Date().toISOString()}] [INFO] [${context}] ${message}\n`;
fs.appendFileSync(logFile, logEntry);
},
error: (context, error) => {
if (!this.config.monitoring?.errorLogging) return;
const logEntry = `[${new Date().toISOString()}] [ERROR] [${context}] ${error.stack || error}\n`;
fs.appendFileSync(logFile, logEntry);
}
};
}
/**
* クロスプラットフォーム対応機能をMCPサーバーに統合
*/
integrate() {
try {
this.logger.info('統合開始', 'クロスプラットフォーム対応機能の統合を開始します');
// クロスプラットフォームパスユーティリティをグローバルに統合
this.integratePathUtilities();
// 設定ファイル変換を統合
this.integrateConfigTransformer();
// プラットフォーム検出APIを公開
this.exposePlatformInfo();
this.logger.info('統合完了', 'クロスプラットフォーム対応機能が正常に統合されました');
return true;
} catch (error) {
this.logger.error('統合エラー', error);
return false;
}
}
/**
* パス操作ユーティリティを統合
*/
integratePathUtilities() {
try {
this.logger.info('パスユーティリティ統合', 'クロスプラットフォームパス操作機能を統合します');
// ファイルヘルパーを拡張
if (this.server.fileHelpers) {
// 既存の関数をバックアップ
const originalIsPathAllowed = this.server.fileHelpers.isPathAllowed;
const originalValidateFilePath = this.server.fileHelpers.validateFilePath;
// クロスプラットフォーム対応版で上書き
this.server.fileHelpers.isPathAllowed = (filePath) => {
// パスを正規化
const normalizedPath = this.path.normalize(filePath);
// 許可パスリストも正規化して比較
return this.config.fileOperations?.allowedPaths.some(allowedPath => {
const normalizedAllowedPath = this.path.normalize(allowedPath);
return this.path.isSubPath(normalizedAllowedPath, normalizedPath);
});
};
this.server.fileHelpers.validateFilePath = (filePath) => {
if (!this.server.fileHelpers.isPathAllowed(filePath)) {
throw new Error(`Access denied: ${filePath} is not in an allowed directory`);
}
return this.path.normalize(filePath);
};
// 新しい関数を追加
this.server.fileHelpers.expandTilde = (filePath) => {
return this.path.expandTilde(filePath);
};
this.server.fileHelpers.isAbsolute = (filePath) => {
return this.path.isAbsolute(filePath);
};
this.server.fileHelpers.join = (...paths) => {
return this.path.join(...paths);
};
this.server.fileHelpers.resolve = (...paths) => {
return this.path.resolve(...paths);
};
this.server.fileHelpers.relative = (from, to) => {
return this.path.relative(from, to);
};
}
this.logger.info('パスユーティリティ統合完了', 'クロスプラットフォームパス操作機能が正常に統合されました');
} catch (error) {
this.logger.error('パスユーティリティ統合エラー', error);
throw error;
}
}
/**
* 設定変換モジュールを統合
*/
integrateConfigTransformer() {
try {
this.logger.info('設定変換統合', '設定変換機能を統合します');
// 設定APIの追加
if (this.server.app) {
// 設定検証APIの追加
this.server.app.post("/config/validate", async (req, res) => {
try {
const { config } = req.body;
if (!config) {
return res.status(400).json({ error: "Config object is required" });
}
try {
// 設定を変換
const transformedConfig = this.configTransformer.transformConfig(config);
// パスの検証
let pathValidation = null;
if (transformedConfig.fileOperations?.allowedPaths) {
pathValidation = await this.configTransformer.validatePaths(
transformedConfig.fileOperations.allowedPaths
);
}
res.json({
original: config,
transformed: transformedConfig,
pathValidation,
timestamp: new Date().toISOString()
});
} catch (error) {
this.logger.error(`Config validation error: ${error.message}`, error);
res.status(403).json({ error: error.message });
}
} catch (error) {
this.logger.error("Error validating config:", error);
res.status(500).json({ error: "Failed to validate config" });
}
});
// 設定保存APIの追加
this.server.app.post("/config/save", async (req, res) => {
try {
const { config, path: configPath } = req.body;
if (!config || !configPath) {
return res.status(400).json({
error: "Config object and path are required"
});
}
try {
// 設定を変換
const transformedConfig = this.configTransformer.transformConfig(config);
// 設定ファイルのパスを検証(特別にルートディレクトリへのアクセスを許可)
const normalizedPath = this.path.normalize(configPath);
if (!normalizedPath.includes(this.server.appDir)) {
return res.status(403).json({
error: "Config path must be within app directory"
});
}
// 設定を保存
await this.configTransformer.saveConfig(normalizedPath, transformedConfig);
res.json({
path: normalizedPath,
success: true,
timestamp: new Date().toISOString()
});
} catch (error) {
this.logger.error(`Config save error: ${error.message}`, error);
res.status(403).json({ error: error.message });
}
} catch (error) {
this.logger.error("Error saving config:", error);
res.status(500).json({ error: "Failed to save config" });
}
});
}
this.logger.info('設定変換統合完了', '設定変換機能が正常に統合されました');
} catch (error) {
this.logger.error('設定変換統合エラー', error);
throw error;
}
}
/**
* プラットフォーム情報APIを公開
*/
exposePlatformInfo() {
try {
this.logger.info('プラットフォーム情報API', 'プラットフォーム情報APIを公開します');
// プラットフォーム情報APIの追加
if (this.server.app) {
this.server.app.get("/platform/info", (req, res) => {
try {
// プラットフォーム情報を返す
res.json({
platform: this.platformInfo.platform,
isWindows: this.platformInfo.isWindows,
isMac: this.platformInfo.isMac,
isLinux: this.platformInfo.isLinux,
osName: this.platformInfo.osName,
osVersion: this.platformInfo.osVersion,
homeDir: this.platformInfo.homeDir,
tempDir: this.platformInfo.tempDir,
appDir: this.platformInfo.appDir,
nodeVersion: this.platformInfo.nodeVersion,
pathSeparator: this.platformInfo.pathSeparator,
locale: this.platformInfo.getLocaleInfo(),
timestamp: new Date().toISOString()
});
} catch (error) {
this.logger.error("Error getting platform info:", error);
res.status(500).json({ error: "Failed to get platform info" });
}
});
}
this.logger.info('プラットフォーム情報API公開完了', 'プラットフォーム情報APIが正常に公開されました');
} catch (error) {
this.logger.error('プラットフォーム情報API公開エラー', error);
throw error;
}
}
}
module.exports = CrossPlatformIntegration;