Skip to content

Schemas

Schemas are the blueprints that help you define the shape of your data in Prism. Whether you're building tool parameters or crafting structured outputs, schemas help you clearly communicate what your data should look like.

Quick Start

Let's dive right in with a practical example:

php
use EchoLabs\Prism\Schema\ArraySchema;
use EchoLabs\Prism\Schema\ObjectSchema;
use EchoLabs\Prism\Schema\StringSchema;

$userSchema = new ObjectSchema(
    name: 'user',
    description: 'A user profile with their hobbies',
    properties: [
        new StringSchema('name', 'The user\'s full name'),
        new ArraySchema(
            name: 'hobbies',
            description: 'The user\'s list of hobbies',
            items: new ObjectSchema(
                name: 'hobby',
                description: 'A detailed hobby entry',
                properties: [
                    new StringSchema('name', 'The name of the hobby'),
                    new StringSchema('description', 'A brief description of the hobby'),
                ],
                requiredFields: ['name', 'description']
            )
        ),
    ],
    requiredFields: ['name', 'hobbies']
);

Available Schema Types

StringSchema

For text values of any length. Perfect for names, descriptions, or any textual data.

php
use EchoLabs\Prism\Schema\StringSchema;

$nameSchema = new StringSchema(
    name: 'full_name',
    description: 'The user\'s full name including first and last name'
);

NumberSchema

Handles both integers and floating-point numbers. Great for ages, quantities, or measurements.

php
use EchoLabs\Prism\Schema\NumberSchema;

$ageSchema = new NumberSchema(
    name: 'age',
    description: 'The user\'s age in years'
);

BooleanSchema

For simple true/false values. Perfect for flags and toggles.

php
use EchoLabs\Prism\Schema\BooleanSchema;

$activeSchema = new BooleanSchema(
    name: 'is_active',
    description: 'Whether the user account is active'
);

ArraySchema

For lists of items, where each item follows a specific schema.

php
use EchoLabs\Prism\Schema\ArraySchema;
use EchoLabs\Prism\Schema\StringSchema;

$tagsSchema = new ArraySchema(
    name: 'tags',
    description: 'List of tags associated with the post',
    items: new StringSchema('tag', 'A single tag')
);

EnumSchema

When you need to restrict values to a specific set of options.

php
use EchoLabs\Prism\Schema\EnumSchema;

$statusSchema = new EnumSchema(
    name: 'status',
    description: 'The current status of the post',
    options: ['draft', 'published', 'archived']
);

ObjectSchema

For complex, nested data structures. The Swiss Army knife of schemas!

php
use EchoLabs\Prism\Schema\ObjectSchema;
use EchoLabs\Prism\Schema\StringSchema;
use EchoLabs\Prism\Schema\NumberSchema;

$profileSchema = new ObjectSchema(
    name: 'profile',
    description: 'A user\'s public profile information',
    properties: [
        new StringSchema('username', 'The unique username'),
        new StringSchema('bio', 'A short biography'),
        new NumberSchema('joined_year', 'Year the user joined'),
    ],
    requiredFields: ['username']
);

Nullable Fields

Sometimes, not every field is required. You can make any schema nullable by setting the nullable parameter to true:

php
use EchoLabs\Prism\Schema\StringSchema;

$bioSchema = new StringSchema(
    name: 'bio',
    description: 'Optional user biography',
    nullable: true
);

NOTE

When using OpenAI in strict mode, all fields must be marked as required, so optional fields must be marked as nullable.

Required vs Nullable Fields

Understanding the difference between required fields and nullable fields is crucial when working with schemas in Prism:

Required Fields

Required fields are specified at the object level using the requiredFields parameter. They indicate which properties must be present in the data structure:

php
$userSchema = new ObjectSchema(
    name: 'user',
    description: 'User profile',
    properties: [
        new StringSchema('email', 'Primary email address'),
        new StringSchema('name', 'User\'s full name'),
        new StringSchema('bio', 'User biography'),
    ],
    requiredFields: ['email', 'name'] // email and name must be present
);

Nullable Fields

Nullable fields, on the other hand, are specified at the individual field level using the nullable parameter. They indicate that a field can contain a null value:

php
$userSchema = new ObjectSchema(
    name: 'user',
    description: 'User profile',
    properties: [
        new StringSchema('email', 'Primary email address'),
        new StringSchema('name', 'User\'s full name'),
        new StringSchema('bio', 'User biography', nullable: true), // bio can be null
    ],
    requiredFields: ['email', 'name', 'bio'] // bio must be present, but can be null
);

Key Differences

  1. Required vs Present:

    • A required field must be present in the data structure
    • A non-nullable field must contain a non-null value when present
    • A field can be required but nullable (must be present, can be null)
    • A field can be non-required and non-nullable (when present, cannot be null)
  2. Common Patterns:

php
// Required and Non-nullable (most strict)
new StringSchema('email', 'Primary email', nullable: false);
// requireFields: ['email']

// Required but Nullable (must be present, can be null)
new StringSchema('bio', 'User bio', nullable: true);
// requireFields: ['bio']

// Optional and Non-nullable (can be omitted, but if present cannot be null)
new StringSchema('phone', 'Phone number', nullable: false);
// requireFields: []

// Optional and Nullable (most permissive)
new StringSchema('website', 'Personal website', nullable: true);
// requireFields: []

Provider Considerations

When working with providers that support strict mode (like OpenAI), you'll want to be especially careful with these settings:

php
// For OpenAI strict mode: 
// - All fields should be required
// - Use nullable: true for optional fields
$userSchema = new ObjectSchema(
    name: 'user',
    description: 'User profile',
    properties: [
        new StringSchema('email', 'Required email address'),
        new StringSchema('bio', 'Optional biography', nullable: true),
    ],
    requiredFields: ['email', 'bio'] // Note: bio is required but nullable
);

TIP

When in doubt, be explicit about both requirements. Specify both the nullable status of each field AND which fields are required in your object schemas. This makes your intentions clear to both other developers and AI providers.

Best Practices

  1. Clear Descriptions: Write clear, concise descriptions for each field. Future you (and other developers) will thank you!

    php
    // ❌ Not helpful
    new StringSchema('name', 'the name');
    
    // ✅ Much better
    new StringSchema('name', 'The user\'s display name (2-50 characters)');
  2. Thoughtful Required Fields: Only mark fields as required if they're truly necessary:

    php
    new ObjectSchema(
        name: 'user',
        description: 'User profile',
        properties: [
            new StringSchema('email', 'Primary email address'),
            new StringSchema('phone', 'Optional phone number', nullable: true),
        ],
        requiredFields: ['email']
    );
  3. Nested Organization: Keep your schemas organized when dealing with complex structures:

    php
    // Define child schemas first
    $addressSchema = new ObjectSchema(/*...*/);
    $contactSchema = new ObjectSchema(/*...*/);
    
    // Then use them in your parent schema
    $userSchema = new ObjectSchema(
        name: 'user',
        description: 'Complete user profile',
        properties: [$addressSchema, $contactSchema]
    );

NOTE

Remember that while schemas help define the structure of your data, Prism doesn't currently validate the data against these schemas. Schema validation is planned for a future release!

Released under the MIT License.