Создайте файл DSL с XText из модели Java

Я недавно начал работать с XText. До сих пор мне удавалось определить простую грамматику, заполнить JvmModelInferrer и сгенерировать соответствующие классы java и файлы.java.

Можно ли автоматически генерировать файл DSL (с учетом его грамматики) из набора пользовательских классов Java?

Позвольте мне привести простой пример.

У меня есть следующая грамматика:

MODEL:
    entities+=ENTITY*
;

ENTITY:
    'entity' name=ValidID 'as'
        (elements+=PROPERTY)*
    'end'
;

PROPERTY:
    (many?='many')? 'property' name=ID 'of' type=JvmTypeReference
;

Если у меня есть следующий sample.myDsl

entity Book as
    property title of String
    property numPages of Integer
end

entity Author as
    property name of String
    property surname of String
end

Я получаю файлы Book.java и Author.java. В моем проекте у меня есть процессор, который анализирует java-файлы и создает из них объекты, поэтому, если я запусту процессор на предыдущих Book.java и Author.java, я получу два экземпляра пользовательского java-типа Entity. Каждый экземпляр Entity будет иметь набор экземпляров Property. Итак, модель Java очень похожа на грамматику xtext.

Можно ли "подать" эти два объекта в XText, возможно определить Inferrer, чтобы указать переводы, и с учетом того же файла грамматики.xtext автоматически сгенерировать файл.myDsl?

1 ответ

Решение

С Xtext это обычно не проблема

  • создать модель как аст
  • добавить его в ресурс
  • сохранить ресурс для его сериализации

если вы используете xbase и jvmmodelinferrrer, сборка ast может быть болезненной, если вы ссылаетесь из модели на выводимый элемент jvm или пытаетесь построить выражения xbase, так как ast - простой сложный пример, использующий пример domainmodel

package org.eclipse.xtext.example.domainmodel.tests

import com.google.inject.Injector
import org.eclipse.emf.common.util.URI
import org.eclipse.emf.ecore.resource.Resource
import org.eclipse.emf.ecore.resource.ResourceSet
import org.eclipse.xtext.common.types.JvmParameterizedTypeReference
import org.eclipse.xtext.common.types.util.TypeReferences
import org.eclipse.xtext.example.domainmodel.DomainmodelStandaloneSetup
import org.eclipse.xtext.example.domainmodel.domainmodel.DomainmodelFactory
import org.eclipse.xtext.resource.DerivedStateAwareResource
import org.eclipse.xtext.resource.SaveOptions
import org.eclipse.xtext.xbase.jvmmodel.JvmTypeReferenceBuilder

class Main {
    var static extension DomainmodelFactory factory = DomainmodelFactory.eINSTANCE

    def static void main(String[] args) {
        var Injector injector = new DomainmodelStandaloneSetup().createInjectorAndDoEMFRegistration()
        val ResourceSet resourceSet = injector.getInstance(ResourceSet)
        val Resource r0 = resourceSet.createResource(URI.createURI("base/Base.dmodel"))
        val Resource r1 = resourceSet.createResource(URI.createURI("model/Person.dmodel"))
        val typeReferenceBuilder = injector.getInstance(JvmTypeReferenceBuilder.Factory).create(resourceSet)
        val typeReferences = injector.getInstance(TypeReferences)
        val model = createDomainModel
        r1.contents += model
        val model0 = createDomainModel
        r0.contents += model0
        // build the ast using xtends with clause
        model0 => [
            elements += createPackageDeclaration => [
                name = "base"
                elements += createEntity => [
                    name = "Base"
                    features+= createProperty => [
                        name = "id"
                        type = typeReferenceBuilder.typeRef("java.lang.String")
                        println(type)
                    ]
                ]
            ]
        ]
        //trigger the inferrer on resource 0
        (r0 as DerivedStateAwareResource) => [
            fullyInitialized = false
            installDerivedState(false)
        ]

        // build the ast of the second resource
        model => [
            elements += createPackageDeclaration => [
                name = "model"
                elements += createEntity => [
                    val base = typeReferences.findDeclaredType("base.Base", resourceSet)
                    println(base)
                    superType = typeReferenceBuilder.typeRef(base) as JvmParameterizedTypeReference
                    println(superType)

                    name = "Person"
                    features+= createProperty => [
                        name = "name"
                        type = typeReferenceBuilder.typeRef("java.lang.String")
                        println(type)
                    ]
                ]
            ]
        ]
        //save the resources
        r0.save(SaveOptions.defaultOptions.toOptionsMap)
        r1.save(SaveOptions.defaultOptions.toOptionsMap)
    }

}
Другие вопросы по тегам