Issue Subsystem

Basics

An issue describes a condition in a model. Issues are categorized by severity in the following way

  • Fatal issues indicate situations which should not occur in normal operation

  • Error issues indicate a severe issue which blocks calculation

  • Warning issues point out a possible error

  • Info issues are good to know information

  • Note issues are markers for documentation purposes

Severity is modeled using ISSUE.HasSeverity and ISSUE.Severity.

Issues are somewhat related to events. The main difference between issues and events is that while events happen at a particular time, issues are conditions with a life-cycle. The different issue states are

  • Active (default), which means that the condition is true

  • Resolved (ISSUE.Resolved tag), which means that the condition is no longer true

Further issues can be categorized as

  • User issues (ISSUE.UserIssue tag), which indicates that the user has manually created the condition

  • Hidden issues (ISSUE.Hidden tag), which can be used as a hint for user interfaces.

The following also applies for issues

  • Issues are directly linked to a model with L0.ConsistsOf

  • Issues are given unique and harmless names (e.g. UUIDs)

Issues are identified by the following data

  • A type resource (L0.InstanceOf)

  • A list of context resources (ISSUE.Issue.HasContexts), where the first resource is a main context

All issues declare the following properties needed to e.g. display the issue in the Issue View.

  • Description (L0.HasDescription), which is a one-line text about the issue

  • Resource (ISSUE.Issue.resource), which is a textual representation of the main context resource

  • Path (ISSUE.Issue.path), which is a textual representation of the location of the main context resource

  • Severity (ISSUE.Issue.severity), which is a textual representation of issue severity

  • Creation time (ISSUE.HasCreationTime), which is a textual representation of time of creation of the issue

Typically these are computational properties and e.g. the Description value is asserted in the issue type and computed from context resources.

Issues View

The Issues view shows all issues in the active model(s). Active models are determined from the set of models linked to the current project with L0X.Activates.

User issues

User issues are created by

  • org.simantics.issues.common.IssueUtils.newUserIssue

  • org.simantics.issues.common.IssueUtils.newUserIssueForModel

User issues are removed by

  • org.simantics.db.layer0.util.RemoverUtil.remove

Browsing

For example the following requests are defined for browsing issues

  • org.simantics.issues.common.AllIssues

  • org.simantics.issues.common.AllActiveIssues

  • org.simantics.issues.common.<severity>Issues

  • org.simantics.issues.common.IssuesOfSeverity

Computational issues

The life-cycle of computational issues is determined by issue sources. Issue sources are divided into the following categories

  • Batch issue sources, which can compute on demand a set of issues from given context

  • Continuous issue sources, which can track the set of issues from given context

Batch issue sources

Batch issue source instances (ISSUE.BatchIssueSource) adapt to the interface org.simantics.issues.common.BatchIssueSource.

The code org.simantics.issues.ui.handler.RunActiveValidations performs active batch validations. Currently batch issue sources are searched from the project resource by relation L0X.Activates. The activations are put in place using project features that link the project to the issue sources in the ontology.

Continuous issue sources

Continuous issue source instances (ISSUE.IssueSource with ISSUE.IsContinuous) adapt to the interface org.simantics.issues.common.IssueSource.

Currently the only supported implementation of IssueSource is org.simantics.issues.common.DependencyIssueSource2. The implementation is defined for ISSUE.Sources.DependencyTracker. It uses

  • DependencyChanges metadata for determining which resources need revalidation. The set of changed resources from DependencyChanges is expanded to include all resources reachable by L0.IsDependencyOf.

  • ISSUE.Sources.DependencyTracker.HasType for determining which resource instances to validate

  • ISSUE.Sources.DependencyTracker.HasSearchType for determining which resource to traverse after changes and before applying the extension.

  • ISSUE.Sources.DependencyTracker.HasExtension for applying an extension function for the set of potentially changed resources

  • ISSUE.Sources.DependencyTracker.HasBaseFunction for determining a resource under which changes are tracked (by default before this function the base resource is the L0.PartOf of the issue source)

  • ISSUE.Sources.DependencyTracker.HasConstraint for defining the test to be performed on the selected resources

