Skip to main content

Defining Availability (Alpha)

This guide covers how to configure availability using the SchedulingParameters extension — at both the actor level (per Schedule) and the service type level (via HealthcareService). It covers scheduling constraints, override behavior, timezone handling, and multi-resource scheduling patterns.

The diagram below shows how availability can be defined at both

The Scheduling Parameters Extension

All scheduling constraints are managed through a single consolidated extension: SchedulingParameters. This extension can appear on both HealthcareService (for defaults) and Schedule.

Extension Fields Reference

UrlTypeApplies ToRequiredBehavior when definedBehavior when not defined
serviceReference(HealthcareService)Schedule onlyRequiredApplies configuration only to the specified service type, overriding defaults for that serviceN/A - must be specified
availabilityNested extensionSchedule onlyOptionalBookings must fully fit within the recurring windows (day-of-week + start time + end time).Time is implicitly available by default (unless blocked by Slots or other constraints)
timezoneCodeSchedule onlyOptionalSpecifies the timezone (IANA timezone identifier, e.g., America/New_York) for interpreting availability times. Falls back to the Schedule's actor timezone if not specifiedUses the timezone defined on the Schedule's actor reference
durationDurationBoth Schedule and HealthcareServiceRequiredDetermines how long the time increments for a Slot areN/A - must be specified
bufferBeforeDurationBoth Schedule and HealthcareServiceOptionalRequires prep time before start to also be freeNo prep time required
bufferAfterDurationBoth Schedule and HealthcareServiceOptionalRequires cleanup time after end to also be freeNo cleanup time required
alignmentIntervalDurationBoth Schedule and HealthcareServiceOptionalStart times must align to the interval (e.g., every 15 minutes)Start times are not constrained by an interval grid
alignmentOffsetDurationBoth Schedule and HealthcareServiceOptional, and alignmentInterval must be definedShifts allowed start times by the offset relative to the interval (e.g., with a 15-minute alignmentInterval and a 5-minute alignmentOffset, valid starts are :05, :20, :35, :50)Grid anchored to :00 (no shift)
bookingLimitTimingBoth Schedule and HealthcareServiceOptionalCaps number of appointments per period (multiple entries can stack)No capacity cap for that period
Example of the SchedulingParameters extension
{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
// Required on Schedule: you must specify what type of appointment these parameters apply to
{
"url": "service",
"valueReference": {
"reference": "HealthcareService/5d02acfd-fbe8-4537-84e4-31f5116be105",
"display": "Bariatric Surgery"
}
},

// Optional: specify time zone for availability interpretation (Schedule only)
// Falls back to Schedule's actor time zone if not specified
{
"url": "timezone",
"valueCode": "America/Los_Angeles"
},

// Required: duration determines how long the time increments for a Slot are
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},

// Recurring availability (Schedule only)
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "mon" },
{ "url": "daysOfWeek", "valueCode": "wed" },
{ "url": "daysOfWeek", "valueCode": "fri" },
{ "url": "availableStartTime", "valueTime": "09:00:00" },
{ "url": "availableEndTime", "valueTime": "17:00:00" }
]
}
]
},

// Buffer time before appointment
{
"url": "bufferBefore",
"valueDuration": {
"value": 15,
"unit": "min",
"system": "http://unitsofmeasure.org",
"code": "min"
}
},

// Buffer time after appointment
{
"url": "bufferAfter",
"valueDuration": {
"value": 10,
"unit": "min",
"system": "http://unitsofmeasure.org",
"code": "min"
}
},

// Time alignment interval (appointment start time boundaries)
{
"url": "alignmentInterval",
"valueDuration": {
"value": 15,
"unit": "min",
"system": "http://unitsofmeasure.org",
"code": "min"
}
},

// Time alignment offset (shift from interval boundaries)
{
"url": "alignmentOffset",
"valueDuration": {
"value": 0,
"unit": "min",
"system": "http://unitsofmeasure.org",
"code": "min"
}
},

// Booking limits (can have multiple)
{
"url": "bookingLimit",
"valueTiming": {
"repeat": {
"frequency": 8,
"period": 1,
"periodUnit": "d"
}
}
}
]
}

Actor Level Availability

The Concept of Implicit Availability

Fanoni's scheduling model uses implicit availability: time is assumed to be free by default. You define availability rules using extensions that specify when resources are available based on recurring patterns. Slot resources are only used for explicit overrides—either to mark time as busy (when an appointment is booked) or to block out unavailable time.

