Language Models

Language models are used to generate agent responses to survey questions and can be specified using the Model and ModelList classes.

API keys are required in order to access available models, and should be stored in your private .env file. See the API Keys page for instructions on storing your API keys.

Output for examples shown below can also be viewed in this notebook at Coop.

Available services

The following code will return a table of currently available services (model providers):

from edsl import Model

Model.services()

Output:

Service Name

openai

anthropic

deep_infra

google

groq

bedrock

azure

ollama

test

together

perplexity

mistral

Available models

The following code will return a table of all the available models for all services:

from edsl import Model

Model.available()

This will return a list of the models we can choose from, for all service providers (omitted here for brevity). Run the code on yor own to see an up-to-date list.

To see a list of all models for a specific service, pass the service:

Model.available(service = "google")

Output:

Model Name

gemmini-1.0-pro

gemmini-1.0-flash

gemmini-1.5-pro

gemmini-pro

Service Name

google

google

google

google

Note: It is important to check that selected models are working as expected before running a survey. We recommend running test questions with any models, agents and scenarios that you plan to use in a survey to validate performance before moving onto larger jobs.

Adding a model

Newly available models for these services are added automatically. If you do not see a publicly available model that you want to work with, please send us a feature request to add it or add it yourself by calling the add_model() method:

from edsl import Model

Model.add_model(service_name = "anthropic", model_name = "new_model")

This will add the model new_model to the anthropic service. You can then see the model in the list of available models, and search by service name:

Model.available(service = "anthropic")

Output:

Check models

To check for models where API keys have been stored:

from edsl import Model

Model.check_models()

This will return a list of the available models and a confirmation message whether a valid key exists. The output will look like this (note that the keys are not shown):

Checking all available models...

Now checking: <model name>
OK!

Etc.

Specifying a model

We specify a model to use with a survey by creating a Model object and passing it the name of an available model. We can optionally set other model parameters as well (temperature, etc.). For example, the following code creates a Model object for Claude 3.5 Sonnet with default model parameters:

from edsl import Model

model = Model('gpt-4o')

We can see that the object consists of a model name and a dictionary of parameters:

model

This will show the default parameters of the model:

key

value

model

gpt-4o

parameters:temperature

0.5

parameters:max_tokens

1000

parameters:top_p

1

parameters:frequency_penalty

0

parameters:presence_penalty

0

parameters:logprobs

False

parameters:top_logprobs

3

Running a survey with models

Similar to how we specify Agents and Scenarios in running a survey, we specify the models to use by adding them to a survey with the by() method when the survey is run. We can pass either a single Model object or a list of models to the by() method. If multiple models are to be used they are passed as a list or as a ModelList object. For example, the following code specifies that a survey be run with each of GPT 4 and Gemini Pro:

from edsl import Model, Survey

models = [Model('gpt-4o'), Model('gemini-pro')]

survey = Survey.example()

results = survey.by(models).run()

This code uses ModelList instead of a list of Model objects:

from edsl import Model, ModelList, Survey

models = ModelList(Model(m) for m in ['gpt-4o', 'gemini-pro'])

survey = Survey.example()

results = survey.by(models).run()

This will generate a result for each question in the survey with each model. If agents and/or scenarios are also specified, the responses will be generated for each combination of agents, scenarios and models. Each component is added with its own by() method, the order of which does not matter. The following commands are equivalent:

results = survey.by(scenarios).by(agents).by(models).run()

results = survey.by(models).by(agents).by(scenarios).run()

Default model

If no model is specified, a survey is automatically run with the default model. Run Model() to check the current default model. For example, the following code runs the example survey with the default model (and no agents or scenarios) without needing to import the Model class:

from edsl import Survey

results = Survey.example().run()

Inspecting model details in results

If a survey has been run, we can inspect the models that were used by calling the models method on the Results object. For example, we can verify the default model when running a survey without specifying a model:

from edsl import Survey

survey = Survey.example()

results = survey.run()

results.models

This will return the following information about the default model that was used (note the default model may have changed since this page was last updated):

model

temperature

max_tokens

top_p

frequency_penalty

presence_penalty

logprobs

top_logprobs

gpt-4o

0.5

1000

1

0

0

False

3

To learn more about all the components of a Results object, please see the Results section.

Printing model attributes

If multiple models were used to generate results, we can print the attributes in a table. For example, the following code prints a table of the model names and temperatures for some results:

from edsl import Survey, ModelList, Model

models = ModelList(
   Model(m) for m in ['gpt-4o', 'gemini-1.5-pro']
)