The main codes for continuous validation are

  • org.simantics.issues.common.IssueUtils.listenActiveProjectIssueSources

  • org.simantics.issues.common.DependencyIssueValidator2 for validating a context resource

  • org.simantics.issues.common.DependencyIssueSynchronizer2 for synchronizing the model

In practice, the developer needs to define

  • A set of issue types with asserted properties (with computation) and severity

  • A set of constraints assigned to types and associated issue sources

  • Possible extension functions for ensuring proper updates for all relevant changes

The issue ontology supplies some templates for modelling typical cases

Miscellaneous code

The issue model can be manipulated using org.simantics.issues.common.Issue and especially org.simantics.issues.common.StandardIssue. The interface and implementation supports manipulation, comparison and storing of issues into database. StandardIssue assumes that all properties of an issue are determined by assertions in the issue type (i.e. the issue is fully determined by type, contexts, name and creation time)

Some utility codes can be found in

  • org.simantics.issues.common.IssueUtils

A concrete example "SSP Simulator Connection" issue definition

In this section, we'll walk you through a small example of all the necessary files and parts you need when you want a simple issue to be shown in your Simantics desktop product, for a specific resource type.

Step 1: Ensure you have your base ontology and a Functions library:

org.ththry.sspsimulator.ui.ontology/graph/SSPViewpoint.pgraph

L0 = <http://www.simantics.org/Layer0-1.1>
    
SSPUI = <http://org.ththry.sspsimulator/ui-1.0> : L0.Ontology
    @L0.new
    L0.HasResourceClass "org.ththry.sspsimulator.ui.ontology.SSPBuiltins"

SSPUI.Functions : L0.Library

Step 2: Define your function and reference a Java function that performs the issue validation (we will define this later)

org.ththry.sspsimulator.ui.ontology/graph/Functions.pgraph

L0 = <http://www.simantics.org/Layer0-1.1>
SSPUI = <http://org.ththry.sspsimulator/ui-1.0>

SSPUI.Functions.mismatchedConnectionTypeValidator : L0.Function
  • NOTE: We define the function name mismatchedConnectionTypeValidator which must match the name of the Java function we will reference later.

Step 3: Define the main validation ontology.

org.ththry.sspsimulator.ui.ontology/graph/Validations.pgraph

L0 = <http://www.simantics.org/Layer0-1.1>
L0X = <http://www.simantics.org/Layer0X-1.1>
ISSUE = <http://www.simantics.org/Issue-1.2>

SSPUI = <http://org.ththry.sspsimulator/ui-1.0>

VALIDATIONS = SSPUI.Validations : L0.Library
    L0.HasResourceClass "org.ththry.sspsimulator.ui.ontology.SSPValidations"

// This is a template
VALIDATIONS.constraint : L0.Template
    @template %type %constraint %source %validator
        %type
            L0.HasConstraint %constraint : L0.Constraint
                L0.Constraint.Validator %validator
        %source <T ISSUE.Sources.ListeningDependencyTracker
            L0.Asserts _ : L0.Assertion
                L0.HasPredicate ISSUE.Sources.DependencyTracker.HasType
                L0.HasObject %type
            L0.Asserts _ : L0.Assertion
                L0.HasPredicate ISSUE.Sources.DependencyTracker.HasConstraint
                L0.HasObject %constraint

// Constraints

SSP.ComplexTypes.TSystem.Connections.Connection
    @VALIDATIONS.constraint
        VALIDATIONS.MismatchedConnectionTypeConstraint 
        VALIDATIONS.MismatchedConnectionTypeConstraint.Source 
        SSPUI.Functions.mismatchedConnectionTypeValidator

VALIDATIONS.MismatchedConnectionTypeConstraint.Source 
  L0.HasLabel "Mismatched types"