This approach avoids the need to pre-generate thousands of Slot resources for every possible time slot. Instead, the system calculates available windows dynamically based on the availability rules you define.

The Schedule Resource

The Schedule resource is the foundation for defining actor-level availability for a provider, location, or device.

The Schedule resource should define the service types that it is capable of acting on in its serviceType attribute.

Here is an example of a Schedule resource that defines availability for a Practitioner.

{
"resourceType": "Schedule",
"id": "dr-smith-schedule",
"actor": [{"reference": "Practitioner/dr-smith"}],
"serviceType": [
{
"text": "Office Visit",
"coding": [
{ "code": "office-visit" }
],
"extension": [
{
"url": "https://medplum.com/fhir/service-type-reference",
"valueReference": {
"reference": "HealthcareService/23c3f1cc-4f55-4990-9775-511b02487e7e",
"display": "Office Visit"
}
}
]
}
],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "service",
"valueReference": {
"reference": "HealthcareService/23c3f1cc-4f55-4990-9775-511b02487e7e",
"display": "Office Visit"
}
},
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "availableStartTime", "valueTime": "09:00:00" },
{ "url": "availableEndTime", "valueTime": "17:00:00" },
{ "url": "daysOfWeek", "valueCode": "mon" },
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "wed" },
{ "url": "daysOfWeek", "valueCode": "thu" },
{ "url": "daysOfWeek", "valueCode": "fri" },
]
}
]
}
]
}]
//...
}

availability Extension

The availability sub-extension mirrors the FHIR R5+ Availability datatype shape. It is encoded using nested R4 extensions (because R4 does not have a native Availability data type). This is close to the R4 definition of HealthcareService.availabileTime, which is another possible source of scheduling availability data.

Sub-extensionTypeDescriptionRepeatable
availableTime(nested)One entry per availability windowYes
daysOfWeekvalueCodeOne entry per day (monsun)Yes
allDayvalueBooleanIf true, window spans the full dayNo
availableStartTimevalueTimeOpening time (not allowed when allDay is present)No
availableEndTimevalueTimeClosing time (not allowed when allDay is present)No
notAvailableTime(nested)Typed for future use; not yet processedYes
{
"resourceType": "Schedule",
"id": "dr-smith-schedule",
"actor": [{"reference": "Practitioner/dr-smith"}],
"serviceType": [
{
"text": "Office Visit",
"coding": [
{ "code": "office-visit" }
],
"extension":[
{
"url": "https://medplum.com/fhir/service-type-reference",
"valueReference": {
"reference": "HealthcareService/23c3f1cc-4f55-4990-9775-511b02487e7e",
"display": "Office Visit"
}
}
]
}
],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "service",
"valueReference": {
"reference": "HealthcareService/23c3f1cc-4f55-4990-9775-511b02487e7e",
"display": "Office Visit"
}
},
{
"url": "duration",
"valueDuration": {"value": 1, "unit": "h"}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "mon" },
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "wed" },
{ "url": "daysOfWeek", "valueCode": "thu" },
{ "url": "daysOfWeek", "valueCode": "fri" },
{ "url": "availableStartTime", "valueTime": "09:00:00" },
{ "url": "availableEndTime", "valueTime": "17:00:00" }
]
}
]
}
]
}]
}

Service Level Availability

Service Types and HealthcareService

An HealthcareService gives a mechanism to define the default scheduling parameters for a specific service type (appointment type), which can then be used by multiple Practitioner's Schedules. This allows you to define standard appointment durations, buffer times, alignment intervals, and booking limits once and apply them across multiple providers.

To get the Schedule to use the HealthcareService's scheduling parameters, the Schedule.serviceType must include a reference to the HealthcareService in its extensions.

{
"resourceType": "HealthcareService",
"id": "23c3f1cc-4f55-4990-9775-511b02487e7e",
"type": [{
"text": "Office Visit",
"coding": [{
"system": "http://example.org/appointment-types",
"code": "office-visit"
}]
}],
"availableTime": [
"daysOfWeek": ["mon", "tue", "wed", "thu", "fri"],
"availableStartTime": "09:00:00",
"availableEndTime": "17:00:00"
],
//...
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
}
]
}]
}
{
"resourceType": "Schedule",
"id": "dr-smith-schedule",
"actor": [{"reference": "Practitioner/dr-smith"}],
"serviceType": [
{
"text": "Office Visit",
"coding": [
{ "code": "office-visit" }
],
"extension":[
{
"url": "https://medplum.com/fhir/service-type-reference",
"valueReference": {
"reference": "HealthcareService/23c3f1cc-4f55-4990-9775-511b02487e7e",
"display": "Office Visit"
}
}
]
}
]
//...
}

