Skip to main content

A Strict JSON Framework for LLM Outputs, that fixes problems that json.loads() cannot solve

Project description

Strict JSON

A Strict JSON Framework for LLM Outputs, that fixes problems that json.loads() cannot solve

  • Works for JSON outputs with multiple ' or " or { or } or \ or unmatched braces/brackets that may break a json.loads()
  • Optional Integration with OpenAI JSON Mode
  • Updated: 4 Jan 2024
  • Created: 28 Oct 2023
  • Video tutorial: https://www.youtube.com/watch?v=IjTUKAciTCg
  • Collaborators welcome

How do I install this?

  1. Download entire directory and go to root folder
  2. Download python version 3.11 (https://www.python.org/downloads/) - should work for later versions as well, but this was tested on python 3.11
  3. pip install -r requirements.txt

How do I use this?

  1. Replace <YOUR API KEY HERE> in os.environ['OPENAI_API_KEY'] = '<YOUR API KEY HERE>' with your own OpenAI API key (https://platform.openai.com/account/api-keys)
  2. Copy and paste strict_json and strict_function from Strict_JSON_v2.ipynb
  3. Use the functions as needed (Note: In the future, it will be as a python package/library for easy usage)

How does it work?

  • Extract JSON values as a string using a special regex (add delimiters to key to make ###key###) to split keys and values
  • By default, uses ast.literal_eval to best match the string to a literal (e.g. int, string, dict). Set literal_eval = False when calling strict_json to preserve output fields as string
  • Ensures that all JSON fields are output by LLM, if not it will feed in error message to LLM to iteratively correct its generation (default: 3 tries)

Features:

Basic Generation

  • system_prompt: Write in whatever you want the LLM to become. "You are a <purpose in life>"
  • user_prompt: The user input. Later, when we use it as a function, this is the function input
  • output_format: JSON of output variables in a dictionary, with the key as the output key, and the value as the output description
    • The output keys will be preserved exactly, while GPT will generate content to match the description of the value as best as possible

Example Usage

res = strict_json(system_prompt = 'You are a classifier',
                    user_prompt = 'It is a beautiful and sunny day',
                    output_format = {'Sentiment': 'Type of Sentiment',
                                    'Adjectives': 'List of adjectives',
                                    'Words': 'Number of words'})
                                    
print(res)

Example output

{'Sentiment': 'positive', 'Adjectives': ['beautiful', 'sunny'], 'Words': 7}

Advanced Generation

  • More advanced demonstration involving code that would typically break json.loads()

Example Usage

res = strict_json(system_prompt = 'You are a code generator, generating code to fulfil a task',
                    user_prompt = 'Given array p, output a function named func_sum to return its sum',
                    output_format = {'Elaboration': 'How you would do it',
                                     'C': 'Code',
                                    'Python': 'Code'})
                                    
print(res)

Example output

{'Elaboration': 'To calculate the sum of an array, we can iterate through each element of the array and add it to a running total. Finally, we return the total as the result.',

'C': 'int func_sum(int p[], int size) {\\n int sum = 0;\\n for (int i = 0; i < size; i++) {\\n sum += p[i];\\n }\\n return sum;\\n}',

'Python': 'def func_sum(p):\\n sum = 0\\n for num in p:\\n sum += num\\n return sum'}

Strict JSON Functions

  • Enhances strict_json() with a function-like interface for repeated use of modular LLM-based functions

  • Inputs (compulsory):

    • fn_description - Function description to describe process of transforming input variables to output variables
    • output_format - Dictionary containing output variables names and description for each variable. There must be at least one output variable
  • Inputs (optional):

    • examples - Examples in Dictionary form with the input and output variables (list if more than one)
    • input_type - Dictionary containing input variable names as keys and mapping functions as values (need not contain all variables)
    • output_type - Dictionary containing output variable names as keys and mapping functions as values (need not contain all variables)
    • kwargs - Additional arguments you would like to pass on to the strict_json function
  • Outputs: JSON of output variables in a dictionary (similar to strict_json)

Example Usage 1 (Description only)

# Construct the function: var1 will be first input variable, var2 will be second input variable and so on
fn = strict_function(fn_description = 'Output a sentence with words var1 and var2 in the style of var3', 
                     output_format = {'output': 'sentence'})

# Use the function
fn('ball', 'dog', 'happy')

Example Output 1

{'output': 'The happy dog chased the ball.'}

Example Usage 2 (Examples only)

# Construct the function: infer pattern from just examples without description (here it is multiplication)
fn = strict_function(fn_description = 'Map input to output based on examples', 
                     output_format = {'output': 'final answer'}, 
                     examples = [{'var1': 3, 'var2': 2, 'output': 6}, 
                                 {'var1': 5, 'var2': 3, 'output': 15}, 
                                 {'var1': 7, 'var2': 4, 'output': 28}])

# Use the function
fn(2, 10)

Example Output 2

{'output': 20}

Example Usage 3 (Description, Examples and Type Forcing)

# Construct the function: var1 will be first input variable, var2 will be second input variable and so on
fn = strict_function(fn_description = 'Output the sum and difference of var1 and var2', 
                 output_format = {'sum': 'sum of two numbers', 'difference': 'absolute difference of two numbers'}, 
                 examples = {'var1': 2, 'var2': 4, 'sum': 6, 'difference': '2'}, 
                 input_type = {'var1': int, 'var2': int},           # optional
                 output_type = {'sum': int, 'difference': str})     # optional

# Use the function
fn(3, 4)

Example Output 3

{'sum': 7, 'difference': '1'}

Integrating with OpenAI JSON Mode

  • If you want to use the OpenAI JSON Mode (which is pretty good btw), you can simply add in openai_json_mode = True in strict_json or strict_function
  • Note that the model must be one of gpt-4-1106-preview or gpt-3.5-turbo-1106. We will set it to gpt-3.5-turbo-1106 by default if you provide an invalid model

Example Usage

res = strict_json(system_prompt = 'You are a classifier',
                    user_prompt = 'It is a beautiful and sunny day',
                    output_format = {'Sentiment': 'Type of Sentiment',
                                    'Adjectives': 'List of adjectives',
                                    'Words': 'Number of words'},
                    openai_json_mode = True) # Toggle this to True
                                    
print(res)

Example output

{'Sentiment': 'positive', 'Adjectives': ['beautiful', 'sunny'], 'Words': 6}

Future Features:

  • Agents with Tool Use
  • Conversational Agents

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

strictjson-0.0.1.tar.gz (9.5 kB view hashes)

Uploaded Source

Built Distribution

strictjson-0.0.1-py3-none-any.whl (8.3 kB view hashes)

Uploaded Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page