VALIDATIONS.MismatchedConnectionTypeConstraint.Source
  @L0.assert ISSUE.Sources.DependencyTracker.HasSearchType SSP.ComplexTypes.TSystem.Connections.Connection

// Issues
VALIDATIONS.MismatchedConnectionTypeConstraint.WarningIssue
    @ISSUE.issue ISSUE.Severity.Warning "Connectors have mismatching type."
  • NOTE 1: We create a new library Validations under your SSPUI ontology, just like we did with Functions.

  • NOTE 2: We define a reusable template (VALIDATIONS.constraint) which makes it easier to define constrains later

  • NOTE 3: We use the template to create for the resource type SSP.ComplexTypes.TSystem.Connections.Connection a constraint

  • NOTE 4: We define the constrain and source, as well as reference our function mismatchedConnectionTypeValidator in this constraint.

  • NOTE 5: We need to define the source of the constraint and give it a label, and with an assert inject a HasSearchType for our resource type (SSP.ComplexTypes.TSystem.Connections.Connection)

  • NOTE 6: We need to create a new issue type so that we can create issues of that type from our Java function (which we have not yet defined)

Step 4: Create your Java function class "Func.java"

package org.ththry.sspsimulator.ui.functions;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;

import org.simantics.db.Issue;
import org.simantics.db.ReadGraph;
import org.simantics.db.Resource;
import org.simantics.db.common.issue.StandardIssue;
import org.simantics.db.exception.DatabaseException;
import org.simantics.scl.reflection.annotations.SCLValue;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.ththry.sspsimulator.ui.ontology.SSPBuiltins;

public class Func {

    @SCLValue(type = "ReadGraph -> Resource -> [Issue]")
    public static List<Issue> mismatchedConnectionTypeValidator(ReadGraph graph, Resource connection) throws DatabaseException {
        SSPBuiltins ssp = SSPBuiltins.getInstance(graph);
        Resource type = ssp.Validations_MismatchedConnectionTypeConstraint_WarningIssue;

        if (/* test if there is a mismatch */) {
            return List.of(new StandardIssue(type, connection));
        }
        return List.of();
    }
}
  • NOTE 1: The name of the function "mismatchedConnectionTypeValidator" must match what you defined in your Functions.pgraph and referenced in your Validations.pgraph

  • NOTE 2: In this example, we don't actually show any logic that checks if the Resource fails or succeeds its issue: This example ALWAYS creates an issue for the resource. In your actual code, you should implement logic that checks the situation you were interested in, and only create issues if the situation occurs.

  • NOTE 3: In this example, we're creating a simple "StandardIssue" that doesn't contain the "HasDescription", "Issue_resource" or "Issue_path" metadata that is populated in the Issue table's columns.

  • NOTE 4: We are referencing our Validations.pgraph ontology here, creating an issue with a specific type "Validations_MismatchedConnectionTypeConstraint_WarningIssue"

Step 5: Ensure you have a project feature that triggers the activation of issues

package org.ththry.sspsimulator.ui;

import org.simantics.db.Disposable;
import org.simantics.db.Session;
import org.simantics.db.exception.DatabaseException;
import org.simantics.issues.common.IssueUtils;
import org.simantics.layer0.Layer0;
import org.simantics.project.IProject;
import org.simantics.project.exception.ProjectException;
import org.simantics.project.features.AbstractProjectFeature;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class DesktopProjectFeature extends AbstractProjectFeature {

    private static final Logger LOGGER = LoggerFactory.getLogger(DesktopProjectFeature.class);

    private Disposable issueListener;

    @Override
    public void configure() throws ProjectException {
        final IProject project = getProject();
        final Session session = getSession();
        try {
            issueListener = IssueUtils.listenActiveProjectIssueSources(session, project.get());
            if(issueListener != null) {
            	LOGGER.info("Registered issueListener correctly.");
            }
        } catch (DatabaseException e) {
            throw new ProjectException(e);
        }
    }

    @Override
    public void deconfigure() throws ProjectException {
        if (issueListener != null) {
            issueListener.dispose();
            issueListener = null;
        }
    }
}
  • NOTE 1: This is a bare minimum example of an implementation of an AbstractProjectFeature

  • NOTE 2: You need to create the issueListener and remember to dispose of it later, like in the deconfigure function.