Override Behavior

A Practitioner's Schedule can also override the default HealthcareService's availability for a specific service type by defining a serviceType extension.

Here is an example Schedule that overrides the availability for a specific service type.

{
"resourceType": "Schedule",
"id": "dr-chen-schedule",
"active": true,
"serviceType": [
{
"coding": [{"code": "new-patient-visit"}]
"extension":[
{
"url": "https://medplum.com/fhir/service-type-reference",
"valueReference": {
"reference": "HealthcareService/f44bbf25-bf57-4263-8f10-be060cc91672",
"display": "New Patient visit"
}
}
]
},
{
"coding": [{"code": "follow-up"}]
"extension":[
{
"url": "https://medplum.com/fhir/service-type-reference",
"valueReference": {
"reference": "HealthcareService/5f93269f-b5de-4d76-9672-e35ecb37f201",
"display": "Follow-up visit"
}
}
]
}
],
"actor": [{"reference": "Practitioner/dr-chen"}],
"extension": [
// Service type level availability for this Practitioner - overrides any availability parameters specified elsewhere
{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "service",
"valueReference": {
"reference": "HealthcareService/f44bbf25-bf57-4263-8f10-be060cc91672",
"display": "New Patient visit"
}
},
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "thu" },
{ "url": "availableStartTime", "valueTime": "09:00:00" },
{ "url": "availableEndTime", "valueTime": "13:00:00" }
]
}
]
}
]
}
]
}

All-or-nothing rule: When a SchedulingParameters extension includes a service, it completely replaces the default configuration for that service. No attribute-level merging occurs.

Priority order (highest to lowest):

  1. Schedule where https://medplum.com/fhir/StructureDefinition/SchedulingParameters.service extension matches service-type-reference parameter on $find request.
  2. HealthcareService matching service-type-reference param on $find request. HealthcareService parameters are used.

Blocking Time by Service Type

Here is an example of a Slot resource that blocks time for a specific service type.

{
"resourceType": "Slot",
"schedule": {"reference": "Schedule/dr-johnson-schedule"},
"status": "busy-unavailable",
"start": "2025-12-24T08:00:00Z",
"end": "2025-12-27T07:59:59Z",
"comment": "Holiday vacation",
"serviceType": [{"coding": [{"code": "office-visit"}]}]
}
  • With serviceType: Blocks only that specific service
  • Without serviceType: Blocks all services

Timezone per Scheduling Parameters Entry

The timezone parameter allows you to specify different timezones for different service types within the same Schedule. This is useful when a provider needs to define availability in different timezones for different services (e.g., a doctor who provides cardiac surgery where they might travel to in one time zone and call center availability in another time zone).

Fallback Logic: If no time zone is specified in the scheduling-parameters extension, then the availability will be interpreted in the time zone defined on the Schedule's actor reference (Practitioner, Location, or Device). It looks for the FHIR sanctioned time zone extension:

{
"resourceType": "Practitioner",
"id": "dr-smith",
"extension": [
{
"url": "http://hl7.org/fhir/StructureDefinition/timezone",
"valueCode": "America/Los_Angeles"
}
]
}
Adding a Timezone to an Actor

There is no native timezone field on Practitioner, Location, or Device, so you must add it via the FHIR timezone extension:

{
resourceType: 'Practitioner',
// ...
extension: [
{
url: "http://hl7.org/fhir/StructureDefinition/timezone",
valueCode: "America/Los_Angeles"
}
]
}

Timezone Resolution Order:

  1. If timezone is specified in the scheduling-parameters extension, use that time zone
  2. Otherwise, fall back to the time zone defined on the Schedule's actor reference (Practitioner, Location, or Device)

Important Notes:

  • timezone is only available on Schedule resources, not HealthcareService
  • The time zone value should be an IANA time zone identifier (e.g., America/New_York, America/Los_Angeles, America/Miami)
  • When timezone is specified, all Time values in the availability extension are interpreted in that time zone

Here is an example of a Schedule with multiple service types, each with its own time zone:

