UNPKG

@tripsnek/tmf

Version:

TypeScript Modeling Framework - A TypeScript port of the Eclipse Modeling Framework (EMF)

354 lines 46.8 kB
import { EEnumImpl } from '../metamodel/impl/eenum-impl.js'; import { EClassImpl } from '../metamodel/impl/eclass-impl.js'; import { EDataTypeImpl } from '../metamodel/impl/edata-type-impl.js'; import { EAttributeImpl } from '../metamodel/impl/eattribute-impl.js'; import { TUtils } from '../tutils.js'; /** * Writes EPackage metamodels to Ecore XML string format. * This class handles the core writing logic without file system dependencies. */ export class EcoreStringWriter { indentLevel = 0; indentString = ' '; /** * Converts an EPackage to Ecore XML string format. * @param ePackage The root package to serialize * @returns The XML string representation */ writeToString(ePackage) { const xml = this.buildPackageXml(ePackage, true); return `<?xml version="1.0" encoding="UTF-8"?>\n${xml}`; } /** * Builds the XML for a package. */ buildPackageXml(ePackage, isRoot = false) { const indent = this.getIndent(); let xml = ''; if (isRoot) { // Root package with full namespace declarations xml = `<ecore:EPackage xmi:version="2.0" xmlns:xmi="http://www.omg.org/XMI" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:ecore="http://www.eclipse.org/emf/2002/Ecore"`; } else { xml = `${indent}<eSubpackages`; } // Add package attributes xml += ` name="${this.escapeXml(ePackage.getName())}"`; if (ePackage.getNsURI()) { xml += ` nsURI="${this.escapeXml(ePackage.getNsURI())}"`; } if (ePackage.getNsPrefix()) { xml += ` nsPrefix="${this.escapeXml(ePackage.getNsPrefix())}"`; } xml += '>'; this.indentLevel++; // Add classifiers for (const classifier of ePackage.getEClassifiers()) { xml += '\n' + this.buildClassifierXml(classifier, ePackage); } // Add subpackages for (const subPackage of ePackage.getESubPackages()) { xml += '\n' + this.buildPackageXml(subPackage, false); } this.indentLevel--; if (isRoot) { xml += '\n</ecore:EPackage>'; } else { xml += `\n${indent}</eSubpackages>`; } return xml; } /** * Builds the XML for a classifier (EClass, EEnum, or EDataType). */ buildClassifierXml(classifier, containingPackage) { const indent = this.getIndent(); if (classifier instanceof EEnumImpl) { return this.buildEnumXml(classifier); } else if (classifier instanceof EClassImpl) { return this.buildClassXml(classifier, containingPackage); } else if (classifier instanceof EDataTypeImpl) { return this.buildDataTypeXml(classifier); } return ''; } /** * Builds the XML for an EClass. */ buildClassXml(eClass, containingPackage) { const indent = this.getIndent(); let xml = `${indent}<eClassifiers xsi:type="ecore:EClass" name="${this.escapeXml(eClass.getName())}"`; // Add super types if any const superTypes = eClass.getESuperTypes(); if (superTypes && superTypes.size() > 0) { const superTypeRefs = []; for (let i = 0; i < superTypes.size(); i++) { const superType = superTypes.get(i); superTypeRefs.push(this.getClassifierReference(superType, containingPackage)); } xml += ` eSuperTypes="${superTypeRefs.join(' ')}"`; } // Add abstract/interface flags if set if (eClass.isAbstract()) { xml += ' abstract="true"'; } if (eClass.isInterface()) { xml += ' interface="true"'; } // Check if class has content const hasFeatures = eClass.getEStructuralFeatures() && eClass.getEStructuralFeatures().size() > 0; const hasOperations = eClass.getEOperations() && eClass.getEOperations().size() > 0; if (!hasFeatures && !hasOperations) { xml += '/>'; return xml; } xml += '>'; this.indentLevel++; // Add operations if (hasOperations) { for (let i = 0; i < eClass.getEOperations().size(); i++) { const operation = eClass.getEOperations().get(i); xml += '\n' + this.buildOperationXml(operation, containingPackage); } } // Add structural features if (hasFeatures) { for (let i = 0; i < eClass.getEStructuralFeatures().size(); i++) { const feature = eClass.getEStructuralFeatures().get(i); xml += '\n' + this.buildFeatureXml(feature, containingPackage); } } this.indentLevel--; xml += `\n${indent}</eClassifiers>`; return xml; } /** * Builds the XML for an EOperation. */ buildOperationXml(operation, containingPackage) { const indent = this.getIndent(); let xml = `${indent}<eOperations name="${this.escapeXml(operation.getName())}"`; // Add return type if present if (operation.getEType()) { xml += ` eType="${this.getClassifierReference(operation.getEType(), containingPackage)}"`; } // Add upper bound if not 1 if (operation.getUpperBound() == -1) { xml += ` upperBound="${operation.getUpperBound()}"`; } // Check if operation has parameters const hasParameters = operation.getEParameters() && operation.getEParameters().size() > 0; if (!hasParameters) { xml += '/>'; return xml; } xml += '>'; this.indentLevel++; // Add parameters for (let i = 0; i < operation.getEParameters().size(); i++) { const param = operation.getEParameters().get(i); xml += '\n' + this.buildParameterXml(param, containingPackage); } this.indentLevel--; xml += `\n${indent}</eOperations>`; return xml; } /** * Builds the XML for an EParameter. */ buildParameterXml(parameter, containingPackage) { const indent = this.getIndent(); let xml = `${indent}<eParameters name="${this.escapeXml(parameter.getName())}"`; if (parameter.getUpperBound() == -1) { xml += ` upperBound="${parameter.getUpperBound()}"`; } //TODO: lower bounds for optional parameters? since we can support those in typescript if (parameter.getEType()) { xml += ` eType="${this.getClassifierReference(parameter.getEType(), containingPackage)}"`; } xml += '/>'; return xml; } /** * Builds the XML for a structural feature (EAttribute or EReference). */ buildFeatureXml(feature, containingPackage) { const indent = this.getIndent(); const isAttribute = feature instanceof EAttributeImpl; const featureType = isAttribute ? 'ecore:EAttribute' : 'ecore:EReference'; let xml = `${indent}<eStructuralFeatures xsi:type="${featureType}" name="${this.escapeXml(feature.getName())}"`; // Add type if (feature.getEType()) { xml += ` eType="${this.getClassifierReference(feature.getEType(), containingPackage)}"`; } //TODO: lowerbound? only seems to make sense if we have 'required' features if (feature.getUpperBound() == -1) { xml += ` upperBound="${feature.getUpperBound()}"`; } // Add feature-specific attributes if (isAttribute) { const attr = feature; if (attr.isId()) { xml += ' iD="true"'; } } else { const ref = feature; if (ref.isContainment()) { xml += ' containment="true"'; } if (ref.getEOpposite()) { xml += ` eOpposite="${this.getFeatureReference(ref.getEOpposite())}"`; } } // Add other properties if (feature.isTransient()) { xml += ' transient="true"'; } // if (feature.isDerived()) { // xml += ' derived="true"'; // } if (!feature.isChangeable()) { xml += ' changeable="false"'; } if (feature.isVolatile()) { xml += ' volatile="true"'; } if (feature.getDefaultValueLiteral()) { xml += ` defaultValueLiteral="${this.escapeXml(feature.getDefaultValueLiteral())}"`; } xml += '/>'; return xml; } /** * Builds the XML for an EEnum. */ buildEnumXml(eEnum) { const indent = this.getIndent(); let xml = `${indent}<eClassifiers xsi:type="ecore:EEnum" name="${this.escapeXml(eEnum.getName())}"`; const hasLiterals = eEnum.getELiterals() && eEnum.getELiterals().size() > 0; if (!hasLiterals) { xml += '/>'; return xml; } xml += '>'; this.indentLevel++; // Add enum literals for (let i = 0; i < eEnum.getELiterals().size(); i++) { const literal = eEnum.getELiterals().get(i); xml += '\n' + this.buildEnumLiteralXml(literal); } this.indentLevel--; xml += `\n${indent}</eClassifiers>`; return xml; } /** * Builds the XML for an EEnumLiteral. */ buildEnumLiteralXml(literal) { const indent = this.getIndent(); let xml = `${indent}<eLiterals name="${this.escapeXml(literal.getName())}"`; if (literal.getValue() !== undefined && literal.getValue() !== null) { xml += ` value="${literal.getValue()}"`; } if (literal.getLiteral() && literal.getLiteral() !== literal.getName()) { xml += ` literal="${this.escapeXml(literal.getLiteral())}"`; } xml += '/>'; return xml; } /** * Builds the XML for an EDataType. */ buildDataTypeXml(dataType) { const indent = this.getIndent(); let xml = `${indent}<eClassifiers xsi:type="ecore:EDataType" name="${this.escapeXml(dataType.getName())}"`; // if (dataType.getInstanceClassName()) { // xml += ` instanceClassName="${this.escapeXml(dataType.getInstanceClassName())}"`; // } xml += '/>'; return xml; } /** * Gets a reference string for a classifier. */ getClassifierReference(classifier, fromPackage) { // Check if it's a primitive type const primitiveTypes = TUtils.PRIMITIVES; if (primitiveTypes.includes(classifier.getName())) { return `ecore:EDataType http://www.eclipse.org/emf/2002/Ecore#//${classifier.getName()}`; } // Build reference path return this.buildReferencePath(classifier, fromPackage); } /** * Gets a reference string for a feature. */ getFeatureReference(feature) { const containingClass = feature.getEContainingClass(); if (!containingClass) { return `#//${feature.getName()}`; } //root packages don't require package paths if (!containingClass.getEPackage().getESuperPackage()) return `#//${containingClass.getName()}/${feature.getName()}`; else return `#//${this.getPackagePath(containingClass.getEPackage())}/${containingClass.getName()}/${feature.getName()}`; } /** * Builds a reference path to a classifier. */ buildReferencePath(classifier, fromPackage) { const classifierPackage = classifier.getEPackage(); // If in same package, use simple reference // if (classifierPackage === fromPackage) { // return `#//${classifier.getName()}`; // } // Build full path const path = this.getPackagePath(classifierPackage); if (path.length > 0) return `#//${path.join('/')}/${classifier.getName()}`; //root package elements don't require paths return `#//${classifier.getName()}`; } /** * Gets the path to a package from root. */ getPackagePath(pkg) { const path = []; let current = pkg; while (current) { path.unshift(current.getName()); current = current.getESuperPackage(); } // Remove root package name from path if (path.length > 0) { path.shift(); } return path; } /** * Escapes special XML characters. */ escapeXml(str) { if (!str) return ''; return str .replace(/&/g, '&amp;') .replace(/</g, '&lt;') .replace(/>/g, '&gt;') .replace(/"/g, '&quot;') .replace(/'/g, '&apos;'); } /** * Gets the current indentation string. */ getIndent() { return this.indentString.repeat(this.indentLevel); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecore-string-writer.js","sourceRoot":"","sources":["../../../src/lib/ecore/ecore-string-writer.ts"],"names":[],"mappings":"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","sourcesContent":["import { EPackage } from '../metamodel/api/epackage.js';\r\nimport { EClass } from '../metamodel/api/eclass.js';\r\nimport { EClassifier } from '../metamodel/api/eclassifier.js';\r\nimport { EDataType } from '../metamodel/api/edata-type.js';\r\nimport { EEnum } from '../metamodel/api/eenum.js';\r\nimport { EStructuralFeature } from '../metamodel/api/estructural-feature.js';\r\nimport { EAttribute } from '../metamodel/api/eattribute.js';\r\nimport { EReference } from '../metamodel/api/ereference.js';\r\nimport { EOperation } from '../metamodel/api/eoperation.js';\r\nimport { EParameter } from '../metamodel/api/eparameter.js';\r\nimport { EEnumLiteral } from '../metamodel/api/eenum-literal.js';\r\nimport { EEnumImpl } from '../metamodel/impl/eenum-impl.js';\r\nimport { EClassImpl } from '../metamodel/impl/eclass-impl.js';\r\nimport { EDataTypeImpl } from '../metamodel/impl/edata-type-impl.js';\r\nimport { EAttributeImpl } from '../metamodel/impl/eattribute-impl.js';\r\nimport { TUtils } from '../tutils.js';\r\n\r\n/**\r\n * Writes EPackage metamodels to Ecore XML string format.\r\n * This class handles the core writing logic without file system dependencies.\r\n */\r\nexport class EcoreStringWriter {\r\n  private indentLevel = 0;\r\n  private indentString = '  ';\r\n  \r\n  /**\r\n   * Converts an EPackage to Ecore XML string format.\r\n   * @param ePackage The root package to serialize\r\n   * @returns The XML string representation\r\n   */\r\n  public writeToString(ePackage: EPackage): string {\r\n    const xml = this.buildPackageXml(ePackage, true);\r\n    return `<?xml version=\"1.0\" encoding=\"UTF-8\"?>\\n${xml}`;\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for a package.\r\n   */\r\n  private buildPackageXml(ePackage: EPackage, isRoot: boolean = false): string {\r\n    const indent = this.getIndent();\r\n    let xml = '';\r\n    \r\n    if (isRoot) {\r\n      // Root package with full namespace declarations\r\n      xml = `<ecore:EPackage xmi:version=\"2.0\" xmlns:xmi=\"http://www.omg.org/XMI\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\r\n    xmlns:ecore=\"http://www.eclipse.org/emf/2002/Ecore\"`;\r\n    } else {\r\n      xml = `${indent}<eSubpackages`;\r\n    }\r\n    \r\n    // Add package attributes\r\n    xml += ` name=\"${this.escapeXml(ePackage.getName())}\"`;\r\n    if (ePackage.getNsURI()) {\r\n      xml += ` nsURI=\"${this.escapeXml(ePackage.getNsURI())}\"`;\r\n    }\r\n    if (ePackage.getNsPrefix()) {\r\n      xml += ` nsPrefix=\"${this.escapeXml(ePackage.getNsPrefix())}\"`;\r\n    }\r\n    xml += '>';\r\n    \r\n    this.indentLevel++;\r\n    \r\n    // Add classifiers\r\n    for (const classifier of ePackage.getEClassifiers()) {\r\n      xml += '\\n' + this.buildClassifierXml(classifier, ePackage);\r\n    }\r\n    \r\n    // Add subpackages\r\n    for (const subPackage of ePackage.getESubPackages()) {\r\n      xml += '\\n' + this.buildPackageXml(subPackage, false);\r\n    }\r\n    \r\n    this.indentLevel--;\r\n    \r\n    if (isRoot) {\r\n      xml += '\\n</ecore:EPackage>';\r\n    } else {\r\n      xml += `\\n${indent}</eSubpackages>`;\r\n    }\r\n    \r\n    return xml;\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for a classifier (EClass, EEnum, or EDataType).\r\n   */\r\n  private buildClassifierXml(classifier: EClassifier, containingPackage: EPackage): string {\r\n    const indent = this.getIndent();\r\n    \r\n    if (classifier instanceof EEnumImpl) {\r\n      return this.buildEnumXml(classifier as EEnum);\r\n    } else if (classifier instanceof EClassImpl) {\r\n      return this.buildClassXml(classifier as EClass, containingPackage);\r\n    } else if (classifier instanceof EDataTypeImpl) {\r\n      return this.buildDataTypeXml(classifier as EDataType);\r\n    }\r\n    \r\n    return '';\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for an EClass.\r\n   */\r\n  private buildClassXml(eClass: EClass, containingPackage: EPackage): string {\r\n    const indent = this.getIndent();\r\n    let xml = `${indent}<eClassifiers xsi:type=\"ecore:EClass\" name=\"${this.escapeXml(eClass.getName())}\"`;\r\n    \r\n    // Add super types if any\r\n    const superTypes = eClass.getESuperTypes();\r\n    if (superTypes && superTypes.size() > 0) {\r\n      const superTypeRefs: string[] = [];\r\n      for (let i = 0; i < superTypes.size(); i++) {\r\n        const superType = superTypes.get(i);\r\n        superTypeRefs.push(this.getClassifierReference(superType, containingPackage));\r\n      }\r\n      xml += ` eSuperTypes=\"${superTypeRefs.join(' ')}\"`;\r\n    }\r\n    \r\n    // Add abstract/interface flags if set\r\n    if (eClass.isAbstract()) {\r\n      xml += ' abstract=\"true\"';\r\n    }\r\n    if (eClass.isInterface()) {\r\n      xml += ' interface=\"true\"';\r\n    }\r\n    \r\n    // Check if class has content\r\n    const hasFeatures = eClass.getEStructuralFeatures() && eClass.getEStructuralFeatures().size() > 0;\r\n    const hasOperations = eClass.getEOperations() && eClass.getEOperations().size() > 0;\r\n    \r\n    if (!hasFeatures && !hasOperations) {\r\n      xml += '/>';\r\n      return xml;\r\n    }\r\n    \r\n    xml += '>';\r\n    this.indentLevel++;\r\n    \r\n    // Add operations\r\n    if (hasOperations) {\r\n      for (let i = 0; i < eClass.getEOperations().size(); i++) {\r\n        const operation = eClass.getEOperations().get(i);\r\n        xml += '\\n' + this.buildOperationXml(operation, containingPackage);\r\n      }\r\n    }\r\n    \r\n    // Add structural features\r\n    if (hasFeatures) {\r\n      for (let i = 0; i < eClass.getEStructuralFeatures().size(); i++) {\r\n        const feature = eClass.getEStructuralFeatures().get(i);\r\n        xml += '\\n' + this.buildFeatureXml(feature, containingPackage);\r\n      }\r\n    }\r\n    \r\n    this.indentLevel--;\r\n    xml += `\\n${indent}</eClassifiers>`;\r\n    \r\n    return xml;\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for an EOperation.\r\n   */\r\n  private buildOperationXml(operation: EOperation, containingPackage: EPackage): string {\r\n    const indent = this.getIndent();\r\n    let xml = `${indent}<eOperations name=\"${this.escapeXml(operation.getName())}\"`;\r\n    \r\n    // Add return type if present\r\n    if (operation.getEType()) {\r\n      xml += ` eType=\"${this.getClassifierReference(operation.getEType()!, containingPackage)}\"`;\r\n    }\r\n    \r\n    // Add upper bound if not 1\r\n    if (operation.getUpperBound() ==-1) {\r\n      xml += ` upperBound=\"${operation.getUpperBound()}\"`;\r\n    }\r\n    \r\n    // Check if operation has parameters\r\n    const hasParameters = operation.getEParameters() && operation.getEParameters().size() > 0;\r\n    \r\n    if (!hasParameters) {\r\n      xml += '/>';\r\n      return xml;\r\n    }\r\n    \r\n    xml += '>';\r\n    this.indentLevel++;\r\n    \r\n    // Add parameters\r\n    for (let i = 0; i < operation.getEParameters().size(); i++) {\r\n      const param = operation.getEParameters().get(i);\r\n      xml += '\\n' + this.buildParameterXml(param, containingPackage);\r\n    }\r\n    \r\n    this.indentLevel--;\r\n    xml += `\\n${indent}</eOperations>`;\r\n    \r\n    return xml;\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for an EParameter.\r\n   */\r\n  private buildParameterXml(parameter: EParameter, containingPackage: EPackage): string {\r\n    const indent = this.getIndent();\r\n    let xml = `${indent}<eParameters name=\"${this.escapeXml(parameter.getName())}\"`;\r\n    \r\n    if (parameter.getUpperBound()==-1) {\r\n      xml += ` upperBound=\"${parameter.getUpperBound()}\"`;\r\n    }\r\n\r\n    //TODO: lower bounds for optional parameters? since we can support those in typescript\r\n\r\n    if (parameter.getEType()) {\r\n      xml += ` eType=\"${this.getClassifierReference(parameter.getEType(), containingPackage)}\"`;\r\n    }\r\n      \r\n    xml += '/>';\r\n    return xml;\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for a structural feature (EAttribute or EReference).\r\n   */\r\n  private buildFeatureXml(feature: EStructuralFeature, containingPackage: EPackage): string {\r\n    const indent = this.getIndent();\r\n    const isAttribute = feature instanceof EAttributeImpl;\r\n    const featureType = isAttribute ? 'ecore:EAttribute' : 'ecore:EReference';\r\n    \r\n    let xml = `${indent}<eStructuralFeatures xsi:type=\"${featureType}\" name=\"${this.escapeXml(feature.getName())}\"`;\r\n    \r\n    // Add type\r\n    if (feature.getEType()) {\r\n      xml += ` eType=\"${this.getClassifierReference(feature.getEType()!, containingPackage)}\"`;\r\n    }\r\n    \r\n    //TODO: lowerbound? only seems to make sense if we have 'required' features  \r\n    if (feature.getUpperBound()==-1) {\r\n      xml += ` upperBound=\"${feature.getUpperBound()}\"`;\r\n    }\r\n    \r\n    // Add feature-specific attributes\r\n    if (isAttribute) {\r\n      const attr = feature as EAttribute;\r\n      if (attr.isId()) {\r\n        xml += ' iD=\"true\"';\r\n      }\r\n    } else {\r\n      const ref = feature as EReference;\r\n      if (ref.isContainment()) {\r\n        xml += ' containment=\"true\"';\r\n      }\r\n      if (ref.getEOpposite()) {\r\n        xml += ` eOpposite=\"${this.getFeatureReference(ref.getEOpposite()!)}\"`;\r\n      }\r\n    }\r\n    \r\n    // Add other properties\r\n    if (feature.isTransient()) {\r\n      xml += ' transient=\"true\"';\r\n    }\r\n    // if (feature.isDerived()) {\r\n    //   xml += ' derived=\"true\"';\r\n    // }\r\n    if (!feature.isChangeable()) {\r\n      xml += ' changeable=\"false\"';\r\n    }\r\n    if (feature.isVolatile()) {\r\n      xml += ' volatile=\"true\"';\r\n    }\r\n    if (feature.getDefaultValueLiteral()) {\r\n      xml += ` defaultValueLiteral=\"${this.escapeXml(feature.getDefaultValueLiteral())}\"`;\r\n    }\r\n    \r\n    xml += '/>';\r\n    return xml;\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for an EEnum.\r\n   */\r\n  private buildEnumXml(eEnum: EEnum): string {\r\n    const indent = this.getIndent();\r\n    let xml = `${indent}<eClassifiers xsi:type=\"ecore:EEnum\" name=\"${this.escapeXml(eEnum.getName())}\"`;\r\n    \r\n    const hasLiterals = eEnum.getELiterals() && eEnum.getELiterals().size() > 0;\r\n    \r\n    if (!hasLiterals) {\r\n      xml += '/>';\r\n      return xml;\r\n    }\r\n    \r\n    xml += '>';\r\n    this.indentLevel++;\r\n    \r\n    // Add enum literals\r\n    for (let i = 0; i < eEnum.getELiterals().size(); i++) {\r\n      const literal = eEnum.getELiterals().get(i);\r\n      xml += '\\n' + this.buildEnumLiteralXml(literal);\r\n    }\r\n    \r\n    this.indentLevel--;\r\n    xml += `\\n${indent}</eClassifiers>`;\r\n    \r\n    return xml;\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for an EEnumLiteral.\r\n   */\r\n  private buildEnumLiteralXml(literal: EEnumLiteral): string {\r\n    const indent = this.getIndent();\r\n    let xml = `${indent}<eLiterals name=\"${this.escapeXml(literal.getName())}\"`;\r\n    \r\n    if (literal.getValue() !== undefined && literal.getValue() !== null) {\r\n      xml += ` value=\"${literal.getValue()}\"`;\r\n    }\r\n    \r\n    if (literal.getLiteral() && literal.getLiteral() !== literal.getName()) {\r\n      xml += ` literal=\"${this.escapeXml(literal.getLiteral())}\"`;\r\n    }\r\n    \r\n    xml += '/>';\r\n    return xml;\r\n  }\r\n  \r\n  /**\r\n   * Builds the XML for an EDataType.\r\n   */\r\n  private buildDataTypeXml(dataType: EDataType): string {\r\n    const indent = this.getIndent();\r\n    let xml = `${indent}<eClassifiers xsi:type=\"ecore:EDataType\" name=\"${this.escapeXml(dataType.getName())}\"`;\r\n    \r\n    // if (dataType.getInstanceClassName()) {\r\n    //   xml += ` instanceClassName=\"${this.escapeXml(dataType.getInstanceClassName())}\"`;\r\n    // }\r\n    \r\n    xml += '/>';\r\n    return xml;\r\n  }\r\n  \r\n  /**\r\n   * Gets a reference string for a classifier.\r\n   */\r\n  private getClassifierReference(classifier: EClassifier, fromPackage: EPackage): string {\r\n    // Check if it's a primitive type\r\n    const primitiveTypes = TUtils.PRIMITIVES;\r\n    if (primitiveTypes.includes(classifier.getName())) {\r\n      return `ecore:EDataType http://www.eclipse.org/emf/2002/Ecore#//${classifier.getName()}`;\r\n    }\r\n    \r\n    // Build reference path\r\n    return this.buildReferencePath(classifier, fromPackage);\r\n  }\r\n  \r\n  /**\r\n   * Gets a reference string for a feature.\r\n   */\r\n  private getFeatureReference(feature: EStructuralFeature): string {\r\n    const containingClass = feature.getEContainingClass();\r\n    if (!containingClass) {\r\n      return `#//${feature.getName()}`;\r\n    }\r\n    //root packages don't require package paths\r\n    if(!containingClass.getEPackage().getESuperPackage())\r\n      return `#//${containingClass.getName()}/${feature.getName()}`;\r\n    else\r\n     return `#//${this.getPackagePath(containingClass.getEPackage())}/${containingClass.getName()}/${feature.getName()}`;\r\n  }\r\n  \r\n  /**\r\n   * Builds a reference path to a classifier.\r\n   */\r\n  private buildReferencePath(classifier: EClassifier, fromPackage: EPackage): string {\r\n    const classifierPackage = classifier.getEPackage();\r\n    \r\n    // If in same package, use simple reference\r\n    // if (classifierPackage === fromPackage) {\r\n    //   return `#//${classifier.getName()}`;\r\n    // }\r\n    \r\n    // Build full path\r\n    const path = this.getPackagePath(classifierPackage);\r\n    if(path.length>0)\r\n      return `#//${path.join('/')}/${classifier.getName()}`;\r\n\r\n    //root package elements don't require paths\r\n    return `#//${classifier.getName()}`;\r\n  }\r\n  \r\n  /**\r\n   * Gets the path to a package from root.\r\n   */\r\n  private getPackagePath(pkg: EPackage): string[] {\r\n    const path: string[] = [];\r\n    let current = pkg;\r\n    \r\n    while (current) {\r\n      path.unshift(current.getName());\r\n      current = current.getESuperPackage();\r\n    }\r\n    \r\n    // Remove root package name from path\r\n    if (path.length > 0) {\r\n      path.shift();\r\n    }\r\n    \r\n    return path;\r\n  }\r\n  \r\n  /**\r\n   * Escapes special XML characters.\r\n   */\r\n  private escapeXml(str: string): string {\r\n    if (!str) return '';\r\n    return str\r\n      .replace(/&/g, '&amp;')\r\n      .replace(/</g, '&lt;')\r\n      .replace(/>/g, '&gt;')\r\n      .replace(/\"/g, '&quot;')\r\n      .replace(/'/g, '&apos;');\r\n  }\r\n  \r\n  /**\r\n   * Gets the current indentation string.\r\n   */\r\n  private getIndent(): string {\r\n    return this.indentString.repeat(this.indentLevel);\r\n  }\r\n}"]}