private void addPropertyLinks(
     MutableModelNode modelNode,
     ModelSchemaStore schemaStore,
     NodeInitializerRegistry nodeInitializerRegistry) {
   for (ManagedProperty<?> property : bindings.getManagedProperties().values()) {
     addPropertyLink(modelNode, property, schemaStore, nodeInitializerRegistry);
   }
   if (isNamedType()) {
     // Only initialize "name" child node if the schema has such a managed property.
     // This is not the case for a managed subtype of an unmanaged type that implements Named.
     if (bindings.getManagedProperties().containsKey("name")) {
       MutableModelNode nameLink = modelNode.getLink("name");
       if (nameLink == null) {
         throw new IllegalStateException("expected name node for " + modelNode.getPath());
       }
       nameLink.setPrivateData(ModelType.of(String.class), modelNode.getPath().getName());
     }
   }
 }
 private <P> void validateProperty(
     ModelSchema<P> propertySchema,
     ManagedProperty<P> property,
     NodeInitializerRegistry nodeInitializerRegistry) {
   if (propertySchema instanceof ManagedImplSchema) {
     if (!property.isWritable()) {
       if (isCollectionOfManagedTypes(propertySchema)) {
         CollectionSchema<P, ?> propertyCollectionsSchema =
             (CollectionSchema<P, ?>) propertySchema;
         ModelType<?> elementType = propertyCollectionsSchema.getElementType();
         nodeInitializerRegistry.ensureHasInitializer(
             forProperty(elementType, property, bindings.getPublicSchema().getType()));
       }
       if (property.isDeclaredAsHavingUnmanagedType()) {
         throw new UnmanagedPropertyMissingSetterException(property.getName());
       }
     }
   } else if (!shouldHaveANodeInitializer(property, propertySchema)
       && !property.isWritable()
       && !isNamePropertyOfANamedType(property)) {
     throw new ReadonlyImmutableManagedPropertyException(
         bindings.getPublicSchema().getType(), property.getName(), property.getType());
   }
 }
  private <P> void addPropertyLink(
      MutableModelNode modelNode,
      ManagedProperty<P> property,
      ModelSchemaStore schemaStore,
      NodeInitializerRegistry nodeInitializerRegistry) {
    ModelType<P> propertyType = property.getType();
    ModelSchema<P> propertySchema = schemaStore.getSchema(propertyType);
    ModelType<T> publicType = bindings.getPublicSchema().getType();

    validateProperty(propertySchema, property, nodeInitializerRegistry);

    ModelPath childPath = modelNode.getPath().child(property.getName());
    if (propertySchema instanceof ManagedImplSchema) {
      if (!property.isWritable()) {
        ModelRegistrations.Builder builder =
            managedRegistrationBuilder(childPath, property, nodeInitializerRegistry, publicType);
        addLink(modelNode, builder, property.isInternal());
      } else {
        if (propertySchema instanceof ScalarCollectionSchema) {
          ModelRegistrations.Builder builder =
              managedRegistrationBuilder(childPath, property, nodeInitializerRegistry, publicType);
          addLink(modelNode, builder, property.isInternal());
        } else {
          modelNode.addReference(property.getName(), propertyType, modelNode.getDescriptor());
        }
      }
    } else {
      ModelRegistrations.Builder registrationBuilder;
      if (shouldHaveANodeInitializer(property, propertySchema)) {
        registrationBuilder =
            managedRegistrationBuilder(childPath, property, nodeInitializerRegistry, publicType);
      } else {
        registrationBuilder = ModelRegistrations.of(childPath);
      }
      registrationBuilder.withProjection(new UnmanagedModelProjection<P>(propertyType));
      addLink(modelNode, registrationBuilder, property.isInternal());
    }
  }
 private boolean isNamedType() {
   return Named.class.isAssignableFrom(bindings.getPublicSchema().getType().getRawClass());
 }