{
"resourceType": "Schedule",
"id": "dr-smith-schedule",
"actor": [{"reference": "Practitioner/dr-smith"}],
"serviceType": [
{
"text": "Cardiac Surgery",
"coding": [
{ "code": "cardiac-surgery" }
]
},
{
"text": "Call Center Availability",
"coding": [
{ "code": "call-center-availability" }
]
}
],
"extension": [
{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "serviceType",
"valueCodeableConcept": {
"coding": [{"code": "cardiac-surgery"}]
}
},
{
"url": "timezone",
"valueCode": "America/Los_Angeles"
},
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "mon" },
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "wed" },
{ "url": "availableStartTime", "valueTime": "11:00:00" }, // Interpreted in America/Los_Angeles
{ "url": "availableEndTime", "valueTime": "15:00:00" } // Interpreted in America/Los_Angeles
]
}
]
}
]
},
{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "timezone",
"valueCode": "America/New_York"
},
{
"url": "serviceType",
"valueCodeableConcept": {
"coding": [{"code": "call-center-availability"}]
}
},
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "mon" },
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "wed" },
{ "url": "availableStartTime", "valueTime": "09:00:00" }, // Interpreted in America/New_York
{ "url": "availableEndTime", "valueTime": "17:00:00" } // Interpreted in America/New_York
]
}
]
}
]
}
]
}

In this example:

  • Cardiac surgery availability is defined in America/Los_Angeles time zone (Mon-Wed 11am-3pm America/Los Angeles)
  • Call Center availability is defined in America/New_York time zone (Mon-Wed 9am-5pm Eastern)
  • Each service type's availability times are interpreted independently based on their respective timezones

Examples

Example 1: Simple Primary Care Office with Appointment Type Defaults

This example shows how to define availability for a simple primary care office where Practitioner's Schedules inherit default scheduling parameters from an HealthcareService.

HealthcareService: Office Visit Defaults

This HealthcareService defines default scheduling parameters for a 30-minute office visit with 5-minute buffers and 15-minute alignment intervals.

{
"resourceType": "HealthcareService",
"id": "office-visit",
"type": [{
"text": "Office Visit",
"coding": [{
"system": "http://example.org/appointment-types",
"code": "office-visit",
"display": "Office Visit"
}]
}],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "duration",
"valueDuration": {
"value": 30,
"unit": "min"
}
},
{
"url": "bufferBefore",
"valueDuration": {"value": 5, "unit": "min"}
},
{
"url": "bufferAfter",
"valueDuration": {"value": 5, "unit": "min"}
},
{
"url": "alignmentInterval",
"valueDuration": {"value": 15, "unit": "min"}
},
{
"url": "alignmentOffset",
"valueDuration": {"value": 0, "unit": "min"}
}
]
}]
}
Schedule: Practitioner's Schedule without Overrides

This Schedule shows Dr. Johnson's availability (Mon-Fri 9am-5pm) that inherits all default parameters from the HealthcareService without any service-specific overrides.

{
"resourceType": "Schedule",
"id": "dr-johnson-schedule",
"active": true,
"actor": [{
"reference": "Practitioner/dr-johnson",
"display": "Dr. Sarah Johnson"
}],
"planningHorizon": {
"start": "2025-01-01T00:00:00Z",
"end": "2025-12-31T23:59:59Z"
},
"serviceType": [
// This entry will allow using the office-visit shared HealthcareService definitions
{
"text": "Office Visit",
"coding": [
{ "code": "office-visit" }
]
}
],
"extension": [
// No values here, everything is inherited from shared definitions
]
}

Result: Dr. Johnson's schedule inherits all the default parameters from the HealthcareService for an office visit:

  • $find called with service-type=office-visit: For office visits, available to start every 15 minutes (:00, :15, :30, :45) with 5-minute buffers [from HealthcareService]

Example 2: Multi-Service Provider with Multiple Appointment Types and Overrides

A provider who offers different appointment types with varying availability and constraints. Overrides the default scheduling parameters for new patient visits.

HealthcareService: New Patient Visit

This HealthcareService defines a 60-minute new patient visit with 15-minute buffers, 30-minute alignment intervals, and a booking limit of 3 per day.

{
"resourceType": "HealthcareService",
"id": "new-patient-visit",
"type": [{
"text": "New Patient Visit",
"coding": [{
"system": "http://example.org/appointment-types",
"code": "new-patient-visit",
"display": "New Patient Visit"
}]
}],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{"url": "bufferBefore", "valueDuration": {"value": 15, "unit": "min"}},
{"url": "bufferAfter", "valueDuration": {"value": 15, "unit": "min"}},
{"url": "alignmentInterval", "valueDuration": {"value": 30, "unit": "min"}},
{
"url": "bookingLimit",
"valueTiming": {
"repeat": {"frequency": 3, "period": 1, "periodUnit": "d"}
}
}
]
}]
}
HealthcareService: Follow-up Visit