survey = Survey.example()

results = survey.by(models).run()

results.select("model", "temperature") # This is equivalent to: results.select("model.model", "model.temperature")

Output:

model.model

model.temperature

gpt-4o

0.5

gemini-1.5-pro

0.5

We can also print model attributes together with other components of results. We can see a list of all components by calling the columns method on the results:

results.columns

Output:

0

agent.agent_instruction

agent.agent_name

answer.q0

answer.q1

answer.q2

comment.q0_comment

comment.q1_comment

comment.q2_comment

generated_tokens.q0_generated_tokens

generated_tokens.q1_generated_tokens

generated_tokens.q2_generated_tokens

iteration.iteration

model.frequency_penalty

model.logprobs

model.maxOutputTokens

model.max_tokens

model.model

model.presence_penalty

model.stopSequences

model.temperature

model.topK

model.topP

model.top_logprobs

model.top_p

prompt.q0_system_prompt

prompt.q0_user_prompt

prompt.q1_system_prompt

prompt.q1_user_prompt

prompt.q2_system_prompt

prompt.q2_user_prompt

question_options.q0_question_options

question_options.q1_question_options

question_options.q2_question_options

question_text.q0_question_text

question_text.q1_question_text

question_text.q2_question_text

question_type.q0_question_type

question_type.q1_question_type

question_type.q2_question_type

raw_model_response.q0_cost

raw_model_response.q0_one_usd_buys

raw_model_response.q0_raw_model_response

raw_model_response.q1_cost

raw_model_response.q1_one_usd_buys

raw_model_response.q1_raw_model_response

raw_model_response.q2_cost

raw_model_response.q2_one_usd_buys

raw_model_response.q2_raw_model_response

The following code will display a table of the model names together with the simulated answers:

results.select("model", "answer.*")

Output:

model.model

answer.q0

answer.q1

answer.q2

gpt-4o

no

killer bees in cafeteria

gemini-1.5-pro

yes

other

To learn more about methods of inspecting and printing results, please see the Results section.

ModelList class

class edsl.language_models.ModelList.ModelList(data: LanguageModel | None = None)[source]

Bases: Base, UserList

__init__(data: LanguageModel | None = None)[source]

Initialize the ScenarioList class.

>>> from edsl import Model
>>> m = ModelList(Model.available())
code()[source]

This method should be implemented by subclasses.

classmethod example(randomize: bool = False) ModelList[source]

Returns an example ModelList instance.

Parameters:

randomize – If True, uses Model’s randomize method.

classmethod from_available_models(available_models_list: AvailableModels)[source]

Create a ModelList from an AvailableModels object

classmethod from_dict(data)[source]

Create a ModelList from a dictionary.

>>> newm = ModelList.from_dict(ModelList.example().to_dict())
>>> assert ModelList.example() == newm
classmethod from_names(*args, **kwargs)[source]

A a model list from a list of names

property names[source]
>>> ModelList.example().names
{'...'}
table(*fields, tablefmt: str | None = None, pretty_labels: dict | None = None)[source]
>>> ModelList.example().table('model')
model
-------
gpt-4o
gpt-4o
gpt-4o
to_dict(sort=False, add_edsl_version=True)[source]

This method should be implemented by subclasses.

to_list() list[source]
to_scenario_list()[source]
tree(node_list: List[str] | None = None)[source]

LanguageModel class

This module contains the LanguageModel class, which is an abstract base class for all language models.

Terminology:

raw_response: The JSON response from the model. This has all the model meta-data about the call.

edsl_augmented_response: The JSON response from model, but augmented with EDSL-specific information, such as the cache key, token usage, etc.

generated_tokens: The actual tokens generated by the model. This is the output that is used by the user. edsl_answer_dict: The parsed JSON response from the model either {‘answer’: …} or {‘answer’: …, ‘comment’: …}

class edsl.language_models.LanguageModel.LanguageModel(tpm: float | None = None, rpm: float | None = None, omit_system_prompt_if_empty_string: bool = True, key_lookup: 'KeyLookup' | None = None, **kwargs)[source]

Bases: PersistenceMixin, RepresentationMixin, HashingMixin, ABC

ABC for Language Models.

DEFAULT_RPM = 100[source]
DEFAULT_TPM = 1000[source]
__init__(tpm: float | None = None, rpm: float | None = None, omit_system_prompt_if_empty_string: bool = True, key_lookup: 'KeyLookup' | None = None, **kwargs)[source]

Initialize the LanguageModel.

