Skip to main content
This page explains the core data model behind the transaction-based Policy API. Understanding these concepts will help you predict how the API behaves when you create, endorse, cancel, and reinstate policies.

Transaction Model

Every change to a policy is recorded as an immutable transaction. Transactions are the single source of truth — policy state is always derived from them, never authored directly.
1

Configure

Define your fields, option sets, and exposure types via the Configuration API.
2

Create

POST /transaction/new-business creates a policy with initial state covering the full term.
3

Endorse

POST /{policyId}/transaction/endorse modifies the policy — add exposures, change field values, adjust coverage.
4

Cancel

POST /{policyId}/transaction/cancel cancels the policy from a specified date.
5

Reinstate

POST /{policyId}/transaction/reinstate reinstates a cancelled policy.
6

Renew

POST /transaction/renew starts a new policy term linked to the previous one.

Transaction Types

NEW_BUSINESS

Creates the policy. The effective date is the policy start date. Produces one segment covering the full term.

ENDORSE

Modifies policy state from a given effective date. Accepts an array of field-level deltas. May split existing segments or merge them if the change aligns state across periods.

CANCEL

Sets policyStatus to "Cancelled" from the cancellation date through end of term. Optionally accepts billing-only deltas (e.g., short-rate penalties).

REINSTATE

Sets policyStatus back to "Active" from the reinstatement date. Optionally accepts billing-only deltas (e.g., reinstatement fees).

RENEW

Creates a new policy term linked to the previous via previousPolicyId. Accepts a full fieldModelV1Data payload — the caller provides the complete initial state for the new term.

Effective Date vs Transaction Timestamp

Each transaction has two timestamps:
  • effectiveDate — when the policy state changes. For an endorsement, this is when the new coverage begins. For a cancellation, this is when coverage ends.
  • transactionTimestamp — when the business decision was made. Defaults to the current time, but can be set explicitly for imports (e.g., aligning to a bordereau booking date).

Deltas

A delta is a single field-level change within a transaction. Endorsements carry one or more deltas that describe exactly what changed, where, and for what date range.

Delta Structure

FieldTypeDescription
pathstringDot-notation path to the field being changed
actionstringModify, Add, or Remove
valueanyThe new value, value to add, or value to remove
startDatestringStart of the date range this change applies to
endDatestringEnd of the date range this change applies to

Delta Actions

Modify

Replace a scalar value or an entire object. This is the most common action.
{
  "path": "policy.exposures[exp-1].bedCount",
  "action": "Modify",
  "value": 110
}

Add

Append to a collection. Uses set semantics — objects are matched by id, primitives by equality. If the value already exists, the delta is a no-op.
{
  "path": "policy.exposures[exp-1].coveredSpecialties",
  "action": "Add",
  "value": "Neurology"
}

Remove

Remove from a collection. Same matching rules as Add. If the value is not present, the delta is a no-op.
{
  "path": "policy.exposures[exp-1].namedPhysicians",
  "action": "Remove",
  "value": "Dr. Nguyen"
}

Path Notation

Paths use dot-notation to target fields at any depth. Bracket syntax targets specific items in an array by their id field.
PathTargets
policy.deductibleA scalar field on the policy
policy.exposures[exp-1].bedCountA field on a specific exposure
policy.exposuresThe exposures collection itself (for Add/Remove of entire exposure objects)
policy.fullTermPolicyBillingA cross-segment invariant (same value in every segment of the version)
policy.policyStatusThe policy status (used internally by Cancel/Reinstate)
Modify on an indexed path (e.g., policy.exposures[exp-1]) replaces the entire exposure object. Add on a collection path (e.g., policy.exposures) appends an exposure. These are different operations targeting different levels.

Example: Endorsement with Deltas