This HealthcareService defines a 20-minute follow-up visit with 5-minute buffers and 10-minute alignment intervals for more frequent scheduling.

It defines default availability of Monday-Friday, 9am-5pm.

{
"resourceType": "HealthcareService",
"id": "follow-up-visit",
"type": {
"text": "Follow-up Visit",
"coding": [{
"system": "http://example.org/appointment-types",
"code": "follow-up",
"display": "Follow-up Visit"
}]
},
"availableTime": [{
"daysOfWeek": ["mon","tue","wed","thu","fri"],
"availableStartTime": "09:00:00",
"availableEndTime": "17:00:00",
}],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "duration",
"valueDuration": {
"value": 20,
"unit": "min"
}
},
{"url": "bufferBefore", "valueDuration": {"value": 5, "unit": "min"}},
{"url": "bufferAfter", "valueDuration": {"value": 5, "unit": "min"}},
{"url": "alignmentInterval", "valueDuration": {"value": 10, "unit": "min"}}
]
}]
}
Schedule: Multi-Service with Overrides

This schedule declares in its serviceType array that it can be booked for New Patient visits and Follow-Up visits.

This Schedule will use the default availability for the "Follow-Up" service (Mon-Fri 9am-5pm), then override new patient visits to only be available on Tuesday and Thursday mornings (9am-1pm).

{
"resourceType": "Schedule",
"id": "dr-chen-schedule",
"active": true,
"actor": [{"reference": "PractitionerRole/dr-chen"}],
"planningHorizon": {
"start": "2025-01-01T00:00:00Z",
"end": "2025-12-31T23:59:59Z"
},
"serviceType": [
{
"text": "New Patient Visit",
"coding": [
{
"system": "http://example.org/appointment-types",
"code": "new-patient-visit"
}
]
},
{
"text": "Follow-up Visit",
"coding": [
{
"system": "http://example.org/appointment-types",
"code": "follow-up"
}
]
}
],
"extension": [
// New patient visits only on Tuesday and Thursday mornings
{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "serviceType",
"valueCodeableConcept": {
"coding": [{
"system": "http://example.org/appointment-types",
"code": "new-patient-visit"
}]
}
},
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "thu" },
{ "url": "availableStartTime", "valueTime": "09:00:00" },
{ "url": "availableEndTime", "valueTime": "13:00:00" }
]
}
]
}
]
}
]
}

Result:

  • New patient visits (ie. $find called with service-type=new-patient-visit): Tue/Thu 9am-1pm only, 60 minutes, can start every 30 minutes, max 3 per day, 15-min buffers
  • Follow-ups (ie. $find called with service-type=follow-up): Mon-Fri 9am-5pm, 20 minutes, can start every 10 minutes, 5-min buffers

Example 3: Location-Specific Complex Surgical Scheduling

A bariatric surgery requiring surgeon, OR room, and anesthesiologist coordination.

HealthcareService: Bariatric Surgery

This HealthcareService defines scheduling for a 120-minute surgical procedure with 45/30-minute buffers and multiple booking limits (2 per day, 8 per week).

{
"resourceType": "HealthcareService",
"id": "bariatric-surgery",
"type": [{
"coding": [{
"system": "http://snomed.info/sct",
"code": "287809009",
"display": "Bariatric Surgery"
}]
}],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "duration",
"valueDuration": {
"value": 120,
"unit": "min"
}
},
{"url": "bufferBefore", "valueDuration": {"value": 45, "unit": "min"}},
{"url": "bufferAfter", "valueDuration": {"value": 30, "unit": "min"}},
{"url": "alignmentInterval", "valueDuration": {"value": 30, "unit": "min"}},
{
"url": "bookingLimit",
"valueTiming": {
"repeat": {"frequency": 2, "period": 1, "periodUnit": "d"}
}
},
{
"url": "bookingLimit",
"valueTiming": {
"repeat": {"frequency": 8, "period": 1, "periodUnit": "wk"}
}
}
]
}]
}
Schedule: Surgeon Availability

This Schedule shows Dr. Martinez's availability for bariatric surgeries, limited to Tuesday and Thursday mornings (8am-4pm).