Step 6: Fix your plugin.xml

org.ththry.sspsimulator.ui/plugin.xml

<?xml version="1.0" encoding="UTF-8"?>
<plugin>
   <extension
         point="org.simantics.project.feature">
      <feature
            label="SSP Simulator Desktop Project"
            description="SSP Simulator desktop project feature."
            class="org.ththry.sspsimulator.ui.DesktopProjectFeature"
            published="true"
            id="org.ththry.sspsimulator.ui">
            <installGroup id="omnipresent" version="[0.0.0,9.0.0)"/>
      </feature>
   </extension>
   <extension
         point="org.simantics.scl.reflection.binding">
      <namespace
            path="http://org.ththry.sspsimulator/ui-1.0/Functions">
         <externalClass
               className="org.simantics.db.Issue">
         </externalClass>
         <externalClass
               className="org.simantics.db.Resource">
         </externalClass>
         <externalClass
               className="org.simantics.db.layer0.variable.Variable">
         </externalClass>
         <externalClass
               className="org.simantics.db.layer0.variable.VariableMap">
         </externalClass>
         <externalClass
               className="org.simantics.db.layer0.variable.ValueAccessor">
         </externalClass>
         <externalClass
               className="org.simantics.db.ReadGraph">
         </externalClass>
         <externalClass
               className="org.simantics.db.WriteGraph">
         </externalClass>
         <externalClass
               className="org.simantics.db.RelationContext">
         </externalClass>
         <class
               className="org.ththry.sspsimulator.ui.functions.Func">
         </class>
      </namespace>
   </extension>
</plugin>
  • NOTE 1: The plugin.xml can exist in the same plugin as your DesktopProjectFeature and Func.java

  • NOTE 2: You need to register your Project Feature. The "id=org.ththry.sspsimulator.ui" needs to match the plugin's name where your DesktopProjectFeature is located.

  • NOTE 3: You need to bind the Functions ontology and let it know where to find the implementation of your functions. This is done with the class element's className field. Make sure the className points to the path of your `Func.java``, like in the example above.

Step 7: Ensure your model activates the issue source

This step can be done in various ways, and should be done in such a way that new models created in the desktop product always have new issue sources created for them (automatically).

If you simply want to test things, you can create the necessary resources and claims with SCL. Open your SCL Console and write:

import "Simantics/All"
import "http://org.ththry.sspsimulator/ui-1.0" as UI

model = resource "http://Projects/Development%20Project/Model"
source = newResource ()
claim source L0.InstanceOf UI.Validations.MismatchedConnectionTypeConstraint.Source
claimRelatedValue source L0.HasName "Find mismatched connection data types"
claim model L0X.Activates source
claim model L0.ConsistsOf source
  • NOTE 1: The URI "http://Projects/Development%20Project/Model" should be the URI of your model so the model must have a valid URI.

  • NOTE 2: The imported ontology (http://org.ththry.sspsimulator/ui-1.0) should be your base ontology

  • NOTE 3: The instance type of the source resource should be the Source you defined in Validations.pgraph

  • NOTE 4: The name can be anything, but should describe what this resource represents (not a random uuid).

Step 8: Test your implementation

When you now create a new resource of the specificed type (specificed in Validations.pgraph), an issue will be created in the Simantics desktop "Issues" view. To see this, you need to include the org.simantics.issues.ui.feature feature in your product's features. At the time of writing (2025.02.05), you can debug the implementation by adding breakpoints to DependencyIssueSource2.java, to see if the source is triggered and the model is correctly found for the sources.