An endorsement effective April 1 that adds a new exposure and updates billing:
{
  "policyId": "550e8400-e29b-41d4-a716-446655440001",
  "effectiveDate": "2025-04-01",
  "deltas": [
    {
      "startDate": "2025-04-01",
      "endDate": "2025-12-31",
      "path": "policy.exposures",
      "action": "Add",
      "value": {
        "id": "exp-2",
        "exposureType": "OutpatientClinic",
        "facilityName": "Greenfield West Clinic",
        "bedCount": 0,
        "coveredSpecialties": ["Dermatology", "Family Medicine"],
        "namedPhysicians": ["Dr. Kim"]
      }
    },
    {
      "startDate": "2025-01-01",
      "endDate": "2025-12-31",
      "path": "policy.fullTermPolicyBilling",
      "action": "Modify",
      "value": {
        "policyPremium": 98000,
        "policyTaxes": 4900,
        "policyFees": 500,
        "policyGrandTotal": 103400
      }
    }
  ]
}
The exposure delta applies from April 1 through the end of the term and will split the existing segment at that boundary. The fullTermPolicyBilling delta spans the full policy term (Jan 1 – Dec 31) because it’s a cross-segment invariant — the system applies it identically to every segment.

Segments

A segment is a maximal contiguous date range where the policy state is identical.
Segments are NOT one-to-one with transactions. A single endorsement may split one segment into many. Backdated corrections can merge segments back together. Six transactions can produce two segments — or one.

Segment Properties

Every version’s segments satisfy three invariants:
  • No overlaps — segments never share a date
  • Full coverage — segments span the entire policy term with no gaps
  • No adjacent duplicates — adjacent segments with identical state are automatically merged

How Segments Change

A new policy starts with one segment covering the full term.
#Date RangeExposures
1Jan 1 – Dec 31Main Hospital (120 beds)
Segments reflect the final state of the policy, not its change history. The full audit trail is preserved in the transaction history.

Versions

Each transaction produces a new policy version. A version is a complete snapshot — it contains the full set of segments representing the policy at that point in the transaction history.
PropertyDescription
policyVersionSequential integer (1, 2, 3, …)
transactionIdThe transaction that produced this version
segmentsComplete set of segments for this version
startDate / endDatePolicy term boundaries
You can query any historical version to see what the policy looked like after a specific transaction.

How It Works

When you submit a transaction, the system:
  1. Loads the previous version — gets the current segments
  2. Applies deltas — applies each delta to all segments whose date range overlaps the delta’s date range
  3. Normalizes — produces deterministic JSON and computes a hash for each resulting segment
  4. Merges — collapses adjacent segments with identical hashes into one
  5. Persists — stores the new version with its segments
You don’t need to understand the internal algorithm to use the API — just know that the system automatically handles segment splitting and merging based on the deltas you submit.
No-op deltas are safe. Adding a value that already exists or removing one that’s already gone has no effect. This means a delta that spans a wide date range may change some segments and leave others untouched — the system handles it correctly.

Premiums and Rating

The API does not calculate premiums. When you submit a transaction, you supply field values and billing totals yourself. The system stores what you send — it does not rate, pro-rate, or re-aggregate. Policy financial data lives at two levels:

Full-Term Billing

fullTermPolicyBilling is a cross-segment invariant — identical across every segment in a version. It contains the billing summary for the entire policy term: policyPremium, policyTaxes, policyFees, policyGrandTotal. Every endorsement should include a fullTermPolicyBilling delta spanning the full policy term to keep billing current.

Per-Segment Rating

Each segment can carry its own rating data via PolicyRatingResponse (on the policy) and ExposureRatingResponse (on each exposure). These typically include:
  • annualPremium — the premium as if that segment’s state applied for the full year
  • proratedPremium — the actual premium contribution for that time slice