{
"resourceType": "Schedule",
"id": "surgeon-martinez-schedule",
"active": true,
"serviceType": [
{
"coding": [{"system": "http://snomed.info/sct", "code": "287809009"}]
"extension": [{
"url": "https://medplum.com/fhir/service-type-reference",
"valueReference": { "reference": "HealthcareService/bariatric-surgery" }
}]
}
],
"actor": [{
"reference": "PractitionerRole/surgeon-martinez",
"display": "Dr. Maria Martinez - Bariatric Surgeon"
}],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "service",
"valueReference": { "reference": "HealthcareService/bariatric-surgery" }
},
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "thu" },
{ "url": "availableStartTime", "valueTime": "08:00:00" },
{ "url": "availableEndTime", "valueTime": "16:00:00" }
]
}
]
}
]
}]
}
Schedule: Operating Room Availability

This Schedule shows Operating Room 3's availability for surgical procedures, available weekdays 7am-7pm with extended 12-hour blocks.

{
"resourceType": "Schedule",
"id": "or-3-schedule",
"active": true,
"actor": [{
"reference": "Location/or-3",
"display": "Operating Room 3"
}],
"serviceType": [
{
"coding": [{"system": "http://snomed.info/sct", "code": "287809009"}]
"extension": [{
"url": "https://medplum.com/fhir/service-type-reference",
"valueReference": { "reference": "HealthcareService/bariatric-surgery" }
}]
}
],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "service",
"valueReference": { "reference": "HealthcareService/bariatric-surgery" }
}
},
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "mon" },
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "wed" },
{ "url": "daysOfWeek", "valueCode": "thu" },
{ "url": "daysOfWeek", "valueCode": "fri" },
{ "url": "availableStartTime", "valueTime": "07:00:00" },
{ "url": "availableEndTime", "valueTime": "19:00:00" }
]
}
]
}
]
}]
}
Schedule: Anesthesiologist Availability

This Schedule shows Dr. Kim's availability for surgical procedures, covering weekdays 7am-5pm (10-hour blocks).

{
"resourceType": "Schedule",
"id": "anesthesiologist-kim-schedule",
"active": true,
"serviceType": [
{
"coding": [{"system": "http://snomed.info/sct", "code": "287809009"}]
"extension": [{
"url": "https://medplum.com/fhir/service-type-reference",
"valueReference": { "reference": "HealthcareService/bariatric-surgery" }
}]
}
],
"actor": [{
"reference": "PractitionerRole/anesthesiologist-kim",
"display": "Dr. James Kim - Anesthesiologist"
}],
"extension": [{
"url": "https://medplum.com/fhir/StructureDefinition/SchedulingParameters",
"extension": [
{
"url": "service",
"valueReference": { "reference": "HealthcareService/bariatric-surgery" }
},
{
"url": "duration",
"valueDuration": {
"value": 1,
"unit": "h"
}
},
{
"url": "availability",
"extension": [
{
"url": "availableTime",
"extension": [
{ "url": "daysOfWeek", "valueCode": "mon" },
{ "url": "daysOfWeek", "valueCode": "tue" },
{ "url": "daysOfWeek", "valueCode": "wed" },
{ "url": "daysOfWeek", "valueCode": "thu" },
{ "url": "daysOfWeek", "valueCode": "fri" },
{ "url": "availableStartTime", "valueTime": "07:00:00" },
{ "url": "availableEndTime", "valueTime": "17:00:00" }
]
}
]
}
]
}]
}

Result: When booking a bariatric surgery, the system queries all three schedules, calculates the intersection of availability, and creates atomic transaction bundles to book all required resources simultaneously.

Location Modeling

Location Hierarchy Pattern

Organization (Surgery Center)
└─ Location (Building) [mode=kind]
├─ Location (Operating Rooms) [mode=kind, type=OR]
│ ├─ Location (OR-1) [mode=instance]
│ ├─ Location (OR-2) [mode=instance]
│ └─ Location (OR-3) [mode=instance]
└─ Location (Recovery Rooms) [mode=kind, type=RR]
├─ Location (Recovery-A) [mode=instance]
└─ Location (Recovery-B) [mode=instance]

Specific vs. "Any Available" Room

Specific room required:

  • Query Schedule?actor=Location/or-3

Any OR room acceptable:

  • Query: Schedule?actor:Location.partof:Location.type=OR

Best Practices

1. Use All-or-Nothing Overrides

If adding a service configuration on Schedule, specify ALL attributes to avoid confusion about inheritance.

2. Minimize Pre-Generated Slots

Only create Slot resources for:

  • Booked appointments (status: busy)
  • Blocked time (status: busy-unavailable)

Let $find calculate available windows dynamically.

3. Transaction Bundles for Multi-Resource Booking

Always use FHIR transaction bundles when booking appointments that require multiple resources to ensure atomicity.