@tripsnek/tmf
Version:
TypeScript Modeling Framework - A TypeScript port of the Eclipse Modeling Framework (EMF)
354 lines • 46.8 kB
JavaScript
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, '&')
.replace(/</g, '<')
.replace(/>/g, '>')
.replace(/"/g, '"')
.replace(/'/g, ''');
}
/**
* 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}"]}