Per-segment rating provides visibility into which exposures contribute how much premium over which time periods.
fullTermPolicyBilling is not necessarily the sum of prorated slices. Full-term billing can include flat premium minimums, surplus lines taxes, policy fees, or other adjustments that are independent of per-segment proration. The FullTermPolicyRatingResponse (a separate cross-segment field) captures these aggregate rating details.
Callers don’t have to pass per-segment field changes. You could submit endorsements that only modify fullTermPolicyBilling and leave per-segment data untouched. However, this reduces the system to importing bordereau — you’d know the billing changed, but not what policy details produced the change. Per-segment deltas capture the actual changing characteristics of the policy over time.
When using the application UI (not headless) with a rater configured, an aggregation rater automatically computes billing and rating totals across segments. Through the API, this is your responsibility.

Worked Example

A medical facility policy (Greenfield Medical Center) for Jan 1 – Dec 31 with one exposure. This shows the core segment behaviors — splitting, maintaining, and merging — with the actual delta payloads. Each endorsement also includes a fullTermPolicyBilling update (omitted from the segment tables since it’s the same in every segment).
Create the policy with initial state spanning the full term. Grand total: 89,750.
{
  "policyStartDate": "2025-01-01",
  "policyEndDate": "2025-12-31",
  "fieldModelV1Data": {
    "policy": {
      "policyStatus": "Active",
      "fullTermPolicyBilling": {
        "policyPremium": 85000, "policyTaxes": 4250,
        "policyFees": 500, "policyGrandTotal": 89750
      },
      "exposures": [{
        "id": "exp-1",
        "exposureType": "MedicalFacility",
        "facilityName": "Greenfield Main Campus",
        "bedCount": 120,
        "coveredSpecialties": ["Cardiology", "Orthopedics", "General Surgery"],
        "namedPhysicians": ["Dr. Patel", "Dr. Nguyen", "Dr. Hoffman"]
      }]
    }
  }
}
Segments:
#Date RangeExposures
1Jan 1 – Dec 31Main Campus (120 beds, 3 physicians)
A satellite clinic opens. The Add action appends a new exposure to the collection. Grand total increases to 103,400 (+13,650) — the additional exposure adds risk for the remaining 9 months.
{
  "effectiveDate": "2025-04-01",
  "deltas": [
    {
      "startDate": "2025-04-01", "endDate": "2025-12-31",
      "path": "policy.exposures",
      "action": "Add",
      "value": {
        "id": "exp-2", "exposureType": "OutpatientClinic",
        "facilityName": "Greenfield West Clinic", "bedCount": 0,
        "coveredSpecialties": ["Dermatology", "Family Medicine"],
        "namedPhysicians": ["Dr. Kim"]
      }
    },
    {
      "startDate": "2025-01-01", "endDate": "2025-12-31",
      "path": "policy.fullTermPolicyBilling",
      "action": "Modify",
      "value": { "policyPremium": 98000, "policyTaxes": 4900,
                 "policyFees": 500, "policyGrandTotal": 103400 }
    }
  ]
}
Segments:
#Date RangeExposures
1Jan 1 – Mar 31Main Campus
2Apr 1 – Dec 31Main Campus + West Clinic
The original segment split at April — different exposure count on each side.
A new surgeon joins the main campus. Neurology added as a covered specialty. Grand total increases to 111,800 (+8,400) — the additional physician and expanded specialty coverage increase risk.
{
  "effectiveDate": "2025-06-01",
  "deltas": [
    {
      "startDate": "2025-06-01", "endDate": "2025-12-31",
      "path": "policy.exposures[exp-1].namedPhysicians",
      "action": "Add", "value": "Dr. Okafor"
    },
    {
      "startDate": "2025-06-01", "endDate": "2025-12-31",
      "path": "policy.exposures[exp-1].coveredSpecialties",
      "action": "Add", "value": "Neurology"
    },
    {
      "startDate": "2025-01-01", "endDate": "2025-12-31",
      "path": "policy.fullTermPolicyBilling",
      "action": "Modify",
      "value": { "policyPremium": 106000, "policyTaxes": 5300,
                 "policyFees": 500, "policyGrandTotal": 111800 }
    }
  ]
}
Segments:
#Date RangePhysiciansSpecialties
1Jan 1 – Mar 31Patel, Nguyen, HoffmanCardiology, Orthopedics, Surgery
2Apr 1 – May 31Patel, Nguyen, HoffmanCardiology, Orthopedics, Surgery
3Jun 1 – Dec 31Patel, Nguyen, Hoffman, OkaforCardiology, Orthopedics, Surgery, Neurology
The Apr–Dec segment from version 2 split at the June boundary.
Internal audit reveals the physician change, bed reduction, and Neurology addition should all have been effective April 1, not June 1. A single endorsement corrects everything retroactively. Grand total decreases to 106,550 (-5,250) — the physician departure and bed reduction reduce risk, partially offset by Neurology covering a longer period.
{
  "effectiveDate": "2025-04-01",
  "deltas": [
    {
      "startDate": "2025-04-01", "endDate": "2025-12-31",
      "path": "policy.exposures[exp-1].bedCount",
      "action": "Modify", "value": 110
    },
    {
      "startDate": "2025-04-01", "endDate": "2025-12-31",
      "path": "policy.exposures[exp-1].namedPhysicians",
      "action": "Remove", "value": "Dr. Nguyen"
    },
    {
      "startDate": "2025-04-01", "endDate": "2025-12-31",
      "path": "policy.exposures[exp-1].namedPhysicians",
      "action": "Add", "value": "Dr. Okafor"
    },
    {
      "startDate": "2025-04-01", "endDate": "2025-12-31",
      "path": "policy.exposures[exp-1].coveredSpecialties",
      "action": "Add", "value": "Neurology"
    },
    {
      "startDate": "2025-01-01", "endDate": "2025-12-31",
      "path": "policy.fullTermPolicyBilling",
      "action": "Modify",
      "value": { "policyPremium": 101000, "policyTaxes": 5050,
                 "policyFees": 500, "policyGrandTotal": 106550 }
    }
  ]
}
All four per-segment deltas span Apr 1 – Dec 31. The Jun–Dec segment already had beds=110, Nguyen removed, Okafor present, and Neurology — those deltas are no-ops there. Only Apr–May changes. After applying, Apr–May and Jun–Dec have converged to identical state. The system merges them:Segments:
#Date RangeBedsPhysiciansSpecialties
1Jan 1 – Mar 31120Patel, Nguyen, HoffmanCardiology, Orthopedics, Surgery
2Apr 1 – Dec 31110Patel, Hoffman, OkaforCardiology, Orthopedics, Surgery, Neurology
Four transactions, two segments. The Apr–May / Jun–Dec boundary vanished — not because a transaction was reversed, but because the correction converged the per-segment state on both sides. fullTermPolicyBilling didn’t affect the merge — it’s the same in every segment.
For the complete 9-transaction walkthrough — including cancellation, reinstatement, transaction deletion, and both types of segment merging — see the Lifecycle Walkthrough.

Cancellation and Reinstatement

Cancel and reinstate modify only the policyStatus field. They are simpler than endorsements but follow the same segment mechanics.
  • Cancel sets policyStatus to "Cancelled" from the cancellation date through end of term. This typically splits the existing segment at the cancellation boundary.
  • Reinstate sets policyStatus back to "Active" from the reinstatement date. If this makes the reinstated segment identical to its predecessor, they merge.
  • Both optionally accept billing-only deltas (restricted to fullTermPolicyBilling paths) for short-rate penalties or reinstatement fees.
A cancel followed by a reinstate at the same date is invisible in the final segments — the policy returns to the same state it had before cancellation. Both transactions are preserved in the audit trail, but the derived segments are identical to the pre-cancellation version.

Transaction Deletion

Only the most recent transaction on a policy can be deleted. Deleting a transaction:
  • Produces a new version derived from the prior transaction’s state
  • Preserves the audit trail — the deleted transaction is archived, not erased
  • Is irreversible through the API once deleted (the transaction can be re-created manually)
Transaction deletion undoes the most recent change. It does not allow arbitrary transaction removal from the middle of the history.