Note
For the Gradle build configuration see Build section.

This tutorial assumes you have at least skimmed the Domain model part of the Codion manual.

Each section of the tutorial below is based on a single table and has the following subsections:

SQL

The underlying table DDL.

Domain

The domain model specification for an entity based on the table.

Model

The application model components associated with the entity (if any).

UI

The application UI components associated with the entity.

Domain model

The domain model is created by extending the DefaultDomain class and defining a DomainType constant identifying the domain model.

In the constructor we call methods each adding a single Entity definition to the domain model. The Domain subsections below continue the Petclinic class.

public final class Petclinic extends DomainModel {

  public static final DomainType DOMAIN = domainType("Petclinic");

  public Petclinic() {
    super(DOMAIN);
    add(vet(), specialty(), vetSpecialty(), petType(), owner(), pet(), visit());
  }
Display full Petclinic domain model class
/*
 * This file is part of Codion Petclinic Demo.
 *
 * Codion Petclinic Demo is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * Codion Petclinic Demo is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Codion Petclinic Demo.  If not, see <http://www.gnu.org/licenses/>.
 *
 * Copyright (c) 2004 - 2025, Björn Darri Sigurðsson.
 */
package is.codion.demos.petclinic.domain;

import is.codion.demos.petclinic.domain.Petclinic.Owner.PhoneType;
import is.codion.framework.domain.DomainModel;
import is.codion.framework.domain.DomainType;
import is.codion.framework.domain.entity.EntityDefinition;
import is.codion.framework.domain.entity.EntityType;
import is.codion.framework.domain.entity.OrderBy;
import is.codion.framework.domain.entity.StringFactory;
import is.codion.framework.domain.entity.attribute.Column;
import is.codion.framework.domain.entity.attribute.Column.Converter;
import is.codion.framework.domain.entity.attribute.ForeignKey;

import java.sql.Statement;
import java.time.LocalDate;

import static is.codion.framework.domain.DomainType.domainType;
import static is.codion.framework.domain.entity.KeyGenerator.identity;
import static is.codion.framework.domain.entity.OrderBy.ascending;

public final class Petclinic extends DomainModel {

  public static final DomainType DOMAIN = domainType("Petclinic");

  public Petclinic() {
    super(DOMAIN);
    add(vet(), specialty(), vetSpecialty(), petType(), owner(), pet(), visit());
  }

  public interface Vet {
    EntityType TYPE = DOMAIN.entityType("petclinic.vet");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> FIRST_NAME = TYPE.stringColumn("first_name");
    Column<String> LAST_NAME = TYPE.stringColumn("last_name");
  }