property api_token: str[source]
ask_question(question: QuestionBase) str[source]

Ask a question and return the response.

Parameters:

question – The question to ask.

abstract async async_execute_model_call(system_prompt: str)[source]

Execute the model call and returns a coroutine.

async async_get_response(user_prompt: str, system_prompt: str, cache: Cache, iteration: int = 1, files_list: List[FileStore] | None = None, **kwargs) dict[source]

Get response, parse, and return as string.

Parameters:
  • user_prompt – The user’s prompt.

  • system_prompt – The system’s prompt.

  • cache – The cache to use.

  • iteration – The iteration number.

  • files_list – The list of files to use.

cost(raw_response: dict[str, Any]) float | str[source]

Return the dollar cost of a raw response.

Parameters:

raw_response – The raw response from the model.

classmethod example(test_model: bool = False, canned_response: str = 'Hello world', throw_exception: bool = False) LanguageModel[source]

Return a default instance of the class.

>>> from edsl.language_models import LanguageModel
>>> m = LanguageModel.example(test_model = True, canned_response = "WOWZA!")
>>> isinstance(m, LanguageModel)
True
>>> from edsl import QuestionFreeText
>>> q = QuestionFreeText(question_text = "What is your name?", question_name = 'example')
>>> q.by(m).run(cache = False, disable_remote_cache = True, disable_remote_inference = True).select('example').first()
'WOWZA!'
>>> m = LanguageModel.example(test_model = True, canned_response = "WOWZA!", throw_exception = True)
>>> r = q.by(m).run(cache = False, disable_remote_cache = True, disable_remote_inference = True, print_exceptions = True)
Exception report saved to ...
Also see: ...
execute_model_call(**kwargs)[source]
from_cache(cache: Cache) LanguageModel[source]
classmethod from_dict(data: dict) Type[LanguageModel][source]

Convert dictionary to a LanguageModel child instance.

NB: This method does not use the stores inference_service but rather just fetches a model class based on the name.

classmethod get_generated_token_string(raw_response: dict[str, Any]) str[source]

Return the generated token string from the raw response.

>>> m = LanguageModel.example(test_model = True)
>>> raw_response = m.execute_model_call("Hello, model!", "You are a helpful agent.")
>>> m.get_generated_token_string(raw_response)
'Hello world'
get_response(user_prompt: str, system_prompt: str, cache: Cache, iteration: int = 1, files_list: List[FileStore] | None = None, **kwargs) dict[source]

Get response, parse, and return as string.

Parameters:
  • user_prompt – The user’s prompt.

  • system_prompt – The system’s prompt.

  • cache – The cache to use.

  • iteration – The iteration number.

  • files_list – The list of files to use.

classmethod get_usage_dict(raw_response: dict[str, Any]) dict[str, Any][source]

Return the usage dictionary from the raw response.

has_valid_api_key() bool[source]

Check if the model has a valid API key.

>>> LanguageModel.example().has_valid_api_key() : 
True

This method is used to check if the model has a valid API key.

hello(verbose=False)[source]

Runs a simple test to check if the model is working.

key_sequence = None[source]
classmethod parse_response(raw_response: dict[str, Any]) EDSLOutput[source]

Parses the API response and returns the response text.

async remote_async_execute_model_call(user_prompt: str, system_prompt: str)[source]

Execute the model call and returns the result as a coroutine, using Coop.

response_handler = <edsl.language_models.RawResponseHandler.RawResponseHandler object>[source]
property rpm[source]
set_key_lookup(key_lookup: KeyLookup) None[source]

Set the key lookup, later

simple_ask(question: QuestionBase, system_prompt='You are a helpful agent pretending to be a human.', top_logprobs=2)[source]

Ask a question and return the response.

to_dict(add_edsl_version: bool = True) dict[str, Any][source]

Convert instance to a dictionary

Parameters:

add_edsl_version – Whether to add the EDSL version to the dictionary.

>>> m = LanguageModel.example()
>>> m.to_dict()
{'model': '...', 'parameters': {'temperature': ..., 'max_tokens': ..., 'top_p': ..., 'frequency_penalty': ..., 'presence_penalty': ..., 'logprobs': False, 'top_logprobs': ...}, 'inference_service': 'openai', 'edsl_version': '...', 'edsl_class_name': 'LanguageModel'}
property tpm[source]
class edsl.language_models.LanguageModel.classproperty(method)[source]

Bases: object

__init__(method)[source]
edsl.language_models.LanguageModel.handle_key_error(func)[source]

Handle KeyError exceptions.

Other methods