  private EntityDefinition vet() {
    return Vet.TYPE.define(
                    Vet.ID.define()
                            .primaryKey(),
                    Vet.FIRST_NAME.define()
                            .column()
                            .caption("First name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false),
                    Vet.LAST_NAME.define()
                            .column()
                            .caption("Last name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false))
            .keyGenerator(identity())
            .caption("Vets")
            .stringFactory(StringFactory.builder()
                    .value(Vet.LAST_NAME)
                    .text(", ")
                    .value(Vet.FIRST_NAME)
                    .build())
            .orderBy(ascending(Vet.LAST_NAME, Vet.FIRST_NAME))
            .smallDataset(true)
            .build();
  }

  public interface Specialty {
    EntityType TYPE = DOMAIN.entityType("petclinic.specialty");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> NAME = TYPE.stringColumn("name");
  }

  private EntityDefinition specialty() {
    return Specialty.TYPE.define(
                    Specialty.ID.define()
                            .primaryKey(),
                    Specialty.NAME.define()
                            .column()
                            .caption("Name")
                            .searchable(true)
                            .maximumLength(80)
                            .nullable(false))
            .keyGenerator(identity())
            .caption("Specialties")
            .stringFactory(Specialty.NAME)
            .smallDataset(true)
            .build();
  }

  public interface VetSpecialty {
    EntityType TYPE = DOMAIN.entityType("petclinic.vet_specialty");

    Column<Integer> VET = TYPE.integerColumn("vet");
    Column<Integer> SPECIALTY = TYPE.integerColumn("specialty");

    ForeignKey VET_FK = TYPE.foreignKey("vet_fk", VET, Vet.ID);
    ForeignKey SPECIALTY_FK = TYPE.foreignKey("specialty_fk", SPECIALTY, Specialty.ID);
  }

  private EntityDefinition vetSpecialty() {
    return VetSpecialty.TYPE.define(
                    VetSpecialty.VET.define()
                            .primaryKey(0)
                            .updatable(true),
                    VetSpecialty.SPECIALTY.define()
                            .primaryKey(1)
                            .updatable(true),
                    VetSpecialty.VET_FK.define()
                            .foreignKey()
                            .caption("Vet"),
                    VetSpecialty.SPECIALTY_FK.define()
                            .foreignKey()
                            .caption("Specialty"))
            .caption("Vet specialties")
            .stringFactory(StringFactory.builder()
                    .value(VetSpecialty.VET_FK)
                    .text(" - ")
                    .value(VetSpecialty.SPECIALTY_FK)
                    .build())
            .build();
  }

  public interface PetType {
    EntityType TYPE = DOMAIN.entityType("petclinic.pet_type");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> NAME = TYPE.stringColumn("name");
  }

  private EntityDefinition petType() {
    return PetType.TYPE.define(
                    PetType.ID.define()
                            .primaryKey(),
                    PetType.NAME.define()
                            .column()
                            .caption("Name")
                            .searchable(true)
                            .maximumLength(80)
                            .nullable(false))
            .keyGenerator(identity())
            .caption("Pet types")
            .stringFactory(PetType.NAME)
            .orderBy(ascending(PetType.NAME))
            .smallDataset(true)
            .build();
  }

  public interface Owner {
    EntityType TYPE = DOMAIN.entityType("petclinic.owner");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> FIRST_NAME = TYPE.stringColumn("first_name");
    Column<String> LAST_NAME = TYPE.stringColumn("last_name");
    Column<String> ADDRESS = TYPE.stringColumn("address");
    Column<String> CITY = TYPE.stringColumn("city");
    Column<String> TELEPHONE = TYPE.stringColumn("telephone");
    Column<PhoneType> PHONE_TYPE = TYPE.column("phone_type", PhoneType.class);

    enum PhoneType {
      MOBILE, HOME, WORK
    }
  }

  private EntityDefinition owner() {
    return Owner.TYPE.define(
                    Owner.ID.define()
                            .primaryKey(),
                    Owner.FIRST_NAME.define()
                            .column()
                            .caption("First name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false),
                    Owner.LAST_NAME.define()
                            .column()
                            .caption("Last name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false),
                    Owner.ADDRESS.define()
                            .column()
                            .caption("Address")
                            .maximumLength(255),
                    Owner.CITY.define()
                            .column()
                            .caption("City")
                            .maximumLength(80),
                    Owner.TELEPHONE.define()
                            .column()
                            .caption("Telephone")
                            .maximumLength(20),
                    Owner.PHONE_TYPE.define()
                            .column()
                            .caption("Phone type")
                            .columnClass(String.class, new PhoneTypeConverter()))
            .keyGenerator(identity())
            .caption("Owners")
            .stringFactory(StringFactory.builder()
                    .value(Owner.LAST_NAME)
                    .text(", ")
                    .value(Owner.FIRST_NAME)
                    .build())
            .orderBy(ascending(Owner.LAST_NAME, Owner.FIRST_NAME))
            .build();
  }

  private static final class PhoneTypeConverter implements Converter<PhoneType, String> {

    @Override
    public String toColumnValue(PhoneType value, Statement statement) {
      return value.name();
    }

    @Override
    public PhoneType fromColumnValue(String columnValue) {
      return PhoneType.valueOf(columnValue);
    }
  }

  public interface Pet {
    EntityType TYPE = DOMAIN.entityType("petclinic.pet");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> NAME = TYPE.stringColumn("name");
    Column<LocalDate> BIRTH_DATE = TYPE.localDateColumn("birth_date");
    Column<Integer> PET_TYPE_ID = TYPE.integerColumn("type_id");
    Column<Integer> OWNER_ID = TYPE.integerColumn("owner_id");

    ForeignKey PET_TYPE_FK = TYPE.foreignKey("type_fk", PET_TYPE_ID, PetType.ID);
    ForeignKey OWNER_FK = TYPE.foreignKey("owner_fk", OWNER_ID, Owner.ID);
  }

  private EntityDefinition pet() {
    return Pet.TYPE.define(
                    Pet.ID.define()
                            .primaryKey(),
                    Pet.NAME.define()
                            .column()
                            .caption("Name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false),
                    Pet.BIRTH_DATE.define()
                            .column()
                            .caption("Birth date")
                            .nullable(false),
                    Pet.PET_TYPE_ID.define()
                            .column()
                            .nullable(false),
                    Pet.PET_TYPE_FK.define()
                            .foreignKey()
                            .caption("Pet type"),
                    Pet.OWNER_ID.define()
                            .column()
                            .nullable(false),
                    Pet.OWNER_FK.define()
                            .foreignKey()
                            .caption("Owner"))
            .keyGenerator(identity())
            .caption("Pets")
            .stringFactory(Pet.NAME)
            .orderBy(ascending(Pet.NAME))
            .build();
  }

  public interface Visit {
    EntityType TYPE = DOMAIN.entityType("petclinic.visit");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<Integer> PET_ID = TYPE.integerColumn("pet_id");
    Column<LocalDate> VISIT_DATE = TYPE.localDateColumn("visit_date");
    Column<String> DESCRIPTION = TYPE.stringColumn("description");

    ForeignKey PET_FK = TYPE.foreignKey("pet_fk", PET_ID, Pet.ID);
  }

  private EntityDefinition visit() {
    return Visit.TYPE.define(
                    Visit.ID.define()
                            .primaryKey(),
                    Visit.PET_ID.define()
                            .column()
                            .nullable(false),
                    Visit.PET_FK.define()
                            .foreignKey()
                            .caption("Pet"),
                    Visit.VISIT_DATE.define()
                            .column()
                            .caption("Date")
                            .nullable(false),
                    Visit.DESCRIPTION.define()
                            .column()
                            .caption("Description")
                            .maximumLength(255))
            .keyGenerator(identity())
            .orderBy(OrderBy.builder()
                    .ascending(Visit.PET_ID)
                    .descending(Visit.VISIT_DATE)
                    .build())
            .caption("Visits")
            .build();
  }
}

Owners

owners

Owner

SQL

CREATE TABLE petclinic.owner (
  id INTEGER GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
  first_name VARCHAR(30) NOT NULL,
  last_name VARCHAR(30) NOT NULL,
  address VARCHAR(255),
  city VARCHAR(80),
  telephone VARCHAR(20),
  phone_type VARCHAR(20) NOT NULL
);

Domain

API

We start by creating an Owner interface and defining the domain API constants for the table and its columns, providing the table and column names as parameters.

We also define a PhoneType enum, which we use as a custom type for the phone_type column.

  public interface Owner {
    EntityType TYPE = DOMAIN.entityType("petclinic.owner");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> FIRST_NAME = TYPE.stringColumn("first_name");
    Column<String> LAST_NAME = TYPE.stringColumn("last_name");
    Column<String> ADDRESS = TYPE.stringColumn("address");
    Column<String> CITY = TYPE.stringColumn("city");
    Column<String> TELEPHONE = TYPE.stringColumn("telephone");
    Column<PhoneType> PHONE_TYPE = TYPE.column("phone_type", PhoneType.class);

    enum PhoneType {
      MOBILE, HOME, WORK
    }
  }
Implementation

We then define an Entity along with its columns based on the domain API constants, configuring each for persistance and presentation.

For the Owner.PHONE_TYPE column we call columnClass() where we specify the underlying column type and provide a Column.Converter implementation, for converting the enum to and from the underlying column value.

We use the StringFactory.builder() method to build a Function<Entity, String> instance which provides the toString() implementation for entities of this type.

We also specify a default OrderBy clause to use when selecting entities of this type.

  private EntityDefinition owner() {
    return Owner.TYPE.define(
                    Owner.ID.define()
                            .primaryKey(),
                    Owner.FIRST_NAME.define()
                            .column()
                            .caption("First name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false),
                    Owner.LAST_NAME.define()
                            .column()
                            .caption("Last name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false),
                    Owner.ADDRESS.define()
                            .column()
                            .caption("Address")
                            .maximumLength(255),
                    Owner.CITY.define()
                            .column()
                            .caption("City")
                            .maximumLength(80),
                    Owner.TELEPHONE.define()
                            .column()
                            .caption("Telephone")
                            .maximumLength(20),
                    Owner.PHONE_TYPE.define()
                            .column()
                            .caption("Phone type")
                            .columnClass(String.class, new PhoneTypeConverter()))
            .keyGenerator(identity())
            .caption("Owners")
            .stringFactory(StringFactory.builder()
                    .value(Owner.LAST_NAME)
                    .text(", ")
                    .value(Owner.FIRST_NAME)
                    .build())
            .orderBy(ascending(Owner.LAST_NAME, Owner.FIRST_NAME))
            .build();
  }

  private static final class PhoneTypeConverter implements Converter<PhoneType, String> {

    @Override
    public String toColumnValue(PhoneType value, Statement statement) {
      return value.name();
    }

    @Override
    public PhoneType fromColumnValue(String columnValue) {
      return PhoneType.valueOf(columnValue);
    }
  }

UI

We extend the EntityEditPanel class, with a constructor taking a SwingEntityEditModel parameter, which we simply propagate to the super constructor. We implement the initializeUI() method in which we create the components using the create…​() methods and add them to the panel using addInputPanel().

NOTE

The create…​() methods return a ComponentBuilder instance, providing a way to configure the input fields.

  • Set the initial focus attribute, which specifies which input field should receive the focus when the panel is cleared or initialized.

  • Create input fields for the columns.

  • Set a layout and add the input fields to the panel.

package is.codion.demos.petclinic.ui;

import is.codion.demos.petclinic.domain.Petclinic.Owner;
import is.codion.swing.framework.model.SwingEntityEditModel;
import is.codion.swing.framework.ui.EntityEditPanel;

import static is.codion.swing.common.ui.layout.Layouts.gridLayout;

public final class OwnerEditPanel extends EntityEditPanel {

  public OwnerEditPanel(SwingEntityEditModel editModel) {
    super(editModel);
  }

  @Override
  protected void initializeUI() {
    focus().initial().set(Owner.FIRST_NAME);

    createTextField(Owner.FIRST_NAME);
    createTextField(Owner.LAST_NAME);
    createTextField(Owner.ADDRESS);
    createTextField(Owner.CITY);
    createTextField(Owner.TELEPHONE);
    createComboBox(Owner.PHONE_TYPE);

    setLayout(gridLayout(3, 2));

    addInputPanel(Owner.FIRST_NAME);
    addInputPanel(Owner.LAST_NAME);
    addInputPanel(Owner.ADDRESS);
    addInputPanel(Owner.CITY);
    addInputPanel(Owner.TELEPHONE);
    addInputPanel(Owner.PHONE_TYPE);
  }
}

Pet

SQL

CREATE TABLE petclinic.pet (
  id INTEGER GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
  name VARCHAR(30) NOT NULL,
  birth_date DATE,
  type_id INTEGER NOT NULL,
  owner_id INTEGER NOT NULL,
  CONSTRAINT fk_pet_owners FOREIGN KEY (owner_id)
      REFERENCES owner (id),
  CONSTRAINT fk_pet_pet_type FOREIGN KEY (type_id)
      REFERENCES pet_type (id)
);

Domain

API

For the Pet Entity we define a ForeignKey domain API constant for each foreign key, providing a name along with the columns comprising the underlying reference.

  public interface Pet {
    EntityType TYPE = DOMAIN.entityType("petclinic.pet");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> NAME = TYPE.stringColumn("name");
    Column<LocalDate> BIRTH_DATE = TYPE.localDateColumn("birth_date");
    Column<Integer> PET_TYPE_ID = TYPE.integerColumn("type_id");
    Column<Integer> OWNER_ID = TYPE.integerColumn("owner_id");

    ForeignKey PET_TYPE_FK = TYPE.foreignKey("type_fk", PET_TYPE_ID, PetType.ID);
    ForeignKey OWNER_FK = TYPE.foreignKey("owner_fk", OWNER_ID, Owner.ID);
  }
Implementation

We define the foreign keys the same way we define columns.

NOTE

The underlying foreign key reference columns do not have a caption specified, which means they will be hidden in table views by default.

  private EntityDefinition pet() {
    return Pet.TYPE.define(
                    Pet.ID.define()
                            .primaryKey(),
                    Pet.NAME.define()
                            .column()
                            .caption("Name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false),
                    Pet.BIRTH_DATE.define()
                            .column()
                            .caption("Birth date")
                            .nullable(false),
                    Pet.PET_TYPE_ID.define()
                            .column()
                            .nullable(false),
                    Pet.PET_TYPE_FK.define()
                            .foreignKey()
                            .caption("Pet type"),
                    Pet.OWNER_ID.define()
                            .column()
                            .nullable(false),
                    Pet.OWNER_FK.define()
                            .foreignKey()
                            .caption("Owner"))
            .keyGenerator(identity())
            .caption("Pets")
            .stringFactory(Pet.NAME)
            .orderBy(ascending(Pet.NAME))
            .build();
  }

UI

For the PetEditPanel we use createForeignKeyComboBox() to create a EntityComboBox for the Pet.OWNER_FK foreign key, which provides a ComboBox populated with the underlying entities.

For the Pet.PET_TYPE_FK foreign key we use createForeignKeyComboBoxPanel() which creates a EntityComboBox on a panel, which can also include buttons for adding a new item or editing the selected one. In this case we provide a button for adding a new PetType by calling add(true). The createForeignKeyComboBoxPanel() method takes a Supplier<EntityEditPanel> parameter, responsible for providing the EntityEditPanel instance to display when adding or editing items.

The createTemporalFieldPanel() method creates a temporal field for editing dates, on a panel which includes a button for displaying a calendar.

package is.codion.demos.petclinic.ui;

import is.codion.demos.petclinic.domain.Petclinic.Pet;
import is.codion.demos.petclinic.domain.Petclinic.PetType;
import is.codion.swing.framework.model.SwingEntityEditModel;
import is.codion.swing.framework.ui.EntityEditPanel;

import static is.codion.swing.common.ui.layout.Layouts.gridLayout;

public final class PetEditPanel extends EntityEditPanel {

  public PetEditPanel(SwingEntityEditModel editModel) {
    super(editModel);
  }

  @Override
  protected void initializeUI() {
    focus().initial().set(Pet.NAME);

    createComboBox(Pet.OWNER_FK);
    createTextField(Pet.NAME);
    createComboBoxPanel(Pet.PET_TYPE_FK, this::createPetTypeEditPanel)
            .includeAddButton(true);
    createTemporalFieldPanel(Pet.BIRTH_DATE);

    setLayout(gridLayout(2, 2));

    addInputPanel(Pet.OWNER_FK);
    addInputPanel(Pet.NAME);
    addInputPanel(Pet.PET_TYPE_FK);
    addInputPanel(Pet.BIRTH_DATE);
  }

  private PetTypeEditPanel createPetTypeEditPanel() {
    return new PetTypeEditPanel(new SwingEntityEditModel(PetType.TYPE, editModel().connectionProvider()));
  }
}

Visit

SQL

CREATE TABLE petclinic.visit (
  id INTEGER GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
  pet_id INTEGER NOT NULL,
  visit_date DATE NOT NULL,
  description VARCHAR(255),
  CONSTRAINT fk_visit_pets FOREIGN KEY (pet_id)
      REFERENCES pet (id)
);

Domain

API
  public interface Visit {
    EntityType TYPE = DOMAIN.entityType("petclinic.visit");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<Integer> PET_ID = TYPE.integerColumn("pet_id");
    Column<LocalDate> VISIT_DATE = TYPE.localDateColumn("visit_date");
    Column<String> DESCRIPTION = TYPE.stringColumn("description");

    ForeignKey PET_FK = TYPE.foreignKey("pet_fk", PET_ID, Pet.ID);
  }
Implementation

Here we create a OrderBy instance using a builder instead of a factory method.

  private EntityDefinition visit() {
    return Visit.TYPE.define(
                    Visit.ID.define()
                            .primaryKey(),
                    Visit.PET_ID.define()
                            .column()
                            .nullable(false),
                    Visit.PET_FK.define()
                            .foreignKey()
                            .caption("Pet"),
                    Visit.VISIT_DATE.define()
                            .column()
                            .caption("Date")
                            .nullable(false),
                    Visit.DESCRIPTION.define()
                            .column()
                            .caption("Description")
                            .maximumLength(255))
            .keyGenerator(identity())
            .orderBy(OrderBy.builder()
                    .ascending(Visit.PET_ID)
                    .descending(Visit.VISIT_DATE)
                    .build())
            .caption("Visits")
            .build();
  }

UI

When using a JTextArea requiring a JScrollPane we supply the scroll pane containing the text area, retrived via component(Attribute), as a parameter when calling addInputPanel().

package is.codion.demos.petclinic.ui;

import is.codion.demos.petclinic.domain.Petclinic.Visit;
import is.codion.swing.framework.model.SwingEntityEditModel;
import is.codion.swing.framework.ui.EntityEditPanel;

import javax.swing.JPanel;
import javax.swing.JScrollPane;
import java.awt.BorderLayout;

import static is.codion.swing.common.ui.component.Components.gridLayoutPanel;
import static is.codion.swing.common.ui.layout.Layouts.borderLayout;

public final class VisitEditPanel extends EntityEditPanel {

  public VisitEditPanel(SwingEntityEditModel editModel) {
    super(editModel);
  }

  @Override
  protected void initializeUI() {
    focus().initial().set(Visit.PET_FK);

    createComboBox(Visit.PET_FK);
    createTemporalFieldPanel(Visit.VISIT_DATE);
    createTextArea(Visit.DESCRIPTION)
            .rowsColumns(4, 20);

    JPanel northPanel = gridLayoutPanel(1, 2)
            .add(createInputPanel(Visit.PET_FK))
            .add(createInputPanel(Visit.VISIT_DATE))
            .build();

    setLayout(borderLayout());
    add(northPanel, BorderLayout.NORTH);
    addInputPanel(Visit.DESCRIPTION, new JScrollPane(component(Visit.DESCRIPTION).get()), BorderLayout.CENTER);
  }
}

Support tables

Pet Type

pet types

SQL

CREATE TABLE petclinic.pet_type (
  id INTEGER GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
  name VARCHAR(80) NOT NULL,
  CONSTRAINT pet_type_uk UNIQUE (name)
);

Domain

API

  public interface PetType {
    EntityType TYPE = DOMAIN.entityType("petclinic.pet_type");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> NAME = TYPE.stringColumn("name");
  }
Implementation
  private EntityDefinition petType() {
    return PetType.TYPE.define(
                    PetType.ID.define()
                            .primaryKey(),
                    PetType.NAME.define()
                            .column()
                            .caption("Name")
                            .searchable(true)
                            .maximumLength(80)
                            .nullable(false))
            .keyGenerator(identity())
            .caption("Pet types")
            .stringFactory(PetType.NAME)
            .orderBy(ascending(PetType.NAME))
            .smallDataset(true)
            .build();
  }

UI

package is.codion.demos.petclinic.ui;

import is.codion.demos.petclinic.domain.Petclinic.PetType;
import is.codion.swing.framework.model.SwingEntityEditModel;
import is.codion.swing.framework.ui.EntityEditPanel;

import static is.codion.swing.common.ui.layout.Layouts.gridLayout;

public final class PetTypeEditPanel extends EntityEditPanel {

  public PetTypeEditPanel(SwingEntityEditModel editModel) {
    super(editModel);
  }

  @Override
  protected void initializeUI() {
    focus().initial().set(PetType.NAME);

    createTextField(PetType.NAME);

    setLayout(gridLayout(1, 1));

    addInputPanel(PetType.NAME);
  }
}

Specialty

specialties

SQL

CREATE TABLE petclinic.specialty (
  id INTEGER GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
  name VARCHAR(80) NOT NULL
);

Domain

API
  public interface Specialty {
    EntityType TYPE = DOMAIN.entityType("petclinic.specialty");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> NAME = TYPE.stringColumn("name");
  }
Implementation
  private EntityDefinition specialty() {
    return Specialty.TYPE.define(
                    Specialty.ID.define()
                            .primaryKey(),
                    Specialty.NAME.define()
                            .column()
                            .caption("Name")
                            .searchable(true)
                            .maximumLength(80)
                            .nullable(false))
            .keyGenerator(identity())
            .caption("Specialties")
            .stringFactory(Specialty.NAME)
            .smallDataset(true)
            .build();
  }

UI

package is.codion.demos.petclinic.ui;

import is.codion.demos.petclinic.domain.Petclinic.Specialty;
import is.codion.swing.framework.model.SwingEntityEditModel;
import is.codion.swing.framework.ui.EntityEditPanel;

import static is.codion.swing.common.ui.layout.Layouts.gridLayout;

public final class SpecialtyEditPanel extends EntityEditPanel {

  public SpecialtyEditPanel(SwingEntityEditModel editModel) {
    super(editModel);
  }

  @Override
  protected void initializeUI() {
    focus().initial().set(Specialty.NAME);

    createTextField(Specialty.NAME);

    setLayout(gridLayout(1, 1));

    addInputPanel(Specialty.NAME);
  }
}

Vet

vets

SQL

CREATE TABLE petclinic.vet (
  id INTEGER GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
  first_name VARCHAR(30) NOT NULL,
  last_name VARCHAR(30) NOT NULL
);

Domain

API
  public interface Vet {
    EntityType TYPE = DOMAIN.entityType("petclinic.vet");

    Column<Integer> ID = TYPE.integerColumn("id");
    Column<String> FIRST_NAME = TYPE.stringColumn("first_name");
    Column<String> LAST_NAME = TYPE.stringColumn("last_name");
  }
Implementation
  private EntityDefinition vet() {
    return Vet.TYPE.define(
                    Vet.ID.define()
                            .primaryKey(),
                    Vet.FIRST_NAME.define()
                            .column()
                            .caption("First name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false),
                    Vet.LAST_NAME.define()
                            .column()
                            .caption("Last name")
                            .searchable(true)
                            .maximumLength(30)
                            .nullable(false))
            .keyGenerator(identity())
            .caption("Vets")
            .stringFactory(StringFactory.builder()
                    .value(Vet.LAST_NAME)
                    .text(", ")
                    .value(Vet.FIRST_NAME)
                    .build())
            .orderBy(ascending(Vet.LAST_NAME, Vet.FIRST_NAME))
            .smallDataset(true)
            .build();
  }

UI

package is.codion.demos.petclinic.ui;

import is.codion.demos.petclinic.domain.Petclinic.Vet;
import is.codion.swing.framework.model.SwingEntityEditModel;
import is.codion.swing.framework.ui.EntityEditPanel;

import static is.codion.swing.common.ui.layout.Layouts.gridLayout;

public final class VetEditPanel extends EntityEditPanel {

  public VetEditPanel(SwingEntityEditModel editModel) {
    super(editModel);
  }

  @Override
  protected void initializeUI() {
    focus().initial().set(Vet.FIRST_NAME);

    createTextField(Vet.FIRST_NAME);
    createTextField(Vet.LAST_NAME);

    setLayout(gridLayout(2, 1));

    addInputPanel(Vet.FIRST_NAME);
    addInputPanel(Vet.LAST_NAME);
  }
}

Vet Specialty

SQL

CREATE TABLE petclinic.vet_specialty (
  vet INTEGER NOT NULL,
  specialty INTEGER NOT NULL,
  CONSTRAINT fk_vet_specialty_vet FOREIGN KEY (vet)
      REFERENCES petclinic.vet (id),
  CONSTRAINT fk_vet_specialty_specialty FOREIGN KEY (specialty)
      REFERENCES petclinic.specialty (id)
);

Domain

API
  public interface VetSpecialty {
    EntityType TYPE = DOMAIN.entityType("petclinic.vet_specialty");

    Column<Integer> VET = TYPE.integerColumn("vet");
    Column<Integer> SPECIALTY = TYPE.integerColumn("specialty");

    ForeignKey VET_FK = TYPE.foreignKey("vet_fk", VET, Vet.ID);
    ForeignKey SPECIALTY_FK = TYPE.foreignKey("specialty_fk", SPECIALTY, Specialty.ID);
  }
Implementation
  private EntityDefinition vetSpecialty() {
    return VetSpecialty.TYPE.define(
                    VetSpecialty.VET.define()
                            .primaryKey(0)
                            .updatable(true),
                    VetSpecialty.SPECIALTY.define()
                            .primaryKey(1)
                            .updatable(true),
                    VetSpecialty.VET_FK.define()
                            .foreignKey()
                            .caption("Vet"),
                    VetSpecialty.SPECIALTY_FK.define()
                            .foreignKey()
                            .caption("Specialty"))
            .caption("Vet specialties")
            .stringFactory(StringFactory.builder()
                    .value(VetSpecialty.VET_FK)
                    .text(" - ")
                    .value(VetSpecialty.SPECIALTY_FK)
                    .build())
            .build();
  }

Model

Here we extend SwingEntityEditModel in order to provide validation for a Vet/Specialty combination. And yes, this should obviously be done with a unique key in the underlying table, but let’s assume we can’t do that for some reason.

The validate() method is called before inserting or updating the given entity instance.

In the constructor we initialize the ComboBox models used, creating and populating them, in order to prevent it from happening automatically when the UI is initialized on the EDT.

We also specify that the VetSpecialty.VET_FK and VetSpecialty.SPECIALTY_FK values should not persist when the edit model is cleared.

package is.codion.demos.petclinic.model;

import is.codion.demos.petclinic.domain.Petclinic.VetSpecialty;
import is.codion.framework.db.EntityConnectionProvider;
import is.codion.framework.domain.entity.DefaultEntityValidator;
import is.codion.framework.domain.entity.Entity;
import is.codion.framework.domain.entity.exception.ValidationException;
import is.codion.swing.framework.model.SwingEntityEditModel;

import static is.codion.framework.db.EntityConnection.Count.where;
import static is.codion.framework.domain.entity.condition.Condition.and;

public final class VetSpecialtyEditModel extends SwingEntityEditModel {

  public VetSpecialtyEditModel(EntityConnectionProvider connectionProvider) {
    super(VetSpecialty.TYPE, connectionProvider);
    initializeComboBoxModels(VetSpecialty.VET_FK, VetSpecialty.SPECIALTY_FK);
    value(VetSpecialty.VET_FK).persist().set(false);
    value(VetSpecialty.SPECIALTY_FK).persist().set(false);
    editor().validator().set(new VetSpecialtyValidator());
  }

  private final class VetSpecialtyValidator extends DefaultEntityValidator {

    @Override
    public void validate(Entity entity) {
      super.validate(entity);
      int rowCount = connection().count(where(and(
              VetSpecialty.SPECIALTY.equalTo(entity.get(VetSpecialty.SPECIALTY)),
              VetSpecialty.VET.equalTo(entity.get(VetSpecialty.VET)))));
      if (rowCount > 0) {
        throw new ValidationException(VetSpecialty.SPECIALTY_FK,
                entity.get(VetSpecialty.SPECIALTY_FK), "Vet/specialty combination already exists");
      }
    }
  }
}

UI

package is.codion.demos.petclinic.ui;

import is.codion.demos.petclinic.domain.Petclinic.Specialty;
import is.codion.demos.petclinic.domain.Petclinic.VetSpecialty;
import is.codion.swing.framework.model.SwingEntityEditModel;
import is.codion.swing.framework.ui.EntityEditPanel;

import static is.codion.swing.common.ui.layout.Layouts.gridLayout;

public final class VetSpecialtyEditPanel extends EntityEditPanel {

  public VetSpecialtyEditPanel(SwingEntityEditModel editModel) {
    super(editModel);
  }

  @Override
  protected void initializeUI() {
    focus().initial().set(VetSpecialty.VET_FK);

    createComboBox(VetSpecialty.VET_FK)
            .preferredWidth(200);
    createComboBoxPanel(VetSpecialty.SPECIALTY_FK, this::createSpecialtyEditPanel)
            .includeAddButton(true)
            .preferredWidth(200);

    setLayout(gridLayout(2, 1));

    addInputPanel(VetSpecialty.VET_FK);
    addInputPanel(VetSpecialty.SPECIALTY_FK);
  }

  private SpecialtyEditPanel createSpecialtyEditPanel() {
    return new SpecialtyEditPanel(new SwingEntityEditModel(Specialty.TYPE, editModel().connectionProvider()));
  }
}

PetclinicAppModel

The application model holds the SwingEntityModel instances used by the application, here we create a setupEntityModels() method for creating and configuring the application model layer. Note that we initialize ComboBox models and refresh the Owners table model, in order for that not to happen on the EDT when the UI is created.

package is.codion.demos.petclinic.model;

import is.codion.common.version.Version;
import is.codion.demos.petclinic.domain.Petclinic.Owner;
import is.codion.demos.petclinic.domain.Petclinic.Pet;
import is.codion.demos.petclinic.domain.Petclinic.Visit;
import is.codion.framework.db.EntityConnectionProvider;
import is.codion.swing.framework.model.SwingEntityApplicationModel;
import is.codion.swing.framework.model.SwingEntityModel;

public final class PetclinicAppModel extends SwingEntityApplicationModel {

  public static final Version VERSION = Version.parse(PetclinicAppModel.class, "/version.properties");

  public PetclinicAppModel(EntityConnectionProvider connectionProvider) {
    super(connectionProvider, VERSION);
    setupEntityModels(connectionProvider);
  }

  private void setupEntityModels(EntityConnectionProvider connectionProvider) {
    SwingEntityModel ownersModel = new SwingEntityModel(Owner.TYPE, connectionProvider);
    SwingEntityModel petsModel = new SwingEntityModel(Pet.TYPE, connectionProvider);
    petsModel.editModel().initializeComboBoxModels(Pet.OWNER_FK, Pet.PET_TYPE_FK);
    SwingEntityModel visitModel = new SwingEntityModel(Visit.TYPE, connectionProvider);
    visitModel.editModel().initializeComboBoxModels(Visit.PET_FK);

    ownersModel.detailModels().add(petsModel);
    petsModel.detailModels().add(visitModel);

    ownersModel.tableModel().items().refresh();

    entityModels().add(ownersModel);
  }
}

PetclinicAppPanel

The application panel holds the EntityPanel instances used by the application, which are created in the createEntityPanels() method. These EntityPanels are based on the SwingEntityModels we get from the application model.

We also override createSupportPanelBuilders(), where we create EntityPanel.Builder instances on which to base the Support tables main menu.

package is.codion.demos.petclinic.ui;

import is.codion.common.model.CancelException;
import is.codion.common.user.User;
import is.codion.demos.petclinic.domain.Petclinic;
import is.codion.demos.petclinic.domain.Petclinic.Owner;
import is.codion.demos.petclinic.domain.Petclinic.Pet;
import is.codion.demos.petclinic.domain.Petclinic.PetType;
import is.codion.demos.petclinic.domain.Petclinic.Specialty;
import is.codion.demos.petclinic.domain.Petclinic.Vet;
import is.codion.demos.petclinic.domain.Petclinic.VetSpecialty;
import is.codion.demos.petclinic.domain.Petclinic.Visit;
import is.codion.demos.petclinic.model.PetclinicAppModel;
import is.codion.demos.petclinic.model.VetSpecialtyEditModel;
import is.codion.plugin.flatlaf.intellij.themes.arc.Arc;
import is.codion.swing.framework.model.SwingEntityModel;
import is.codion.swing.framework.ui.EntityApplicationPanel;
import is.codion.swing.framework.ui.EntityPanel;
import is.codion.swing.framework.ui.ReferentialIntegrityErrorHandling;

import java.util.List;
import java.util.Locale;

public final class PetclinicAppPanel extends EntityApplicationPanel<PetclinicAppModel> {

  public PetclinicAppPanel(PetclinicAppModel appModel) {
    super(appModel);
  }

  @Override
  protected List<EntityPanel> createEntityPanels() {
    SwingEntityModel ownersModel = applicationModel().entityModels().get(Owner.TYPE);
    SwingEntityModel petsModel = ownersModel.detailModels().get(Pet.TYPE);
    SwingEntityModel visitsModel = petsModel.detailModels().get(Visit.TYPE);

    EntityPanel ownersPanel = new EntityPanel(ownersModel,
            new OwnerEditPanel(ownersModel.editModel()));
    EntityPanel petsPanel = new EntityPanel(petsModel,
            new PetEditPanel(petsModel.editModel()));
    EntityPanel visitsPanel = new EntityPanel(visitsModel,
            new VisitEditPanel(visitsModel.editModel()));

    ownersPanel.detailPanels().add(petsPanel);
    petsPanel.detailPanels().add(visitsPanel);

    return List.of(ownersPanel);
  }

  @Override
  protected List<EntityPanel.Builder> createSupportEntityPanelBuilders() {
    EntityPanel.Builder petTypePanelBuilder =
            EntityPanel.builder(PetType.TYPE)
                    .editPanel(PetTypeEditPanel.class)
                    .caption("Pet types");
    EntityPanel.Builder specialtyPanelBuilder =
            EntityPanel.builder(Specialty.TYPE)
                    .editPanel(SpecialtyEditPanel.class)
                    .caption("Specialties");

    SwingEntityModel.Builder vetSpecialtyModelBuilder =
            SwingEntityModel.builder(VetSpecialty.TYPE)
                    .editModel(VetSpecialtyEditModel.class);
    SwingEntityModel.Builder vetModelBuilder =
            SwingEntityModel.builder(Vet.TYPE)
                    .detailModel(vetSpecialtyModelBuilder);

    EntityPanel.Builder vetSpecialtyPanelBuilder =
            EntityPanel.builder(vetSpecialtyModelBuilder)
                    .editPanel(VetSpecialtyEditPanel.class)
                    .caption("Specialty");
    EntityPanel.Builder vetPanelBuilder =
            EntityPanel.builder(vetModelBuilder)
                    .editPanel(VetEditPanel.class)
                    .detailPanel(vetSpecialtyPanelBuilder)
                    .caption("Vets");

    return List.of(petTypePanelBuilder, specialtyPanelBuilder, vetPanelBuilder);
  }

  public static void main(String[] args) throws CancelException {
    Locale.setDefault(Locale.of("en", "EN"));
    ReferentialIntegrityErrorHandling.REFERENTIAL_INTEGRITY_ERROR_HANDLING
            .set(ReferentialIntegrityErrorHandling.DISPLAY_DEPENDENCIES);
    EntityApplicationPanel.builder(PetclinicAppModel.class, PetclinicAppPanel.class)
            .applicationName("Petclinic")
            .applicationVersion(PetclinicAppModel.VERSION)
            .domainType(Petclinic.DOMAIN)
            .displayStartupDialog(false)
            .defaultLookAndFeel(Arc.class)
            .defaultLoginUser(User.parse("scott:tiger"))
            .start();
  }
}

Domain unit test

package is.codion.demos.petclinic.domain;

import is.codion.demos.petclinic.domain.Petclinic.Owner;
import is.codion.demos.petclinic.domain.Petclinic.Pet;
import is.codion.demos.petclinic.domain.Petclinic.PetType;
import is.codion.demos.petclinic.domain.Petclinic.Specialty;
import is.codion.demos.petclinic.domain.Petclinic.Vet;
import is.codion.demos.petclinic.domain.Petclinic.VetSpecialty;
import is.codion.demos.petclinic.domain.Petclinic.Visit;
import is.codion.framework.domain.test.DomainTest;

import org.junit.jupiter.api.Test;

public final class PetclinicTest extends DomainTest {

  public PetclinicTest() {
    super(new Petclinic());
  }

  @Test
  void vet() {
    test(Vet.TYPE);
  }

  @Test
  void specialty() {
    test(Specialty.TYPE);
  }

  @Test
  void vetSpecialty() {
    test(VetSpecialty.TYPE);
  }

  @Test
  void petType() {
    test(PetType.TYPE);
  }

  @Test
  void owner() {
    test(Owner.TYPE);
  }

  @Test
  void pet() {
    test(Pet.TYPE);
  }

  @Test
  void visit() {
    test(Visit.TYPE);
  }
}

Module Info

/**
 * Petclinic demo.
 */
module is.codion.demos.petclinic {
  requires is.codion.swing.framework.ui;
  requires is.codion.plugin.flatlaf;
  requires is.codion.plugin.flatlaf.intellij.themes;

  exports is.codion.demos.petclinic.model
          to is.codion.swing.framework.model, is.codion.swing.framework.ui;
  exports is.codion.demos.petclinic.ui
          to is.codion.swing.framework.ui;

  provides is.codion.framework.domain.Domain
          with is.codion.demos.petclinic.domain.Petclinic;
}

Build

settings.gradle
plugins {
    id("org.gradle.toolchains.foojay-resolver-convention") version "0.8.0"
}

rootProject.name = "petclinic"

dependencyResolutionManagement {
    repositories {
        mavenCentral()
        mavenLocal()
    }
    versionCatalogs {
        libs {
            version("codion", "0.18.25")
            version("h2", "2.3.232")

            library("codion-dbms-h2", "is.codion", "codion-dbms-h2").versionRef("codion")
            library("codion-framework-domain-test", "is.codion", "codion-framework-domain-test").versionRef("codion")
            library("codion-framework-db-local", "is.codion", "codion-framework-db-local").versionRef("codion")
            library("codion-swing-framework-ui", "is.codion", "codion-swing-framework-ui").versionRef("codion")
            library("codion-plugin-logback-proxy", "is.codion", "codion-plugin-logback-proxy").versionRef("codion")
            library("codion-plugin-flatlaf", "is.codion", "codion-plugin-flatlaf").versionRef("codion")
            library("codion-plugin-flatlaf-intellij-themes", "is.codion", "codion-plugin-flatlaf-intellij-themes").versionRef("codion")

            library("h2", "com.h2database", "h2").versionRef("h2")
        }
    }
}
build.gradle.kts
import org.gradle.internal.os.OperatingSystem

plugins {
    // The Badass Jlink Plugin provides jlink and jpackage
    // functionality and applies the java application plugin
    // https://badass-jlink-plugin.beryx.org
    id("org.beryx.jlink") version "3.1.1"
    // Just for managing the license headers
    id("com.diffplug.spotless") version "7.0.1"
    // For the asciidoctor docs
    id("org.asciidoctor.jvm.convert") version "4.0.4"
}

dependencies {
    // The Codion framework UI module, transitively pulls in all required
    // modules, such as the model layer and the core database module
    implementation(libs.codion.swing.framework.ui)
    // Include all the standard Flat Look and Feels and a bunch of IntelliJ
    // theme based ones, available via the View -> Select Look & Feel menu
    implementation(libs.codion.plugin.flatlaf)
    implementation(libs.codion.plugin.flatlaf.intellij.themes)

    // Provides the Logback logging library as a transitive dependency
    // and provides logging configuration via the Help -> Log menu
    runtimeOnly(libs.codion.plugin.logback.proxy)
    // Provides the local JDBC connection implementation
    runtimeOnly(libs.codion.framework.db.local)
    // The H2 database implementation
    runtimeOnly(libs.codion.dbms.h2)
    // And the H2 database driver
    runtimeOnly(libs.h2)

    // The domain model unit test module
    testImplementation(libs.codion.framework.domain.test)
}

// The application version simply follows the Codion framework version used
version = libs.versions.codion.get().replace("-SNAPSHOT", "")

java {
    toolchain {
        // Use the latest possible Java version
        languageVersion.set(JavaLanguageVersion.of(23))
    }
}

spotless {
    // Just the license headers
    java {
        licenseHeaderFile("${rootDir}/license_header").yearSeparator(" - ")
    }
    format("javaMisc") {
        target("src/**/package-info.java", "src/**/module-info.java")
        licenseHeaderFile("${rootDir}/license_header", "\\/\\*\\*").yearSeparator(" - ")
    }
}

testing {
    suites {
        val test by getting(JvmTestSuite::class) {
            useJUnitJupiter()
            targets {
                all {
                    // System properties required for running the unit tests
                    testTask.configure {
                        // The JDBC url
                        systemProperty("codion.db.url", "jdbc:h2:mem:h2db")
                        // The database initialization script
                        systemProperty("codion.db.initScripts", "classpath:create_schema.sql")
                        // The user to use when running the tests
                        systemProperty("codion.test.user", "scott:tiger")
                    }
                }
            }
        }
    }
}

// Configure the application plugin, the jlink plugin relies
// on this configuration when building the runtime image
application {
    mainModule = "is.codion.demos.petclinic"
    mainClass = "is.codion.demos.petclinic.ui.PetclinicAppPanel"
    applicationDefaultJvmArgs = listOf(
        // This app doesn't require a lot of memory
        "-Xmx64m",
        // Specify a local JDBC connection
        "-Dcodion.client.connectionType=local",
        // The JDBC url
        "-Dcodion.db.url=jdbc:h2:mem:h2db",
        // The database initialization script
        "-Dcodion.db.initScripts=classpath:create_schema.sql",
        // Just in case we're debugging in Linux, nevermind
        "-Dsun.awt.disablegrab=true"
    )
}

tasks.withType<JavaCompile>().configureEach {
    options.encoding = "UTF-8"
    options.isDeprecation = true
}

// Configure the docs generation
tasks.asciidoctor {
    inputs.dir("src")
    baseDirFollowsSourceFile()
    attributes(
        mapOf(
            "codion-version" to project.version,
            "source-highlighter" to "prettify",
            "tabsize" to "2"
        )
    )
    asciidoctorj {
        setVersion("2.5.13")
    }
}

// Create a version.properties file containing the application version
tasks.register<WriteProperties>("writeVersion") {
    destinationFile = file("${temporaryDir.absolutePath}/version.properties")
    property("version", libs.versions.codion.get().replace("-SNAPSHOT", ""))
}

// Include the version.properties file from above in the
// application resources, see usage in PetclinicAppModel
tasks.processResources {
    from(tasks.named("writeVersion"))
}

// Configure the Jlink plugin
jlink {
    // Specify the jlink image name
    imageName = project.name
    // The options for the jlink task
    options = listOf(
        "--strip-debug",
        "--no-header-files",
        "--no-man-pages",
        // Add the modular runtimeOnly dependencies, which are handled by the ServiceLoader.
        // These don't have an associated 'requires' clause in module-info.java
        // and are therefore not added automatically by the jlink plugin.
        "--add-modules",
        // The local JDBC connection implementation
        "is.codion.framework.db.local," +
                // The H2 database implementation
                "is.codion.dbms.h2," +
                // The Logback plugin
                "is.codion.plugin.logback.proxy"
    )

    // H2 database uses slf4j, but is non-modular so the jlink plugin,
    // can't derive that dependency, so here we help it along.
    addExtraDependencies("slf4j-api")

    jpackage {
        if (OperatingSystem.current().isLinux) {
            icon = "src/main/icons/petclinic.png"
            installerOptions = listOf(
                "--linux-shortcut"
            )
        }
        if (OperatingSystem.current().isWindows) {
            icon = "src/main/icons/petclinic.ico"
            installerOptions = listOf(
                "--win-menu",
                "--win-shortcut"
            )
        }
    }
}

// Copies the documentation to the Codion github pages repository, nevermind
tasks.register<Sync>("copyToGitHubPages") {
    group = "documentation"
    from(tasks.asciidoctor)
    into("../codion-pages/doc/" + project.version + "/tutorials/petclinic")
}