This vignette provides an overview of the tcpl package, including set up, Database Structure, Pre-processing Requirements, Assay and Chemical Registration, Data Processing, Data Interpretation, and Data Retrieval with invitrodb and via API.
The ToxCast Data Analysis Pipeline (tcpl) is an R package that manages, curve-fits, plots, and stores ToxCast data to populate its linked MySQL database, invitrodb. The U.S. Environmental Protection Agency (EPA)’s Toxicity Forecaster (ToxCastTM) program includes in vitro medium- and high-throughput screening (HTS) assays for the prioritization and hazard characterization of thousands of chemicals of interest. Targeted and confirmatory assays (like ToxCast assays) comprise Tiers 2-3 of the Computational Toxicology Blueprint (Thomas et al., 2019), and employ automated chemical screening technologies to evaluate the effects of chemical exposure on living cells and biological macromolecules, such as proteins.
The tcpl package is a flexible analysis pipeline is capable of efficiently processing and storing large volumes of data. The diverse data, received in heterogeneous formats from numerous vendors, are transformed to a standard computable format via Level 0 Preprocessing then loaded into the database by vendor-specific R scripts. Describing the specific transformations may be outside the scope of this package, but can be done for virtually any chemical screening effort, provided the data includes the minimum required information. Once data is loaded into the database, generalized processing functions provided in this package process, normalize, model, qualify, and visualize the data.
The original tcplFit() functions performed basic concentration response curve fitting. Processing with tcpl v3 and beyond depends on the stand-alone tcplFit2 package to allow a wider variety of concentration-response models when using invitrodb in the 4.0 schema and beyond. Using tcpl_v3 with the schema from invitrodb versions 2.0-3.5 will still default to tcplFit() modeling with constant, Hill, and gain-loss. The main improvement provided by updating to using tcplFit2 is inclusion of concentration-response models like those contained in the program BMDExpress2. These models include polynomial, exponential, and power functions in addition to the original Hill, gain-loss, and constant models. Similar to the program BMDExpress, tcplFit2 curve-fitting uses a defined Benchmark Response (BMR) level to estimate a benchmark dose (BMD), which is the concentration where the curve-fit intersects with this BMR threshold. One final addition was to let the hit call value be a continuous number ranging from 0 to 1 (in contrast to binary hit call values from tcplFit() ). While developed primarily for ToxCast, the tcpl package is written to be generally applicable to the chemical-screening community.
The tcpl package includes processing functionality for two screening paradigms: (1) single-concentration (SC) and (2) multiple-concentration (MC) screening. SC screening consists of testing chemicals at one to three concentrations, often for the purpose of identifying potentially active chemicals to test in the multiple-concentration format. MC screening consists of testing chemicals across a concentration range, such that the modeled activity can give an estimate of potency, efficacy, etc.
In addition to storing the data, the tcpl database stores every processing and analysis decision at the assay component or assay endpoint level to facilitate transparency and reproducibility. For the illustrative purposes of this vignette, we have included a CSV version of the tcpl database containing a small subset of data from the ToxCast program. tcplLite is no longer supported by tcpl because tcplfit2 can be used to curve-fit data and make hit calls independent of invitrodb, available at https://cran.r-project.org/package=tcplfit2. tcplLite relied on flat files structured like invitrodb to produce curve-fitting and summary information like hit calls and AC50 values. Functionally tcplfit2 replaces tcplLite because interested stakeholders can now curve-fit data and reproduce curve-fitting results independent of the invitrodb schema. For the ToxCast program, it is still important to use invitrodb when curve-fitting as invitrodb serves as a data resource for tracking pipelining decisions and providing a dataset for many interested stakeholders. Using tcpl, the user can upload, process, and retrieve data by connecting to a MySQL database. Additionally, past versions of the ToxCast database, containing all the publicly available ToxCast data, are available for download at: https://www.epa.gov/comptox-tools/exploring-toxcast-data.
Check out the following publications for additional information on the overall US EPA’s Toxicity Forecaster (ToxCast) Program. Assay-specific publications describing assay design or results are available in the assay_references and citations tables.
First, it is highly recommended for users to utilize the data.table package. The tcpl package utilizes the data.table package for all data frame-like objects.
After loading tcpl, the function
tcplConf is used to establish connection to a
database server or the API. While a typical database connection requires
5 parameters to be provided, using an API connection requires the user
to only specify password (pass
) and driver
(drvr
):
tcplConf(pass = "API key provided by emailing CTX API support at [email protected]",
drvr = "API")
NOTE: When tcpl is loaded, the default configuration sets the options to tcpl’s application API key to support new users testing out the package. This default API key is not intended for regular users; instead, it is highly recommended to obtain a personal API key to also access other CTX APIs. For this, send an email request to CTX API support at [email protected].
Every time the package is loaded in a new R session, a message similar to the following will print showing the default package settings:
tcpl (v3.1.0) loaded with the following settings:
TCPL_DB: NA
TCPL_USER: NA
TCPL_HOST: https://api-ccte.epa.gov/bioactivity
TCPL_DRVR: API
Default settings stored in tcpl config file. See ?tcplConf for more information.
Establishing a database connection utilizes the following settings:
Refer to ?tcplConf for more information. At any time, users can check the settings using tcplConfList(). An example of database settings using tcpl would be as follows:
tcplConf(db = "invitrodb",
user = "username",
pass = "password",
host = "localhost",
drvr = "MySQL")
tcplConfList will list connection information. Note, tcplSetOpts will only make changes to the parameters given. The package is always loaded with the settings stored in the TCPL.config file located within the package directory. The user can edit the file, such that the package loads with the desired settings, rather than having to call the tcplSetOpts function every time. The TCPL.config file has to be edited whenever the package is updated or re-installed. With no TCPL.config file available, the package defaults to an “API” connection configuration with the CTX APIs.
The following contains reference tables that describe the structure and fields found in the tcpl populated database. The first sections describe the data-containing tables, followed by sections describing the additional annotation tables.
In general, the single-concentration data and accompanying methods are found in the “sc#” tables, where the number indicates the processing level. Likewise, the multiple-concentration data and accompanying methods are found in the “mc#” tables. Each processing level that has accompanying methods will also have tables with the “_methods” and “_id” naming scheme. For example, the database contains the following tables: “mc5” storing the data from multiple-concentration level 5 processing, “mc5_methods” storing the available level 5 methods, and “mc5_aeid” storing the method assignments for level 5. Note, the table storing the method assignments for level 2 multiple-concentration processing is called “mc2_acid”, because MC2 methods are assigned by assay component ID.
There are two additional tables, “sc2_agg” and “mc4_agg,” that link the data in tables “sc2” and “mc4” to the data in tables “sc1” and “mc3,” respectively. This is necessary because each entry in the database before SC2 and MC4 processing represents a single value; subsequent entries represent summary/modeled values that encompass many values. To know what values were used in calculating the summary/modeled values, the user must use the “_agg” look-up tables.
When using tcpl_v3 with invitrodb schemas v2.0-v3.5, tcplFit model data are structured in mc4 and mc5 tables that are in wide format with a fixed number of columns based on 3 curve-fitting models (see documentation associated with tcpl_v2.1 ). When using tcpl v3+ with invitrodb schemas v4+, mc4 and mc5 tables have been updated to reflect having mc4_param and mc5_param tables. Tables should be reviewed together: mc4 captures summary values calculated for each concentration series, whereas mc4_param includes parameters for all models in long format. mc5 selects the winning model and activity hit call, whereas mc5_param includes model parameters from selected winning (hit) model in long format. These schema changes provide a way to continually expand modeling capabilities in tcpl .
Each of the methods tables have fields analogous to mc5_mthd_id, mc5_mthd, and desc. These fields represent the unique key for the method, the abbreviated method name (used to call the method from the corresponding mc5_mthds function), and a brief description of the method, respectively. The method assignment tables will have fields analogous to mc5_mthd_id matching the method ID from the methods tables, an assay component or assay endpoint ID, and possibly an exec_ordr field indicating the order in which to execute the methods. The method and method assignment tables will not be listed in the tables below to reduce redundancy.
Many of the tables also include the created_date, modified_date, and modified_by fields that store helpful information for tracking changes to the data. These fields will not be discussed further or included in the tables below.
Many of the tables specific to the assay annotation are populated semi-manually based on expert curation of information on assay design; these tables of assay annotation are not currently utilized by the tcpl package, but instead act as meta-data for users. The full complexity of the assay annotation used by the ToxCast program is somewhat beyond the scope of this vignette and the tcpl package. Additionally, a compiled report of assay description documents are available on the ToxCast Downloadable Data page.
The “mc0” and “sc0” tables are identical, other than containing m0id rather than s0id, respectively. See the Level 0 Pre-processing section for more information.
Field | Description |
---|---|
s0id | Level 0 ID |
acid | Assay component ID |
spid | Sample ID |
apid | Assay plate ID |
rowi | Assay plate row index |
coli | Assay plate column index |
wllt | Well type |
wllq | Well quality: 1 was good, else 0 |
conc | Concentration is micromolar |
rval | Raw assay component value or readout |
srcf | Filename of the source file containing the data |
Information about the different well types is available in Level 0 Pre-processing section.
Field | Description |
---|---|
s1id | Level 1 ID |
s0id | Level 0 ID |
acid | Assay component ID |
aeid | Assay component endpoint ID |
conc | Concentration is micromolar |
bval | Baseline value |
pval | Positive control value |
resp | Normalized response value |
Field | Description |
---|---|
s2id | Level 2 ID |
aeid | Assay component endpoint ID |
spid | Sample ID |
bmad | Baseline median absolute deviation |
max_med | Maximum median response value |
coff | Efficacy cutoff value |
hitc | Binary hit call value: 1 if active, 0 if inactive |
tmpi | Ignore, temporary index used for uploading purposes |
Field | Description |
---|---|
aeid | Assay component endpoint ID |
s0id | Level 0 ID |
s1id | Level 1 ID |
s2id | Level 2 ID |
Field | Description |
---|---|
s2id | Level 2 ID |
chid_rep | Representative sample designation for a tested chemical: 1 if representative sample, else 0 |
See the Data Interpretation>Representative Samples section for more details.
Field | Description |
---|---|
m1id | Level 1 ID |
m0id | Level 0 ID |
acid | Assay component ID |
cndx | Concentration index |
repi | Replicate index |
Field | Description |
---|---|
m2id | Level 2 ID |
m0id | Level 0 ID |
acid | Assay component ID |
m1id | Level 1 ID |
cval | Corrected value |
Field | Description |
---|---|
m3id | Level 3 ID |
aeid | Assay endpoint ID |
m0id | Level 0 ID |
acid | Assay component ID |
m1id | Level 1 ID |
m2id | Level 2 ID |
bval | Baseline value |
pval | Positive control value |
conc | Concentration is micromolar |
resp | Normalized response value |
Field | Description |
---|---|
aeid | Assay endpoint ID |
m0id | Level 0 ID |
m1id | Level 1 ID |
m2id | Level 2 ID |
m3id | Level 3 ID |
m4id | Level 4 ID |
Field | Description |
---|---|
m4id | Level 4 ID |
aeid | Assay endpoint ID |
spid | Sample ID |
bmad | Baseline median absolute deviation |
resp_max | Maximum response value |
resp_min | Minimum response value |
max_mean | Maximal mean response at a given concentration |
max_mean_conc | Corresponding concentration of max_mean |
min_mean | Minimum mean response value at a given concentration |
min_mean_conc | Corresponding concentration of min_mean |
max_med | Maximum median response value at a given concentration |
max_med_conc | Corresponding concentration of max_med |
min_med | Minimum median response value at a given concentration |
min_med_conc | Corresponding concentration of min_med |
max_med_diff | Absolute difference between maximal and minimal median response at a given concentration |
max_med_diff_conc | Absolute difference between corresponding concentration of max_med and min_med |
conc_max | Maximum concentration tested |
conc_min | Minimum concentration tested |
nconc | Number of concentrations tested |
npts | Number of points in the concentration series |
nrep | Number of replicates in the concentration series |
nmed_gtbl_pos | Number of median response values greater than baseline of 3 * bmad |
nmed_gtbl_neg | Number of median response values less than baseline of -3 * bmad |
tmpi | Ignore, temporary index used for uploading purposes |
Field | Description |
---|---|
m4id | Level 4 ID |
aeid | Assay endpoint ID |
model | Model that was fit |
model_param | Key for the parameter that was fit with the corresponding model |
model_val | Value for the associated key in the corresponding model |
Field | Description |
---|---|
m5id | Level 5 ID |
m4id | Level 4 ID |
aeid | Assay endpoint ID |
modl | Winning model |
hitc | Activity hitcall |
fitc | Fit category |
coff | Efficacy cutoff value |
actp | Activity probability (1 - const_prob not used with tcplFit2) |
model_type | Model type. Options include: 2: Bidirectional: Data is fit bidirectionally. 3: Gain: Data is fit bidirectionally, but gain is the intended direction of response. Hitcalls (hitc) for winnings models is multiplied by -1 for models fit in the negative analysis direction. 4: Loss: Data is fit bidirectionally, but loss is the intended direction of response. Hitcalls (hitc) for winnings models is multiplied by -1 for models fit in the positive analysis direction. |
See the Data Interpretation>Hit Calls section for more details.
Field | Description |
---|---|
m5id | Level 5 ID |
aeid | Assay endpoint ID |
hit_param | Key for the parameter that was fit with winning model |
hit_val | Value for the associated key in the winning model |
Field | Description |
---|---|
m5id | Level 5 ID |
chid_rep | Representative sample designation for a tested chemical: 1 if representative sample, else 0 |
See the Data Interpretation>Representative Samples section for more details.
Field | Description |
---|---|
m6id | Level 6 ID |
m5id | Level 5 ID |
m4id | Level 4 ID |
aeid | Assay endpoint ID |
mc6_mthd_id | Level 6 method ID |
flag | Short flag description to be displayed in data retrieval and plotting. Extended description available in MC6_Methods table. |
See the Data Interpretation>Flags section for more details.
Field | Description |
---|---|
m7id | Level 7 ID |
m4id | Level 4 ID |
aeid | Assay endpoint ID |
potency_val_type | Potency value type used in the calc_mc_oral_equiv() calculation |
aed_type | Descriptive vector that begins with “aed,” followed by potency metric used, followed by a short name of the httk model used, ending with the percentile from the modeled population with respect to interindividual variability |
aed_val | Numeric value of the AED |
aed_val_unit | Unit associated with AED, mg/kg/day |
interindividual_var_perc | Interindividual variability percentile, either 50th or 95th |
httk_model | The httk model used; 3-compartment steady state (3compartmentss) or pbtk; note that all models used here were for adult humans. |
invitrodb_version | invitrodb version of data |
httk_version | Version of httk R package used |
See the Data Interpretation>Adminstered Equivalent Doses section for more details.
The fields pertinent to the tcpl package are listed in the tables below. More specifics on assay and auxiliary annotations will be provided in later sections.
Table | Description |
---|---|
assay_source | Assay source-level annotation |
assay | Assay-level annotation |
assay_component | Assay component-level annotation |
assay_component_endpoint | Assay endpoint-level annotation |
assay_component_map | Assay component source names and their corresponding assay component ids |
assay_descriptions** | Additional assay descriptions curated per OECD Guidance Document 211 (GD211) |
assay_reagent** | Assay reagent information |
assay_reference** | Map of citations to assay |
chemical | List of chemicals and associated identifiers |
chemical_analytical_qc** | Analytical QC information curated at the chemical substance or sample level to inform applicability domain |
chemical_lists | Lists of chemicals and their presence in curated chemical lists |
citations** | List of citations |
gene** | Gene identifiers and descriptions |
intended_target** | Intended assay target at the assay endpoint level |
organism** | Organism identifiers and descriptions |
sample | Sample identifiers and chemical provenance information |
** indicates tables may have limited tcpl functionality, but data is still retrievable via tcplQuery.
Field | Description |
---|---|
asid | Assay source ID. Required for registration. |
assay_source_name | Assay source name, typically an abbreviation of the assay_source_long_name and abbreviated “asnm” within the package. Required for registration |
assay_source_long_name | Full assay source name |
assay_source_desc | Assay source description |
Field | Description |
---|---|
aid | Assay ID |
asid; | Assay source ID. Required for registration. |
assay_name | Assay name, abbreviated “anm” within the package. Required for registration. |
assay_desc | Assay description |
timepoint_hr | Treatment duration in hours |
organism_id | NCBI taxonomic identifier, available at https://www.ncbi.nlm.nih.gov/taxonomy |
organism | Organism of origin |
tissue | Tissue of origin |
cell_format | Description of cell format |
cell_free_component_source | Description of source for targeted cell-free components |
cell_short_name | Abbreviation of cell line |
cell_growth_mode | Cell growth modality |
assay_footprint | Microtiter plate size. Required for registration. |
assay_format_type | General description of assay format |
assay_format_type_sub | Specific description of assay format |
content_readout_type | Description of well characteristics being measured |
dilution_solvent | Solvent used in sample dilution |
dilution_solvent_percent_max | Maximum percent of dilution solvent used, from 0 to 1 |
Field | Description |
---|---|
acid | Assay component ID |
aid | Assay ID. Required for registration. |
assay_component_name | Assay component name, abbreviated “acnm” within the package. Required for registration. |
assay_component_desc | Assay component description |
assay_component_target_desc | Assay component target description. Generally includes information about mechanism of action with assay target, how disruption is detected, or significance of target disruption. |
parameter_readout_type | Description of parameters measured |
assay_design_type | General description of the biological or physical process is translated into a detectable signal by assay mechanism |
assay_design_type_sub | Specific description of method through which a biological or physical process is translated into a detectable signal measured |
biological_process_target | General biological process being chemically disrupted |
detection_technology_type | General description of assay platform or detection signals measured |
detection_technology_type_sub | Description of signals measured in assay platform |
detection_technology | Specific description of assay platform used |
key_assay_reagent_type | Type of critical reactant being measured |
key_assay_reagent | Critical reactant measured |
technological_target_type | General description of technological target measured in assay platform |
technological_target_type_sub | Specific description of technological target measured in assay platform |
Field | Description |
---|---|
aeid | Assay component endpoint ID |
acid | Assay component ID. Required for registration. |
assay_component_endpoint_name | Assay component endpoint name, abbreviated “aenm” within the package. Required for registration. |
assay_component_endpoint_desc | Assay component endpoint description |
assay_function_type | Description of targeted mechanism and the purpose of the analyzed readout in relation to others from the same assay |
normalized_data_type | Normalization approach for which the data is displayed |
burst_assay | Indicator if endpoint is included in the burst distribution (1) or not (0); Burst phenomenon can describe confounding activity, such as cytotoxicity due to non-specific activation of many targets at certain concentrations. Required for registration. |
key_positive_control | Tested chemical sample expected to produce activity; Used to assess assay validity |
signal_direction | Directionality of raw data signals from assay (gain or loss); Defines analysis direction |
intended_target_type | General group of intended targets measured |
intended_target_type_sub | Specific subgroup of intended targets measured |
intended_target_family | Family of intended target measured; Populated on ToxCast chemical activity plot within CompTox dashboard |
intended_target_family_sub | Specific subfamily of intended target measured |
cell_viability_assay | Indicator of the impact of cytotoxicity in confounding (1) or no cytotoxic impact (0) |
See the Data Interpretation>Cytotoxicity Burst Distribution section for more details on “burst_assay”.
Field | Description |
---|---|
acid | Assay component ID |
acsn | Assay component source name |
Field | Description |
---|---|
aeid | Assay component endpoint ID |
assay_title | Short and descriptive title for the assay; opposed to assay component endpoint name |
assay_objectives | Purpose of the test method: Inserted after assay_component_target_desc; the claimed purpose and rationale for intended use of the method (e.g. alternative to an existing method, screening, provision of novel information in regulatory decision-making, mechanistic information, adjunct test, replacement, etc.) should be explicitly described and documented. The response measured in the assay should be put in the context of the biology/physiology leading to the in vivo response or effect. If the biological activity or response refers to a key event or molecular initiating event (MIE), provide a short description indicating what key event within an existing or developing AOP, or in relation to a mechanism or mode of action, the assay is aiming to characterize (i.e. which level of biological organization the assay may be attributed (e.g. sub-cellular, cellular, tissue, organ or individual), and where the assay might fit in the context of an existing regulatory hazard (i.e. adverse outcome). In the absence of any AOP, provide an indication of the plausible linkage between the mechanism(s) the assay is measuring and the resulting hazard endpoint. |
assay_throughput | Information about the throughput of the assay: Indicate the throughput of the assay to provide an indication of likely resource intensity e.g. low (manual assay, one chemical tested at a time), lowmoderate, moderate, moderate-high, high throughput (e.g. in 96 well-plate and higher), and qualify with e.g. approximate number of chemicals/concentrations per run. If appropriate indicate whether a manual assay could be run in a higher throughput mode. |
scientific_principles | Scientific principle of the method: Provide the scientific rationale, supported by bibliographic references to articles, for the development of the assay. A summary description of the scientific principle including the biological/physiological basis and relevance (e.g. modeling of a specific organ) and/or mechanistic basis (e.g. modeling a particular mechanism by biochemical parameters) should be described. If possible, indicate what the anchor point is within an AOP. |
biological_responses | Response and Response Measurement: Response here makes reference to any biological effect, process, or activity that can be measured. Specify precisely and describe the response and its measurement, e.g. corneal opacity measured using an opacitometer; half maximal activity concentration (AC50) derived from a competitive binding assay in human estrogen receptor assay or from the up-regulation of the proinflammatory antiangiogenic chemokine CXCL10. |
analytical_description | Data analysis: Comment on the response value in terms of a boundary or range to provide a context for interpretation. |
basic_procedures | Description of the experimental system exposure regime: Provide a summary description of the essential information pertaining to the exposure regime (dosage and exposure time including observation frequency) of the test compounds to the experimental system including information on metabolic competence if appropriate; number of doses/concentrations tested or testing range, number of replicates, the use of control(s) and vehicle. Also, describe any specialized equipment needed to perform the assay and measure the response. Indicate whether there might be potential solubility issues with the test system, and solutions proposed to address the issue. |
experimental_system | Tissue, cells or extracts utilised in the assay and the species source: indicate the experimental system for the activity or response being measured. |
xenobiotic_biotransformation | Metabolic competence of the test system: Describe and discuss the extent to which the test system can be considered metabolically competent, either by itself, or with the addition of an enzymatic fraction, if appropriate. Provide reference if available. |
proprietary_elements | Status of method development and uses: Compile information for the following sections if appropriate. Considerations could include: i) Development status: Indicate if the assay is still under development, and the estimated timeline for completion as far as possible ii) Known uses: Summarise the current and/or past use of the assay by different laboratories iii) Evaluation study: Summarise the main conclusions or refer to individual protocol if available iv) Validation study: Indicate participation in a formal validation study/studies and summarise the conclusions and their outcomes or refer to the individual protocol if available v) Regulatory use: Provide details of any potential regulatory application and of the toxicological hazard endpoint being addressed by the assay. |
See the Data Interpretation>Assay Description Documents section for more details.
Field | Description |
---|---|
chid | Chemical ID |
casn | CAS Registry Number |
chnm | Chemical name |
dsstox_substance_id | Unique identifier from U.S. EPA Distributed Structure-Searchable Toxicity (DSSTox) Database |
Chemical ID is the DSSTox GSID within the ToxCast data, but can be any integer and will be auto-generated (if not explicitly defined) for newly registered chemicals.
Field | Description |
---|---|
chemical_lists_id | Chemical List ID |
chid | Chemical ID |
dsstox_substance_id | Unique identifier from U.S. EPA Distributed Structure-Searchable Toxicity (DSSTox) Database |
list_acronym | Chemical list acronym |
list_name | Chemical list name |
list_desc | Chemical list description |
Field | Description |
---|---|
spid | Sample ID |
chid | Chemical ID |
stkc | Stock concentration |
stkc_unit | Stock concentration unit |
tested_conc_unit | The concentration unit for the concentration values in the data-containing tables |
bottle_barcode | Bottle barcode of sample |
source | Source (i.e. manufacturer or supplier) of procured bottle |
bottle_type | Type of bottle, which can reflect storage condition |
lot_number | Lot or batch number of bottle |
purity_percentage | Reported purity percentage from bottle Certificate of Analysis (CoA) |
solubility | Solubility |
The stock concentration fields in the “sample” table allow the user to track the original concentration when the neat sample is solubilized in vehicle before any serial dilutions for testing purposes. The US EPA’s ChemTrack application and database supports chemical procurement and sample management for ToxCast in vitro screening efforts.
This section provides an overview of assay registration process for the tcpl package. Before writing and processing any data to the tcpl database, the user has to register the assay and chemical information, i.e. assay identifiers (and minimal assay annotations) and chemical sample identifiers need to be available in the database before any data processing.
The definition of an “assay” is, for the purposes of this package,
broken into:
Assay source, assay, assay component, and assay endpoint are registered via tcpl scripting into a collection of tables. These assay element tables broadly describe who conducted the assay, what platform was used, what was being measured (raw readout), and how the measurement was interpreted (normalized component data). A hierarchical structure of the assay elements is as follows: assay source > assay > assay component > assay component endpoint.
As one moves down the hierarchy, each additional level has a ‘one-to-many’ relationship with the previous level. For example, an assay component can have multiple assay endpoints, but an assay endpoint can derive only from a single assay component.
Throughout the tcpl R package, the levels of assay hierarchy are defined and referenced by their auto-incremented primary keys in the tcpl database: asid (assay source ID), aid (assay ID), acid (assay component ID), and aeid (assay endpoint ID). These abbreviations mirror the abbreviations for the identifiers (ids) with “nm” in place of “id” in the abbreviations, e.g. assay_component_name is abbreviated acnm.
All processing occurs by assay component or assay endpoint, depending on the processing type (single-concentration or multiple-concentration) and level. No data is stored at the assay or assay source level. The “assay” and “assay_source” tables store annotations to help in the processing and down-stream understanding of the data. Additional details for registering each assay element and updating annotations are provided below. In addition to each assay element’s id, the minimal registration fields in order to ‘pipeline’ are:
Assay source refers to the vendor or origination of the data. To register an assay source, an unused asid must be selected to prevent overwriting of existing data. When adding a new assay source, this should be an abbreviation, as subsequent levels will build on this assay source name.
The tcplRegister function takes the abbreviation for assay_source_name, but the function will also take the unabbreviated form. The same is true of the tcplLoadA- functions, which load the information for the assay annotations stored in the database.
Assay refers to the procedure, conducted by some vendor, to generate the component data. To register an assay, an asid must be provided to map the assay to the correct assay source. One source may have many assays. To ensure consistency of the naming convention, first check how other registered assays within the assay source were conducted and named. The assay names follow an abbreviated and flexible naming convention of Source_Assay. Notable assay design features to describe the assay include:
The most distinguishing features will be selected to create a succinct assay name. Variation depends on the assay itself as well as other assays provided by the vendor. If multiple features are needed to describe an assay, order will be based on relative importance in describing the assay and the assay’s relation to other assays provided by the vendor to limit confusion. “Source_Technology_Format_Target” is a commonly used naming order. However, if one target is screened on different assay platforms by the vendor, “Source_Target_Technology” is a more appropriate naming convention. This is the case for the Tox21 assays. Additional features may be relevant, including agonist or antagonist mode, or “Follow-up” if the assay is a secondary specificity assay. Conversely, some assays utilize a cell-based format to screen a functional profile of targets. These assays follow a naming convention, Source_Format, where specific target information is defined at the component and endpoint level. Bioseek and Attagene are sources that provide cell-based assays. Considering the diversity of the data sources and high throughput assays in ToxCast, a flexible naming approach is best used in conjunction with subject matter expert discretion.
tcplLoadAid(what = "asid", val = 1)
tcplRegister(what = "aid", flds = list(asid = 1, anm = "TOX21_ERa_BLA_Agonist", assay_footprint = "1536 well"))
When registering an assay (aid), the user must give an asid to map the assay to the correct assay source. Registering an assay, in addition to an assay_name (anm) and asid, requires assay_footprint. The assay_footprint field is used in the assay plate visualization functions (discussed later) to define the appropriate plate size. The assay_footprint field can take most string values, but only the numeric value will be extracted, e.g. the text string “hello 384” would indicate to draw a 384-well microtitier plate. Values containing multiple numeric values in assay_footprint may cause errors in plotting plate diagrams.
Assay component, or “component” for short, describes the raw data readouts. Like the previous level, one assay may have many components. To register an assay component and create an acid, an aid must be provided to map the component to the correct assay. The assay component name will build on its respective assay name, to describe the specific feature being measured in each component. If there is only one component, the component name can be the same as the assay name. If there are multiple components measured in an assay, understanding the differences, and how one component may relate to another within an assay, are important naming considerations to prevent confusion. Assay component names will usually follow the naming convention of Source_Assay_Component, where “Component” is a brief description of what is being measured.
tcplLoadAcid(what = "asid", val = 1, add.fld = c("aid", "anm"))
tcplRegister(what = "acid", flds = list(aid = 1, acnm = "TOX21_ERa_BLA_Agonist_ratio"))
The final piece of assay information needed is the assay component source name (abbreviated acsn), stored in the “assay_component_map” table. The assay component source name is intended to simplify level 0 pre-processing by defining unique character strings (concatenating information if necessary) from the source files that identify the specific assay components. An assay component can have multiple acsn values, but an acsn must be unique to one assay component. Assay components can have multiple acsn values to minimize the amount of data manipulation required (and therefore potential errors) during the level 0 pre-processing if assay source files change or are inconsistent. The unique character strings (acsn) get mapped to acid.
Assay component endpoint, or “endpoint” for short, represents the normalized component data. To register an endpoint and create an aeid, an acid must be provided to map the endpoint to the correct component. In past tcpl versions, each component could have up to two endpoints therefore endpoint names would express directionality (*_up/_down*). tcpl v3+ allows bidirectional fitting to capture both the gain and loss of signal. Therefore with tcpl v3+ , the endpoint name will usually be the same as the component name.
tcplLoadAeid(fld = "asid", val = 1, add.fld = c("aid", "anm", "acid", "acnm"))
tcplRegister(what = "aeid", flds = list(acid = 1, aenm = "TOX21_ERa_BLA_Agonist_ratio", normalized_data_type = "percent_activity", export_ready = 1, burst_assay = 0))
Registering an assay endpoint also requires the normalized_data_type field. The normalized_data_type is used when plotting and currently, the package supports the following values: percent_activity, log2_fold_induction, log10_fold_induction, and fold_induction. Any other values will be treated as “percent_activity.”
Other required fields to register an assay endpoint do not have to be explicitly defined and will default to 0 if not provided. These fields represent Boolean values (1 or 0, 1 being TRUE ). The export_ready field indicates (1) the data is done and ready for export or (0) still in progress. The burst_assay field is specific to multiple-concentration processing and indicates (1) the assay endpoint is included in the burst distribution calculation or (0) not.
There are circumstances where assay, assay component, and assay endpoint names change. The aid, acid, and aeid are considered more stable in the database, and these auto-incremented keys should not change. To revise naming for assay elements, the correct id must be specified in the tcplUpdate statement to prevent overwriting data.
Reasons for name changes could include:
Thus, users should be advised that while assay naming is used to infer information about the biology of the assay, assay naming will change over time to reflect progress in building ToxCast as a data resource.
With the minimal assay information registered, the next step is to register the necessary chemical and sample information with tcplRegister. The tcplLoadChem function returns all chemical information or can be filtered for user specified parameters, e.g. the chemical name (chnm) and chemical id (chid).
The “chdat” example below contains the sample and chemical information for the data that will be loaded. The following shows an example of how to load chemical information. Similar to the order in registering assay information, the user must first register chemicals, then register the samples that map to the corresponding chemical.
## spid casn chnm dsstox_substance_id
## <char> <char> <char> <char>
## 1: Tox21_400088 80-05-7 Bisphenol A DTXSID7020182
## 2: Tox21_303655 521-18-6 5alpha-Dihydrotestosterone DTXSID9022364
## 3: Tox21_110011 150-30-1 Phenylalanine DTXSID9023463
## 4: Tox21_400081 22224-92-6 Fenamiphos DTXSID3024102
## 5: DMSO 67-68-5 Dimethyl sulfoxide DTXSID2021735
## 6: Tox21_400037 95-83-0 4-Chloro-1,2-diaminobenzene DTXSID5020283
## code chid
## <char> <char>
## 1: C80057 20182
## 2: C521186 22364
## 3: C150301 23463
## 4: C22224926 24102
## 5: C67685 21735
## 6: C95830 20283
## Register the Chemicals ##
# Obtain chemicals already registered in the database.
cmap <- tcplLoadChem()
# Find chemicals in 'chdat' that are not registered yet.
chdat.register <- chdat[!(chdat$code %in% cmap$code)]
# Register the chemicals not yet in the database.
tcplRegister(what = "chid", flds = chdat.register[,unique(.SD), .SDcols = c("casn", "chnm", "dsstox_substance_id", "code", "chid")])
The “chdat” dataset contains a map of sample to chemical information, but chemical and sample information have to be registered separately because a chemical could potentially have multiple samples. Registering chemicals only takes a chemical CAS registry number (casn) and name (chnm). In the above example, only the unique chemicals were loaded. The casn and chnm fields have unique constraints; trying to register multiple chemicals with the same name or CAS registry number is not possible and will result in an error. With the chemicals registered and loaded, the samples can be registered by mapping the sample ID (spid) to the chemical ID. Note, the user needs to load the chemical information to get the chemical IDs then merge the new chemical IDs with the sample IDs from the original file by chemical name or CASRN.
tcplRegister(what = "spid",
flds = merge(chdat[ , list(spid, casn)],
chdat.register[ , list(casn, chid)], by = "casn")[ , list(spid, chid)])
Optionally, the user can subdivide the chemical IDs based on presence in different chemical lists using tcplLoadChemList. These chemical lists are curated by the US EPA in the Distributed Structure-Searchable Toxicity (DSSTox) database. Chemicals can belong to more than one chemical list, and will be listed as separate entries when loading chemical list information.
Level 0 pre-processing can be done on virtually any high-throughput screening application to prepare data for ToxCast data processing. In the ToxCast program, level 0 processing is done in R by vendor/dataset-specific scripts. The individual R scripts act as the “laboratory notebook” for the data, with all pre-processing decisions clearly commented and explained. The standard Level 0 format to enter the pipeline is identical between testing paradigms, single concentration (“SC”) and multi-concentration (“MC”) as described in the Database Structure>Level 0 section.
Level 0 pre-processing reformats the raw, source data into the standard format for the pipeline, and also can make manual transformations to the data as pre-normalization steps. All manual transformations to the data should be very well documented with justification. Common examples of manual transformations include fixing a sample ID typo, or changing well quality value(s) to 0 after finding obvious problems like a plate row/column missing an assay reagent.
Each row in the level 0 pre-processing data represents one well-assay component combination, containing 11 fields. The only field in level 0 pre-processing not stored at level 0 is the assay component source name (acsn). The assay component source name should be some concatenation of data from the assay source file that identifies the unique assay components. When the data are loaded into the database, the assay component source name is mapped to assay component ID through the assay_component_map table in the database. Assay components can have multiple assay component source names, but each assay component source name can only map to a single assay component.
Field | Description | Required_NA_allowed |
---|---|---|
‘acsn’ or ‘acid’ | Assay component source name can be used to map to assay component ID, or acid can be directly provided | No |
spid | Sample ID | No |
apid | Assay plate ID | Yes |
rowi | Assay plate row index, as an integer | Yes |
coli | Assay plate column index, as an integer | Yes |
wllt | Well type | No |
wllq | Well quality: 1 was good, else 0 | No |
conc | Concentration in micromolar | No |
rval | Raw assay component value or readout from vendor | Yes |
srcf | Filename of the source file containing the data | No |
The N/A column indicates whether the field can be NULL in the pre-processed data when writing Level 0. In past versions of tcpl, there were some exceptions where concentrations could be NULL. All conc values must be numeric for processing. For blank and neutral control wells, NULL concs will automatically be set to zero if not provided. If other concs are NULL, user will get a warning and be unable to write Level 0. Additionally, if the raw value is NULL, well quality must be 0.
The well type (wllt) field is used to differentiate wells in numerous applications, including normalization and definition of the assay noise level. Package users are encouraged to utilize additional well types (for example, well types “x”, “y”, “z”) or suggest new methods to better accommodate their data.
Well.Type | Description |
---|---|
t | Test compound |
c | Gain-of-signal control in multiple concentrations |
p | Gain-of-signal control in single concentration |
n | Neutral/negative control |
m | Loss-of-signal control in multiple concentrations |
o | Loss-of-signal control in single concentration |
b | Blank well |
v | Viability control |
Before writing and processing any data to the database, the user must register the assay and chemical information, as described above. tcpl includes three functions for adding new data:
The final step in level 0 pre-processing is to load the data into the database. The tcplWriteLvl0 function loads data into the database by checking each required field for expected input, such as correct class or registered sample IDs and concentration values for test wells. Assay component source name, if provided, will map to the appropriate acid. The type argument is used throughout the package to distinguish the screening paradigm (SC or MC) and therefore processing required.
As a check to confirm data has been written, the tcplLoadData function is used to load data from the database into user’s R session. Furthermore, the tcplPrepOtpt function can be used in conjunction with tcplLoadData to prepare the data in a readable format with additional chemical and assay annotation information. See Data Retrieval sections for further details.
# Load the level 0 data from the database to R.
tcplLoadData(lvl = 0, fld = "acid", val = 1, type = "mc")
tcplPrepOtpt(tcplLoadData(lvl = 0, fld = "acid", val = 1, type = "mc"))
In the loaded Level 0 data, acsn is replaced with correct acid and the m0id field is added. These “m#” fields in MC data are primary keys for each level of data and can link the various levels of data. All keys are auto-generated and will change anytime data are reprocessed. Note, the primary keys only change for the levels affected, e.g. if the user reprocesses level 1, the level 0 keys will remain the same.
All processing in the tcpl package occurs at the assay component or assay endpoint level. There is no capability in either SC or MC processing to combine data from multiple assay components or assay endpoints. Any combining of data must occur before or after processing. For example, a ratio of two raw values could be processed if the user calculated the ratio during the custom pre-processing and uploaded values as a single “component”.
Once the Level 0 data are loaded, data processing occurs via the tcplRun function for both SC and MC multiple-concentration screening. tcplRun can either take a single ID (acid or aeid, depending on the processing type and level) or an asid. If given an asid, the tcplRun function will attempt to process all corresponding components/endpoints. When processing by acid or aeid, the user must supply correct ID for each level.
The processing is sequential, and every level of processing requires successful processing at the antecedent level. Any processing changes will trigger a “delete cascade,” removing any subsequent data affected by the processing change to ensure complete data fidelity. For example, processing level 3 data will first require data from levels 4 through 6 to be deleted for the corresponding IDs. Changing method assignments will also trigger a delete cascade for any corresponding data.
For tcplRun, the user must supply a starting level (slvl) and ending level (elvl). There are four phases of processing, as reflected by messages printed in console: (1) data for the given IDs are loaded, (2) the data are processed, (3) data for the same ID in subsequent levels are deleted, and (4) the processed data is written to the database. The ‘outfile’ parameter can give the user the option of print this output text to a file. If an id fails while processing multiple levels, the function will not attempt to process the failed ids in subsequent levels. When finished processing, a list indicating the processing success of each ID is returned. For each level processed, the list will contain two elements: (1) “l#” a named Boolean vector where TRUE indicates successful processing, and (2) “l#_failed” containing the names of any ids that failed processing, where “#” is the processing level.
Processing of multiple assay components or endpoints can be executed simultaneously. This is done with the internal utilization of the mclapply function from the parallel package. Parallel processing is done by id. Depending on the system environment and memory constraints, the user may wish to use more or less processing power. For processing on a Windows operating system, the default is mc.cores = 1, unless otherwise specified. For processing on a Unix-based operating system, the default is mc.cores = NULL i.e. to utilize all cores except for one, which is necessary for ‘overhead’ processes. The user can specify more or less processing power by setting the “mc.cores” parameter to the desired level. Note, this specification should meet the following criteria $1 \space \leq \space \mathit{mc.cores} \space \leq \space \mathit{detectCores()}-1$.
Type | Level | InputID | MethodID |
---|---|---|---|
SC | Lvl1 | acid | aeid |
SC | Lvl2 | aeid | aeid |
MC | Lvl1 | acid | N/A |
MC | Lvl2 | acid | acid |
MC | Lvl3 | acid | aeid |
MC | Lvl4 | aeid | N/A |
MC | Lvl5 | aeid | aeid |
MC | Lvl6 | aeid | aeid |
In this table, the Input ID column indicates the ID used for each processing step where Method ID indicates the ID used for assigning methods for data processing, when necessary. SC1 requires an acid input, but the methods are assigned by aeid. The same is true for MC3 processing. SC1 and MC3 are the normalization steps and convert acid to aeid. Only MC2 has methods assigned by acid.
The processing requirements vary by screening paradigm and level, which later sections cover in detail. However, in general, specific method assignments will be required to accommodate different experimental designs or data processing approaches.
To promote reproducibility, all method assignments are saved in the database and utilize methods described in the available list of methods for each processing level. In general, methods data are stored in the “_methods” and “_id” tables for each corresponding level. For example, the “sc1” table is accompanied by the “sc1_methods” table which stores the available list of methods for SC1, and the “sc1_aeid” table which stores the method assignments and execution order.
There are three functions to easily modify and load method
assignments:
The following code blocks provides examples of the method-related functions:
## Methods Assignment ##
# For illustrative purposes, assign level 2 MC methods to ACIDs 97, 98, and 99.
# First check for available methods.
mthds <- tcplMthdList(lvl = 2, type = "mc")
mthds[1:2]
# Assign some methods to ACID 97, 98, and 99.
tcplMthdAssign(lvl = 2,
id = 97:99,
mthd_id = c(3, 4, 2),
ordr = 1:3,
type = "mc")
# Check the assigned methods for ACID 97, 98, and 99 in the database.
tcplMthdLoad(lvl = 2, id = 97:99, type = "mc")
# Methods can be cleared one at a time for the given id(s)
tcplMthdClear(lvl = 2, id = 99, mthd_id = 2, type = "mc")
# Check the assigned methods for the single id updated, namely ACID 99.
tcplMthdLoad(lvl = 2, id = 99, type = "mc")
# Or all methods can be cleared for the given id(s)
tcplMthdClear(lvl = 2, id = 97:98, type = "mc")
# Check the assigned methods for the all updated ids, namely ACID 97 and 98.
tcplMthdLoad(lvl = 2, id = 97:98, type = "mc")
Later sections of this vignette provide level-specific method assignment examples and more details on the methods themselves. Most examples will reflect commonly-used methods assigned, but one should consider the data at hand and all methods available for the level prior to assigning.
Data normalization occurs in both SC and MC processing paradigms at levels 1 and 3, respectively. While the two paradigms use different methods, the normalization approach is the same for both. Data normalization does not have to occur within the package as pre-normalized data can be loaded into the database at Level 0. However, data must be zero-centered. Thus, the data must either be zero-centered in Level 0 pre-processing or the user must pick a methodology from the associated level 1 and 3 methods to zero-center the data before model fitting occurs.
Fold-change and a percent of control are typical approaches to normalization. Given data must be zero-centered, fold-change data in general is log-transformed. Log-scale transformations for fold-change data is typically base 2 (log2), but other bases may be more appropriate in some circumstances.
Normalizing to a percent of control requires three normalization methods:
Normalizing to fold-change also requires three methods: 1. one to define the baseline value, 2. one to calculate the fold-change, and 3. one to log-transform the fold-change values (“resp.fc”).
(cval) is the corrected response value for a test well defined in MC2. Methods defining a baseline value (bval) have the “bval” prefix, methods defining the positive control value (pval) have the “pval” prefix. Pval may be set as 0 if no positive control wells are provided and measuring decreases in signal. Finally, methods that calculate the final response value have the “resp” prefix. For example, “resp.log2” does a log-transformation of the response value using a base value of 2. The formulae for calculating the percent of control and fold-change response are listed in equations 1 and 2, respectively. Note that the fold-change calculation divides by the baseline value and thus must have some non-zero values associated with the baseline to successfully calculate fold-change.
$$ resp.pc = \frac{cval - bval}{pval - bval}*100 $$
$$ resp.fc = \frac{cval}{bval} $$
Order matters when assigning normalization methods. The bval, and pval if normalizing as a percent of control, need to be calculated prior to calculating the response value. Examples of normalization schemes are presented below:
Examples of Normalization Schemes | ||
Fold Change | %Control | |
Scheme 1 |
|
|
Scheme 2 |
|
|
Scheme 3 |
|
|
If the data does not require any normalization, the “none” method will be assigned. The “none” method simply copies the input data to the response field. Without assigning “none”, the response field will not get generated and processing will fail.
With tcpl v2 , responses were only fit in the positive analysis direction. Therefore, a signal in the negative direction needed to be “flipped” to the positive direction during normalization. Multiple endpoints stemming from one component were created to enable multiple normalization approaches when the assay measured gain and loss of signal. Negative direction data was inverted by multiplying the final response values by −1 via the “resp.multneg1” methods. For tcpl v3 onward, the tcplFit2 package is utilized which allows for bidirectional fitting, meaning the “resp.multneg1” method is now only required in special cases.
In addition to the required normalization methods, the user can apply
additional methods to transform the normalized values. For example,
“resp.blineshift.50.spid” corrects for baseline deviations by spid. A
complete list of available methods, by processing type and level, can be
accessed with tcplMthdList. More information
is also available in package documentation,
??tcpl::Methods
.
The goal of single-concentration processing is to identify potentially active compounds from a broad screen at a single concentration. SC processing consists of 2 levels:
Level | Description |
---|---|
Lvl 0 | Pre-processing: Vendor/dataset-specific pre-processing to organize heterogeneous raw data to the uniform format for processing by the tcpl package |
Lvl 1 | Normalize: Apply assay endpoint-specific normalization listed in the ‘sc1_aeid’ table to the raw data to define response |
Lvl 2 | Activity Call: Collapse replicates by median response, define the response cutoff based on methods in the ‘sc2_aeid’ table, and determine activity |
Level 1 processing converts the assay component to assay endpoint(s), defines the normalized-response value (resp), and optionally, derives the baseline value (bval) and positive control value (pval). The purpose of level 1 is to normalize the raw values to either the percentage of a control or fold-change from baseline.
The first step in beginning the processing is to identify which assay endpoints stem from the assay component(s) being processed. With the corresponding endpoints identified, the appropriate methods can be assigned.
# Load the 'aeid' values for acid 2 ##
tcplLoadAeid(fld = "acid", val = 2)
# Assign the level 1 methods to aeid 1 and 2 ##
tcplMthdAssign(lvl = 1, # processing level
id = 1:2, # assay endpoint ID's to assign methods
mthd_id = c(1, 11, 13), # method(s) to be assigned
ordr = 1:3, # order the method(s) should be applied
type = "sc") # the data/processing type
Above, methods 1, 11, and 13 were assigned for both endpoints. The method assignments instruct the processing to: (1) calculate bval for each assay plate ID by taking the median of all data where the well type equals “n”; (2) calculate a fold-change with respect to the bval (i.e. $\frac{resp}{\mathit{bval}}$); (3) log-transform the fold-change values with base 2. For a complete list of normalization methods see tcplMthdList(lvl = 1, type = “sc”) or ?SC1_Methods .
If a user needs to add a method to the end of a normalization sequence, as shown above, then the user can use a second method assignment statement. For example, if AEID 2 should indicate a change of directionality to be more biologically interpretable, all responses can be multiplied by −1. Reminder, the order of methods assignment matters, particularly in the normalization step.
# Assign a fourth step to the normalization processing - for AEID 2 only.
tcplMthdAssign(lvl = 1, # processing level
id = 2, # assay endpoint ID's to assign methods
mthd_id = 16, # method(s) to be assigned
ordr = 4, # order the method(s) should be applied
type = "sc") # the data/processing type
With the normalization methods defined, the data are ready for SC1 processing. Before normalization occurs within tcplRun, all wells with well quality (wllq) 0 are removed.
Level 2 processing defines the baseline median absolute deviation (bmad), collapses any replicates by sample ID (spid), and determines the activity.
Before the data are collapsed by spid, the bmad is calculated as the median absolute deviation of all treatment wells (wllt = t - default option) or neutral control wells (wllt = n). The calculation defines bmad for the entire endpoint. If additional data is added, the bmad for the associated endpoints will be recalculated. Note, this bmad equation is different from MC screening.
bmadsc = 1.4826 * median(|yi − ỹ|)
Where yi is the ith observation of all wells within a well type in the assay component and ỹ is the median across all yi’s. The constant value, 1.4826, is the default adjustment value used in the underlying R function to ensure bmad is a consistent estimator of the standard deviation (σ) assuming the sample size (N) of the observations is large and they are normally distributed (i.e. Gaussian), see mad() in R and unbiased mad for further details.
Method | Description | Observations | ID | Details |
---|---|---|---|---|
1 | Median absolute deviation (MAD) of all treatment wells across the assay component (acid). | yi = y(s, w) | s ∈ {1, ..., nacid}, w = t | |s indicates the sample id within an ‘acid’ & w indicates the well type |
2 | Median absolute deviation (MAD) of all blank wells across the assay component (acid). | yi = y(s, w) | s ∈ {1, ..., nacid}, w = n | |s indicates the sample id within an ‘acid’ & w indicates the well type |
To collapse the data by spid, the median response of replicates at each concentration index is calculated. SC data screening involves testing at 1-3 concentrations, which is insufficient for fitting. Data are then further collapsed by taking the maximum of those median values (max_med). Once collapsed, such that each endpoint-sample has one value, the activity is determined. For an active hit call (hitc), the sample’s max_med must be greater than a specified efficacy cutoff (coff).
In the below code, methods are assigned such that the cutoff value is log2(1.2). Thus, if the maximum median value (max_med) is greater than or equal to the efficacy cutoff (coff = log2(1.2)), then the sample ID is considered active and the hit call (hitc) is set to 1. The coff is defined as the maximum of all values given by the assigned level 2 methods. Failing to assign a level 2 method will result in every sample being called active. For a complete list of level 2 methods, see tcplMthdList(lvl = 2, type = “sc”) or ?SC2_Methods. With the methods assigned and the cutoff set, the data are ready for SC2 processing.
# Assign a cutoff value of log2(1.2) ##
tcplMthdAssign(lvl = 2, # processing level
id = 1, # assay endpoint ID's to assign methods
mthd_id = 3, # method(s) to be assigned
type = "sc") # the data/processing type
## SC2 processing for acid 1 ##
sc2_res <- tcplRun(id = 1, slvl = 2, elvl = 2, type = "sc")
Sometimes it is necessary to prevent hitcalling for responses in the biologically irrelevant direction. SC2 contains two methods for overwriting the max_med value. If applied, negative hitcalls will be given for any max_med greater or less than cutoff in the biologically unintended direction, by comparing the max_med to either the positive or negative cutoff instead examining absolute values.
Method | Method_Name | Description |
---|---|---|
25 | ow_bidirectional_gain | Reponses in in the positive direction only are biologically relevant, therefore overwrite the max_med and max_tmp values, which were calculated using absolute value, to a calculation using a true maximum for uni-directional data. |
27 | ow_bidirectional_loss | Responses in the negative direction only are biologically relevant, therefore overwrite the max_med and max_tmp values, which were calculated using absolute value, to a calculation using a true minimum for uni-directional data. |
The goal of multiple-concentration processing is to estimate the activity, potency, efficacy, and other parameters for sample-assay pairs.
Level | Description |
---|---|
Lvl 0 | Pre-processing: Vendor/dataset-specific pre-processing to organize heterogeneous raw data to the uniform format for processing by the tcpl package |
Lvl 1 | Index: Define the replicate and concentration indices to facilitate all subsequent processing |
Lvl 2 | Transform: Apply assay component (acid) specifc transformations listed in the ‘mc2_acid’ table to the raw data to define the corrected data |
Lvl 3 | Normalize: Apply assay endpoint (aeid) specifc normalization listed in the ‘mc3_aeid’ table to the corrected data to define response |
Lvl 4 | Fit: Model the concentration-response data utilizing ten objective curve-fitting functions from tcplfit2: (1) constant, (2) hill, (3) gain-loss, (4) polynomial-linear, (5) polynomial-quadratic, (6) power, (7) exponential-2, (8) exponential-3, (9) exponential-4, (10) exponential-5 |
Lvl 5 | Model Selection/Acitivty Call: Select the winning model, define the response cutoff based on methods in the ‘mc5_aeid’ table, and determine activity |
Lvl 6 | Flag: Flag potential false positive and false negative fits |
Lvl 7 | Extrapolate: Convert bioactive concentrations to Adminstered Equivalent Doses |
Level 1 processing defines the replicate and concentration index fields to facilitate downstream processing. Because of cost, availability, physicochemical, and technical constraints, screening efforts may utilize different experimental designs. The resulting data may contain an inconsistent number of concentration groups, concentration values, and technical replicates. To enable quick and uniform processing, level 1 processing explicitly defines concentration and replicate indices as 1…N for increasing concentrations and technical replicates, where 1 represents the lowest concentration or first technical replicate.
To assign replicate and concentration indices, we assume one of two experimental designs. The first design assumes samples are plated in multiple concentrations on each assay plate, such that the concentration series all fall on a single assay plate. The second design assumes samples are plated in a single concentration on each assay plate, such that the concentration series falls across many assay plates.
For both experimental designs, data are ordered by source file (srcf), assay plate ID (apid), column index (coli), row index (rowi), sample ID (spid), and concentration (conc). Concentration is rounded to three significant figures to correct for potential rounding errors. After ordering the data, a temporary replicate ID is created for each concentration series. For test compounds in experimental designs with the concentration series on a single plate and all control compounds, the temporary replicate ID consists of the sample ID, well type (wllt), source file, assay plate ID, and concentration. The temporary replicate ID for test compounds in experimental designs with concentration series that span multiple assay plates is defined similarly, but does not include the assay plate ID.
Once the data are ordered, and the temporary replicate ID is defined, the data are scanned from top to bottom and the replicate index (repi) incremented every time a replicate ID is duplicated. Then, for each replicate, the concentration index (cndx) is defined by ranking the unique concentrations, with the lowest concentration starting at 1. No methods need to be applied and the following demonstrates how to carry out the MC1 processing and look at the resulting data:
The package also contains a function, tcplPlotPlate , for visualizing the data at the assay plate level. This function can be used to visualize the data at levels 1 to 3.
In the generated figure, the row and column indices are printed
along the respective edges of the plate, with the raw observed values in
each well represented by color. While the plate does not give sample ID
information, the letter/number codes in the wells indicate the well type
and concentration index, respectively. Wells with poor well quality
(wllq==0
in Level 0) will display with an “X.” The title of
the plate display lists the assay component/assay endpoint and the assay
plate ID (apid).
Level 2 processing removes data where the well quality (wllq) equals 0 and defines the corrected value (cval) field. MC2 also allows for additional transformation of the raw values at the assay component level. Examples of transformations include basic arithmetic manipulations to complex spatial noise reduction algorithms, such as aggregation across biological replicates.
Every assay component needs at least one transformation method assigned to complete level 2 processing, even if no transformations are necessary. In the following example, the “none” method will be assigned so MC2 processing can be completed.
## Methods Assignment ##
# Assign the level 2 transformation method 'none' to ACID 1.
tcplMthdAssign(lvl = 2, # processing level
id = 1, # assay component ID's to assign methods
mthd_id = 1, # method(s) to be assigned
ordr = 1, # order of the method(s) should be assigned
type = "mc") # the data/processing type
## MC2 processing for acid 1 ##
mc2_res <- tcplRun(id = 1, slvl = 2, elvl = 2, type = "mc")
For the complete list of level 2 transformation methods currently available, see tcplMthdList(lvl = 2, type = “mc”) or ?MC2_Methods for more details. The coding methodology used to implement the methods is beyond the scope of this vignette, but, in brief, the method names in the database correspond to a function name in the list of functions returned by mc2_mthds() (the mc2_mthds function is not exported, and not intended for use by the user). Each of the functions in the list given by mc2_mthds only return expression objects that the processing function called by tcplRun executes in the local function environment to avoid making additional copies of the data in memory. We encourage suggestions for new methods.
Level 3 processing converts the assay component to assay endpoint(s) and defines the normalized-response value field (resp); and optionally, the baseline value (bval) and positive control value (pval) fields. MC3 processing normalizes the corrected values to either the percentage of a control or to fold-change from baseline. The normalization process is discussed in greater detail in the Data Normalization section. A primary distinction between MC2 and MC3 processing is ID for each level.
The user first needs to check which assay endpoints stem from the the assay component queued for processing. With the corresponding aeids identified, the normalization methods can be assigned. In the following example, methods 17, 9, and 7 were assigned for both endpoints. These methods involve: (1) calculating bval for each assay plate ID by taking the median of all data where the well type equals “n” or the well type equals “t” and the concentration index is 1 or 2; (2) calculating a fold-change over bval; (3) log-transforming the fold-change values with base 2. For a complete list of normalization methods see tcplMthdList(lvl = 3, type = “mc”) or ?MC3_Methods . With normalization methods defined, the data are ready for MC3 processing.
# Look at the assay endpoints for acid 1 ##
tcplLoadAeid(fld = "acid", val = 1)
## Methods Assignment ##
# Assign the baseline calculation and normalization methods to aeids 1 and 2.
tcplMthdAssign(lvl = 3, # processing level
id = 1:2, # assay endpoint ID to assign methods
mthd_id = c(17, 9, 7), # method(s) to be assigned
ordr = 1:3, # order the method(s) should be applied
type = "mc") # the data/processing type
## MC3 processing for acid 1 ##
mc3_res <- tcplRun(id = 1, slvl = 3, elvl = 3, type = "mc")
Notice that MC3 processing takes an acid, not an aeid, as the input ID. As mentioned in previous sections, the user must will assign MC3 normalization methods by aeid then process by acid. The MC3 processing will attempt to process all endpoints for a given component. If one endpoint fails for any reason (e.g., does not have appropriate methods assigned), the processing for the entire component fails.
NOTE:
The user can provide either an assay source id (asid) or ‘id’ within tcplRun . If the starting level (slvl) is less than 4, then ‘id’ is interpreted as an acid. When slvl is greater than or equal to 4 the ‘id’ is interpreted as an aeid. If an ‘id’ fails, no results are loaded into the database and the ‘id’ is not included in the cue for subsequent processing levels.
Level 4 processing models the activity of each concentration-response series. Each series is bidirectionally fit using methods available in the tcplfit2 R package (Sheffield et al., 2021). Bidirectional fitting means curve inversion is not necessary as in past versions (i.e. observed ‘negative/decreasing’ responses multiplied by −1).
Level 4 processing establishes a noise-band for the endpoint using the baseline median absolute deviation (bmad). Here, the bmad is calculated by the baseline response values, either untreated control wells (e.g. ‘wllt = n’; neutral solvent wells like DMSO) or test samples from the two lowest concentrations (i.e. ‘wllt = t’ & concentration index is 1 or 2). The bmad calculation is done across the entire endpoint. If additional data is added, the bmad values for all associated assay endpoints is recalculated.
bmadmc = 1.4826 * median(|yi − ỹ|) Where yi is the ith baseline observation as defined by the assigned method and ỹ is the median of all the baseline observations. The constant value, 1.4826, is the default adjustment value used in the underlying R function to ensure bmad is a consistent estimator of the standard deviation (σ) assuming the sample size (N) of the baseline observations is large and they are normally distributed (i.e. Gaussian), see mad() in R and unbiased mad for more details.
One standard deviation of baseline response is also calculated, since it is necessary for the calculation of the benchmark response (BMR) in both the curve-fitting and hit-calling functions from tcplfit2 , tcplfit_core and tcplhit2_core , respectively. One standard deviation of the baseline response will be estimated using the two lowest concentration groups of test samples or neutral control wells across all chemicals, depending on bmad method selected. BMR, like bmad, may change if additional data is added. Thus, the BMD estimates may also change. See the Data Interpretation>Benchmark Dose section for more details.
Method | Description | Observations | ID | Details |
---|---|---|---|---|
1 | Median absolute deviation (MAD) of all observations in the lowest two concentrations of across samples (spid) in the assay endpoint (aeid). Standard deviation (SD) of all observations in the lowest two concentrations of across samples (spid) in the assay endpoint (aeid). | yi = y(s, w, d) | s ∈ {1, ..., naeid}, w = t, d ∈ {1, 2} | s indicates the sample id within an ‘aeid’, w indicates the well type, & d indicates the concentration group index |
2 | Median absolute deviation (MAD) of all observations in the solvent/untreated control observations across samples (spid) in the assay endpoint (aeid). Standard deviation (SD) of all observations solvent/untreated control observations of across samples (spid) in the assay endpoint (aeid). | yi = y(s, w) | s ∈ {1, ..., naeid}, w = n | s indicates the sample id within an ‘aeid’, w indicates the well type |
Before the model parameters are estimated, a set of summary values are calculated for each concentration-response series:
The “mean” and “median” response values are defined as the mean or median of all observed response values at each concentration. In other words, the maximum median is the maximum of all median response values across all concentrations in the series. Similarly, the minimum median is the true minimum (or max in the negative direction) of all median response values across the concentrations in the series.
The following code demonstrates how to load the MC3 data for a single aeid.
## Evaluate the MC3 Data ##
# Load the MC3 data from the database.
mc3 <- tcplLoadData(lvl = 3,
type = 'mc',
fld = 'aeid',
val = 80)
# Prepare the data into a readable format.
mc3 <- tcplPrepOtpt(mc3)
For demonstration purposes, the mc_vignette R data object is provided in the package since the vignette is not directly connected to such a database. The mc_vignette object contains a subset of data from levels 3 through 5 from invitrodb v4.2. The following code loads the example mc3 data object, then plots the concentration-response series for an example spid with the summary estimates indicated.
# Load the example data from the `tcpl` package.
data(mc_vignette, package = 'tcpl')
# Allocate the level 3 example data to `mc3`.
mc3_example <- mc_vignette[['mc3']]
# level 3 does not store logc anymore, create it for plotting purposes
mc3_example[, logc := log10(conc)]
# Obtain the MC4 example data.
mc4_example <- mc_vignette[["mc4"]]
# Obtain the minimum response observed and the 'logc' group - 'resp_min'.
level3_min <- mc3_example %>%
dplyr::group_by(spid, chnm) %>%
dplyr::filter(resp == min(resp)) %>%
dplyr::filter(spid == "01504209")
# Obtain the maximum response observed and the 'logc' group - 'resp_max'.
level3_max <- mc3_example %>%
dplyr::group_by(spid, chnm) %>%
dplyr::filter(resp == max(resp)) %>%
dplyr::filter(spid == "01504209")
# Obtain the level 3 data and 'center' estimates for responses per 'logc' group.
level3_summary <- mc3_example %>%
dplyr::filter(spid == "01504209") %>%
dplyr::select(., c(spid, chnm, logc, resp)) %>%
dplyr::group_by(spid, chnm, logc) %>%
dplyr::summarise(mean_resp = mean(resp), med_resp = median(resp))
## Generate Individual Summary Plots ##
# Plot the mean responses for each log-concentration group.
A <- mc3_example %>%
dplyr::filter(spid == "01504209") %>%
ggplot(data = ., aes(logc, resp)) +
geom_point(pch = 1, size = 2) +
geom_point(data = level3_summary,
aes(x = logc, y = mean_resp,
col = 'mean responses'),
alpha = 0.75,size = 2) +
scale_color_manual(values = 'paleturquoise3',
aesthetics = 'col') +
labs(lty = "", colour = "")+
xlab(expression(paste(log[10],"(Concentration) ", mu, "M"))) +
ylab(expression(paste(log[2], "(Fold Induction)"))) +
ggtitle("Mean Responses") +
theme_bw() +
theme(legend.position = 'bottom')
# Plot the median responses for each log-concentration group.
B <- mc3_example %>%
dplyr::filter(spid == "01504209") %>%
ggplot(data = .,aes(logc,resp)) +
geom_point(pch = 1, size = 2) +
geom_point(data = level3_summary,
aes(x = logc, y = med_resp,
col = 'median response'),
alpha = 0.75, size = 2) +
scale_color_manual(values = 'hotpink',
aesthetics = 'col') +
labs(lty = "", colour = "")+
xlab(expression(paste(log[10], "(Concentration) ", mu, "M"))) +
ylab(expression(paste(log[2], "(Fold Induction)"))) +
ggtitle("Median Responses") +
theme_bw() +
theme(legend.position = 'bottom')
# Plot the maximum mean & median responses at the related log-concentration -
# 'max_mean' & 'max_mean_conc'.
C <- mc3_example %>%
dplyr::filter(spid == "01504209") %>%
ggplot(data = .,aes(logc, resp)) +
geom_point(pch = 1, size = 2) +
geom_point(data = dplyr::filter(mc4, spid == "01504209"),
aes(x = log10(max_mean_conc), y = max_mean,
col = 'maximum mean response'),
alpha = 0.75, size = 2)+
scale_color_manual(values = 'paleturquoise3',
aesthetics = 'col') +
labs(lty = "", colour = "")+
xlab(expression(paste(log[10], "(Concentration) ", mu, "M"))) +
ylab(expression(paste(log[2], "(Fold Induction)"))) +
ggtitle(label = "Maximum Mean Response") +
theme_bw() +
theme(legend.position = 'bottom')
# Plot the maximum mean & median responses at the related log-concentration -
# 'max_med' & 'max_med_conc'.
D <- example %>%
dplyr::filter(spid == "01504209") %>%
ggplot(data = ., aes(logc, resp)) +
geom_point(pch = 1, size = 2) +
geom_point(data = dplyr::filter(mc4, spid == "01504209"),
aes(x = log10(max_med_conc), y = max_med,
col = "maximum median response"),
alpha = 0.75, size = 2)+
scale_color_manual(values = 'hotpink',
aesthetics = 'col') +
labs(lty = "", colour = "") +
xlab(expression(paste(log[10], "(Concentration) ", mu, "M"))) +
ylab(expression(paste(log[2], "(Fold Induction)"))) +
ggtitle(label = "Maximum Median Response") +
theme_bw() +
theme(legend.position = 'bottom')
# Plot the minimum & maximum observed responses.
E <- mc3_example %>%
dplyr::filter(spid == "01504209") %>%
ggplot(data = ., aes(logc, resp)) +
geom_point(pch = 1, size = 2) +
geom_point(data = level3_min,
aes(x = logc, y = resp,
col = "minimum response"),
alpha = 0.75, size = 2) +
geom_point(data = level3_max,
aes(x = logc, y = resp,
col = "maximum response"),
alpha = 0.75, size = 2) +
scale_color_manual(values = c('red', 'blue'),
aesthetics = 'col') +
labs(lty = "", colour = "") +
xlab(expression(paste(log[10], "(Concentration) ", mu,"M"))) +
ylab(expression(paste(log[2], "(Fold Induction)"))) +
ggtitle(label = "Minimum & Maximum\nResponses") +
theme_bw() +
theme(legend.position = 'bottom')
# Plot the minimum & maximum experimental log-concentration groups -
# 'logc_min' & 'logc_max'.
G <- mc3_example %>%
dplyr::filter(spid == "01504209") %>%
ggplot(data = ., aes(logc, resp)) +
geom_point(pch = 1, size = 2) +
geom_vline(data = dplyr::filter(mc4, spid == "01504209"),
aes(xintercept = log10(conc_min),
col = 'minimum concentration'),
lty = "dashed") +
geom_vline(data = dplyr::filter(mc4, spid == "01504209"),
aes(xintercept = log10(conc_max),
col = 'maximum concentration'),
lty = "dashed") +
scale_color_manual(values = c('red', 'blue'),
aesthetics = 'col') +
labs(lty = "", colour = "") +
xlab(expression(paste(log[10], "(Concentration) ", mu, "M"))) +
ylab(expression(paste(log[2], "(Fold Induction)"))) +
ggtitle(label = "Minimum & Maximum\nConcentrations") +
theme_bw() +
theme(legend.position = 'bottom')
## Compile Summary Plots in One Figure ##
gridExtra::grid.arrange(
A,B,C,D,E,G,
nrow = 3, ncol = 2,
top = mc3[which(mc4[,spid] == "01504209"), aenm]
)
These plots illustrate summary estimates calculated as part of the level 4 processing, which occurs prior to dose-response modeling. Each plot depicts the observed concentration-response data as white circles, where the x-axis is base 10 log-transformed concentration values. In the upper plots, the mean response values for each concentration group is depicted as turquoise circles (left) where as median response values for each concentration group are hot-pink circles (right). The middle plots depict mean and median responses, but only shows the maximum mean (max_mean) and median (max_med) response estimates (left and right, respectively). The minimum observed response (min_resp) is depicted with a blue circle and the maximum observed response value (max_resp) with a red circle (left lower). Finally, the minimum (min_logc) and maximum max_logc log10-scale concentrations are depicted, respectively, with blue and red vertical dashed lines (right lower).
After summary values are obtained for each concentration-response series, all parametric models in tcplFit2 are fit to each series. Available model details are provided below:
Model | Abbreviation | Equations | OutputParameters | Details |
---|---|---|---|---|
Constant | cnst | f(x) = 0 | Parameters always equals ‘er’. | |
Linear | poly1 | f(x) = ax | a (y-scale) | |
Quadratic | poly2-monotonic only | $f(x) = a(\frac{x}{b}+(\frac{x}{b})^{2})$ | a (y-scale) b (x-scale) | |
Quadratic | poly2-biphasic | f(x) = b1 * x + b2 * x2 | a (y-scale) b (x-scale) | |
Power | pow | f(x) = axp | a (y-scale) p (power) | |
Hill | hill | $f(x) = \frac{tp}{1 + (\frac{ga}{x})^{p}}$ | tp (top) ga (gain AC50) p (gain-power) | Concentrations are converted internally to log10 units and optimized with f(x) = tp/(1 + 10^(p*(gax))), then ga and ga_sd are converted back to regular units before returning. |
Gain-Loss | gnls | $f(x) = \frac{tp}{(1 + (\frac{ga}{x})^{p} )(1 + (\frac{x}{la})^{q} )}$ | tp (top) ga (gain AC50) p (gain power) la (loss AC50) q (loss power) | Concentrations are converted internally to log10 units and optimized with f(x) = tp/[(1 + 10^(p(gax)))(1 + 10^(q(x-la)))], then ga, la, ga_sd, and la_sd are converted back to regular units before returning. |
Exponential 2 | exp2 | $f(x) = a*(exp(\frac{x}{b}) - 1)$ | a (y-scale) b (x-scale) | |
Exponential 3 | exp3 | $f(x) = a*(exp((\frac{x}{b})^{p}) - 1)$ | a (y-scale) b (x-scale) p (power) | |
Exponential 4 | exp4 | $f(x) = tp*(1-2^{\frac{-x}{ga}})$ | tp (top) ga (AC50) | |
Exponential 5 | exp5 | $f(x) = tp*(1-2^{-(\frac{x}{ga})^{p}})$ | tp (top) ga (AC50) p (power) |
Most models in tcplfit2 assume the background response is zero and the absolute response (or initial response) is increasing. In other words, these models fit a monotonic curve in either direction. The polynomial 2 (poly2) model is an exception with two parameterization options. The biphasic parameterization is what is used in tcpl . A biphasic poly2 model fits responses that are increasing first and then decreasing, and vice versa (assuming the background response is zero). If biphasic responses are not reasonable, data can be fit using the monotonic-only parameterization in a standalone application of tcplfit2_core with the parameter biphasic=FALSE assigned. This argument is not available in tcpl. All data is fit bidirectionally then responses in unintended direction may be indicated with negative hit calls if “overwrite” MC5 methods are applied.
Upon completion of model fitting, each model gets a success designation: 1 if the model optimization converges, 0 if the optimization fails, and NA if ‘nofit’ was set to TRUE within tcplFit2::tcplfit2_core function. Similarly, if the Hessian matrix was successfully inverted then 1 indicates a successful covariance calculation (cov); otherwise 0 is returned. Finally, in cases where ‘nofit’ was set to TRUE (within tcplFit2::tcplfit2_core ) or the model fit failed the Akaike information criterion (aic), root mean squared error (rme), model estimated responses (modl), model parameters (parameters), and the standard deviation of model parameters (parameter sds) are set to NA. A complete list of model output parameters is provided below:
FitParameters | Description |
---|---|
er | Error term |
success | Success of Fit/Model Convergenece |
cov | Success of Covariance |
aic | Akaike Information Criteria |
rme | Root Mean Squared Error |
modl | Vector of Model Estimated Values at Given Concentrations |
parameters | Model Parameter Values |
parameters sds | Standard deviation of Model Parameter Values |
pars | Vector of Parameter Names |
sds | Vectors of Parameter Standard Deviation Names |
Maximum likelihood estimation is utilized in the model fitting algorithm to estimate model parameters for all models. Even though tcplfit2 allows the maximum likelihood estimation to assume the error follows a normal or Student’s t-distribution, tcpl assumes the error always follows a t-distribution with four degrees of freedom. Heavier (i.e., wider) tails in the t-distribution diminish the influence of outlier values, and produce more robust estimates than the more commonly used normal distribution. Robust model fitting removes the need to eliminate potential outliers prior to fitting.
Let t(z, ν) be the Student’s t-distribution with ν degrees of freedom, yi be the observed response at the ith observation, and μi be the estimated response at the ith observation. We calculate zi as:
$$ z_{i} = \frac{y_{i} - \mu_{i}}{exp(\sigma)}, $$
where σ is the scale term.
Then the log-likelihood is
$$ \sum_{i=1}^{n} [\ln\left(t(z_{i}, 4)\right) - \sigma]\mathrm{,} $$
where n is the number of observations.
The following plots provide simulated concentration-response curves to illustrate the general curve shapes captured by tcplFit2 models. When fitting ‘real-world’ experimental data, the resulting curve shapes will minimize the error between the observed data and the concentration-response curve. Thus, the shape for each model fit may or may not reflect what is illustrated below:
## Example Data ##
# example fit concentration series
ex_conc <- seq(0.03, 100, length.out = 100)
## Obtain the Continuous Fit of Level 4 Model Estimates ##
fits <- data.frame(
# log-scale concentrations
logc = log10(ex_conc),
# parametric model fits from `tcplfit2`
constant = tcplfit2::cnst(ps = c(er = 0.1), ex_conc),
poly1 = tcplfit2::poly1(ps = c(a = 3.5, er = 0.1),x = ex_conc),
poly2.mono.only = tcplfit2::poly2(ps = c(a = 0.13, b = 2, er = 0.1), x = ex_conc),
poly2.biphasic = tcplfit2::poly2bmds(ps = c(b1 = 14, b2 = -0.1, er = 0.1), x = ex_conc),
power = tcplfit2::pow(ps = c(a = 1.23, p = 1.45, er = 0.1), x = ex_conc),
hill = tcplfit2::hillfn(ps = c(tp = 750, ga = 5, p = 1.76, er = 0.1), x = ex_conc),
gnls = tcplfit2::gnls(ps = c(tp = 750, ga = 15, p = 1.45, la = 50, q = 1.34, er = 0.1),
x = ex_conc),
exp2 = tcplfit2::exp2(ps = c(a = 0.45, b = 13.5, er = 0.1), x = ex_conc),
exp3 = tcplfit2::exp3(ps = c(a = 1.67, b = 12.5, p = 0.87, er = 0.1), x = ex_conc),
exp4 = tcplfit2::exp4(ps = c(tp = 895, ga = 15, er = 0.1), x = ex_conc),
exp5 = tcplfit2::exp5(ps = c(tp = 793, ga = 6.25, p = 1.25, er = 0.1), x = ex_conc)
) %>%
reshape2::melt(data = .,measure.vars = c(
"constant",
"poly1","poly2.mono.only","poly2.biphasic","power",
"hill","gnls","exp2","exp3","exp4","exp5"
))
## Updated Colors ##
fit_cols <-
# choose 10 distinct colors
viridis::magma(n = 11, direction = 1) %>%
# darken the original colors to make them more visible
colorspace::darken(., amount = 0.2)
## Plot ##
fits %>%
ggplot() +
geom_line(aes(x = logc, y = value, lty = variable, colour = variable)) +
facet_wrap(facets = "variable") +
theme_bw() +
labs(lty = "Models", colour = "Models") +
scale_colour_manual(values = fit_cols) +
ggtitle("General Shape of Models Included in `tcplfit2`") +
xlab(expression(paste(log[10], "(Concentration) ", mu, "M"))) +
ylab("Response")
This figure contains simulated concentration-response curves to illustrate the general underlying curve shape covered by each of the models included in the tcplfit2 package and used on the back-end of the level 4 data processing in tcpl. Each sub-plot in the figure corresponds to a single parametric model included in the model fitting process and has a corresponding color and line type to accompany it. All sub-plots are plotted such that the x-axis represents the log-transformed concentration (base = 10) and the y-axis represents the response values.
For demonstrating the assignment of MC4 methods, method ‘1’ will be specified to calculate that the bmad and estimate one standard deviation of baseline using the two lowest concentration groups of treatment wells (well type, or wllt, equal to “t”).
Two examples of MC4 method assignments are provided below: (1) for a single assay endpoint and (2) all assay endpoints in containing string using tcplGetAeid . The assignment in the second approach can adapted for any subset of aeids.
## Methods Assignment #1 ##
# Assign the MC4 processing methods to aeid 80
tcplMthdAssign(
lvl = 4, # processing level
id = 80, # assay endpoint ID(s) to assign method(s)
mthd_id = c(1), # method(s) to be assigned
ordr = 1, # order the method(s) should be applied
type = "mc") # the data/processing type
## Methods Assignment #2 ##
# Obtain the 'aeid' Values for all endpoints containing "ATG" string.
# "ATG" is the abbreviated assay source name of Attagene.
atg.aeid <- tcpl::tcplGetAeid(name = "ATG")
## Assign the MC4 processing methods for subset of aeids
tcpl::tcplMthdAssign( lvl = 4,
id = atg.aeid[, aeid],
mthd_id = c(1), ordr = 1, type = "mc")
With the methods assigned, the MC4 processing can be completed for the desired set of aeids. After MC4 is processed, the user can load the model fit information from database.
A subset of MC4 data is available within the mc_vignette object.
The level 4 data includes fields for each of the ten model fits as well as the ID fields, as defined here. Model fit information are prefaced by the model abbreviations (e.g. cnst, hill, gnls, poly1, etc.). The fields ending in success indicate the convergence status of the model, where 1 means the model converged, 0 otherwise. NA values indicate the fitting algorithm did not attempt to fit the model. Smoothed model fits of the concentration-response data from the MC4 data object are displayed below:
# Load the example data from the `tcpl` package.
data(mc_vignette, package = 'tcpl')
# Allocate the level 3 example data to `mc3`.
mc3_example <- mc_vignette[['mc3']]
# level 3 does not store logc anymore, create it for plotting purposes
mc3_example[, logc := log10(conc)]
# Obtain the MC4 example data.
mc4_example <- mc_vignette[["mc4"]]
## Create a Sequence of Concentration Values within Observed Range ##
X <- seq(
mc4_example[which(mc4_example[, spid] == "01504209"), conc_min],
mc4_example[which(mc4_example[, spid] == "01504209"), conc_max],
length.out = 100
)
## Obtain the Continuous Fit of Level 4 Model Estimates ##
# Apply each model fit to continous concentration values (X) and estimated
# parameters from 'tcplfit2'.
estDR <- mc4_example %>%
dplyr::filter(spid == "01504209") %>%
dplyr::reframe(
cnst = tcplfit2::cnst(.[, c(cnst_er)], x = X),
poly1 = tcplfit2::poly1(.[, c(poly1_a, poly1_er)], x = X),
poly2 = tcplfit2::poly2(.[, c(poly2_a, poly2_b, poly2_er)], x = X),
power = tcplfit2::pow(.[, c(pow_a, pow_p, pow_er)], x = X),
hill = tcplfit2::hillfn(.[, c(hill_tp, hill_ga, hill_p)], x = X),
gnls = tcplfit2::gnls(.[, c(gnls_tp, gnls_ga, gnls_p, gnls_la, gnls_q, gnls_er)], x = X),
exp2 = tcplfit2::exp2(.[,c(exp2_a, exp2_b, exp2_er)], x = X),
exp3 = tcplfit2::exp3(.[,c(exp3_a, exp3_b, exp3_p, exp3_er)], x = X),
exp4 = tcplfit2::exp4(.[,c(exp4_tp, exp4_ga, exp4_er)], x = X),
exp5 = tcplfit2::exp5(.[,c(exp5_tp, exp5_ga, exp5_p, exp5_er)], x = X) )
# Format data into a data.frame for ease of plotting.
estDR <- cbind.data.frame(X, estDR) %>%
reshape2::melt(data = .,measure.vars = c(
"cnst", "poly1", "poly2", "power", "hill", "gnls", "exp2", "exp3", "exp4", "exp5"))
## Updated Colors ##
fit_cols <-
# choose 10 distinct colors
viridis::magma(n = 10,direction = 1) %>%
# darken the original colors to make them more visible
colorspace::darken(., amount = 0.2)
## Plot the Model Fits from Level 4 ##
mc3_example %>%
dplyr::filter(spid == "01504209") %>%
ggplot(.,aes(x = logc, y = resp))+
geom_point(pch = 1, size = 2)+
geom_line(data = estDR,
aes(x = log10(X), y = value, colour = variable, lty = variable)) +
labs(colour = "Models", lty = "Models") +
scale_colour_manual(values = fit_cols) +
xlab(expression(paste(log[10], "(Concentration) ", mu, "M"))) +
ylab(expression(paste(log[2], "(Fold Induction)"))) +# )+
ggtitle(
label = paste("Level 4 Model Fits",
mc4_example[which(mc4_example[,spid] == "01504209"), dsstox_substance_id],
sep = "\n"),
subtitle = paste("Assay Endpoint: ",
mc4_example[which(mc4_example[, spid] == "01504209"), aenm])) +
theme_bw()
The plot depicts the observed concentration-response data with white circles, where the x-axis is base 10 log-transformed concentration values. All the ten model fits are displayed and distinguished by color and line-type.
How well the model is fitting the data (i.e. goodness of fit) can be approximated by the Akaike Information Criterion (AIC) and the root mean square error (RMSE or RME). For the AIC, let log(ℒ(θ̂, y)) be the log-likelihood of the model θ̂ given the observed values y, and K be the number of parameters in θ̂, then,
AIC = −2log (ℒ(θ̂, y)) + 2K.
The RMSE is given by
$$\mathrm{RMSE} = \sqrt{\frac{\sum_{i=1}^{N} (y_{i} - \mu_{i})^2}{N}}\mathrm{,}$$
where N is the number of observations, and μi and yi are the estimated and observed values at the ith observation, respectively.
Level 5 processing determines the winning model and activity for the concentration series, bins all of the concentration series into fitc categories, and calculates various potency estimates.
The model with the lowest AIC value is selected as the winning model (modl ), and is used to determine the activity (or hit call) for the concentration series. If two models have equal AIC values, then the simpler model (i.e. model with fewer parameters) wins. Additionally, if the constant model describes the response best out of all the models fit then modl is reported as ‘none’. An example of side-by-side comparison of AIC values from each of the ten model fits for the example dataset mc_vignette is provided.
dsstox_id | cnst | hill | gnls | poly1 | poly2 | pow | exp2 | exp3 | exp4 | exp5 |
---|---|---|---|---|---|---|---|---|---|---|
DTXSID80379721 | 4.35 | -18.972 | -14.972 | -13.934 | -20.77 | -19.941 | -21.241 | -19.071 | -6.019 | -18.996 |
DTXSID2020216 | 1.938 | -15.577 | -11.577 | -7.373 | -21.385 | -17.577 | -17.572 | -15.587 | -1.356 | -15.583 |
DTXSID1040619 | -7.471 | -19.086 | -15.086 | -13.759 | -21.494 | -21.03 | -20.861 | -19.022 | -7.44 | -19.086 |
DTXSID1026081 | 0.014 | -27.292 | -23.292 | -9.758 | -31.691 | -29.314 | -29.283 | -27.315 | -1.36 | -27.305 |
DTXSID9032589 | -6.489 | -28.185 | -24.185 | -27.936 | -30.586 | -30.401 | -30.784 | -29.155 | -19.685 | -28.258 |
The summary values and estimated parameters from the winning model are stored in the respective mc5 and mc5_param tables. The activity of each concentration-response series is determined by calculating a continuous hit call that may be further binarized into active or inactive, depending on the level of stringency required by the user; herein, hitc < 0.9 are considered inactive. The efficacy cutoff value (coff) is defined as the maximum of all values given by the methods assigned at level 5. When two or more methods (i.e. cutoff values) are applied for processing, the largest cutoff value is always selected as the cutoff for the endpoint. In the event only one method is applied, then that will serve as the efficacy cutoff for the endpoint. Failing to assign a level 5 method will result in every concentration series being called active. For a complete list of level 5 methods, see tcplMthdList(lvl = 5) or ?MC5_Methods . See the Data Interpretation section for more details on hit calls and cutoff.
While the ToxCast pipeline supports bidirectional fitting, sometimes it is necessary to censor the hitc of curves fit in the biologically irrelevant direction. There are two methods for overwriting the hitc value, and if applied, these will overwrite the hitc value for any biologically irrelevant curve by flipping the hitc to a negative value.
Table 11: Level 5 overwrite (ow) methods for hitcalls in unintended direction. | ||
Method | Method_Name | Description |
---|---|---|
27 | ow_bidirectional_loss | Multiply winning model hitcall (hitc) by -1 for models fit in the positive analysis direction. Typically used for endpoints where only negative responses are biologically relevant. |
28 | ow_bidirectional_gain | Multiply winning model hitcall (hitc) by -1 for models fit in the negative analysis direction. Typically used for endpoints where only positive responses are biologically relevant. |
The example we include in this vignette for demonstrating the assignment of level 5 methods specifies three different efficacy cutoff estimates for consideration. These efficacy cutoff estimates include 3 * bmad, log2(1.2), and 5 * bmad, which correspond to mthd_id assignments 1, 3, and 5 respectively, and the largest of these three values will be selected as the cutoff for the endpoint. With the methods assigned, the data are ready for MC5 processing.
# Assign the MC5 Processing Methods to aeid 80
tcplMthdAssign(
lvl = 5, # processing level
id = 80, # assay endpoint ID(s) to assign method(s)
mthd_id = c(1, 3, 5), # method(s) to be assigned
ordr = 1:3, # order the method(s) should be assigned
type = "mc") # the data/processing type
#MC5 processing for aeid 80
tcpl::tcplRun(
id = atg.aeid[, aeid], # assay endpoint id to pipeline
slvl = 5L, # level to start pipelining on
elvl = 5L, # level to end pipelining on
type = 'mc' # endpoint processing type - 'mc' = "multiple concentrations"
)
# Load MC 5 data for aeid 80
mc5 <- tcplLoadData(lvl = 5,
type = 'mc',
fld = 'aeid',
val = 80,
add.fld = TRUE)
# Prepare the data into a readable format.
mc5 <- tcplPrepOtpt(mc5)
A subset of MC5 data is available in the mc_vignette object, and includes fields for the best model fit, the potency estimates, other estimates from the best model fit, as well as the ID fields. The user can visualize the model fitting results using the tcplPlot functions. See the Data Retrieval with invitrodb and via API sections for more information.
# Allocate the level 5 data in `mc_vignette` to the `mc5` object.
mc5_example <- mc_vignette[["mc5"]]
For demonstrative purposes, an alternative visual of the model fits from MC4 and the best model as well as the potency estimates from MC5 data is produced below.{#mc5_plot}
## Obtain Data ##
# Load the example data from the `tcpl` package.
data(mc_vignette,package = 'tcpl')
# Allocate the level 3 example data to `mc3`.
mc3_example <- mc_vignette[['mc3']]
# level 3 does not store logc anymore, create it for plotting purposes
mc3_example[, logc := log10(conc)]
# Obtain the MC4 example data.
mc4_example <- mc_vignette[["mc4"]]
# Obtain the MC4 example data.
mc5_example <- mc_vignette[["mc5"]]
# First, we need to obtain the subset of data related to spid = "01504209",
# which is our example spid.
mc3_ss <- mc3_example %>% dplyr::filter(spid == "01504209") # Level 3 - conc-resp series
mc4_ss <- mc4_example %>% dplyr::filter(spid == "01504209") # Level 4 - model fits
mc5_ss <- mc5_example %>% dplyr::filter(spid == "01504209") # Level 5 - best fit & est.
# Next, we need to obtain the smooth curve estimate for the best model found
# in the Level 5 analyses of the `tcpl` pipeline.
# See Level 4 example above for how estDR is calculated.
estDR <- estDR %>%
dplyr::mutate(., best_modl = ifelse(variable == mc5_ss[, modl],
yes = "best model", no = NA))
## Generate a Base Concentration-Response Plot ##
basePlot <- mc3_ss %>%
# Observed Concentration-Response Data
ggplot()+
geom_point(aes(x = logc,y = resp),pch = 1,size = 2) +
# Cutoff Band
geom_rect(data = mc5_ss,
aes(xmin = log10(conc_min), xmax = log10(conc_max), ymin = -coff, ymax = coff),
alpha = 0.15, fill = "skyblue") +
# Best Model Fit
geom_line(data = dplyr::filter(estDR, variable == mc5_ss[,modl]),
aes(x = log10(X), y = value,color = mc5_ss[,modl])) +
scale_colour_manual(values = c("royalblue3"), aesthetics = "color") +
# Other Model Fits
geom_line(data = dplyr::filter(estDR,variable != mc5_ss[, modl]),
aes(x = log10(X), y = value, lty = variable),
alpha = 0.3, show.legend = TRUE) +
# Legend Information
labs(lty = "Other Models", color = "Best Fit") +
# Titles and Labels
xlab(expression(paste(log[10], "(Concentration) ", mu, "M"))) +
ylab(expression(paste(log[2], "(Fold Induction)"))) +# )+
ggtitle(
label = paste("Level 5 Best Model Fit",
mc4_ss[which(mc4_ss[, spid] == "01504209"), dsstox_substance_id],
sep = "\n"),
subtitle = paste("Assay Endpoint: ",
mc4_ss[which(mc4_ss[,spid] == "01504209"), aenm])) +
# Background Plot Theme
theme_bw()
## Potency Estimate Layers ##
# First, we need to obtain/assign colors for the potency estimates to be displayed.
potency_cols <-
# choose 5 distinct colors
viridis::plasma(n = 5, direction = -1) %>%
# darken the original colors to make them more visible
colorspace::darken(., amount = 0.1)
## Compile the Full Level 5 Plot ##
linePlot <-
# Start with the `basePlot` object.
basePlot +
# Next, add the various potency layers.
# BMD
geom_hline(
data = mc5_ss,
aes(yintercept = bmr),
col = potency_cols[1]
) +
geom_segment(
data = mc5_ss,
aes(x = log10(bmd), xend = log10(bmd), y = -0.5, yend = bmr),
col = potency_cols[1]
) +
geom_hline(
data = mc5_ss,
aes(yintercept = coff),
col = potency_cols[2]
) +
geom_segment(
data = mc5_ss,
aes(x = log10(acc), xend = log10(acc), y = -0.5, yend = coff),
col = potency_cols[2]
) +
geom_hline(
data = mc5_ss,
aes(yintercept = max_med * 0.5),
col = potency_cols[3]
) +
geom_segment(
data = mc5_ss,
aes(
x = log10(ac50), xend = log10(ac50),
y = -0.5, yend = max_med * 0.5
),
col = potency_cols[3]
) +
geom_hline(
data = mc5_ss,
aes(yintercept = max_med * 0.1),
col = potency_cols[4]
) +
geom_segment(
data = mc5_ss,
aes(
x = log10(ac10), xend = log10(ac10),
y = -0.5, yend = max_med * 0.1
),
col = potency_cols[4]
) +
geom_hline(
data = mc5_ss,
aes(yintercept = max_med * 0.05),
col = potency_cols[5]
) +
geom_segment(
data = mc5_ss,
aes(
x = log10(ac5), xend = log10(ac5),
y = -0.5, yend = max_med * 0.05
),
col = potency_cols[5]
)
# create data table for potency estimate points
mc5_points <- mc5_ss %>%
select(bmd, acc, ac50, ac10, ac5) %>%
tidyr::pivot_longer(everything(), names_to = "Potency Estimates") %>%
mutate(x = log10(value)) %>%
mutate(mc_color = potency_cols) %>%
mutate(`Potency Estimates` = toupper(`Potency Estimates`))
yvals <- mc5_ss %>%
select(bmr, coff, max_med) %>%
tidyr::pivot_longer(everything()) %>%
select(value) %>%
mutate(reps = c(1, 1, 3)) %>%
tidyr::uncount(reps) %>%
mutate(y = value * c(1, 1, .5, .1, .05)) %>%
select(y)
mc5_points <- mc5_points %>% cbind(yvals)
# add Potency Estimate Points and set colors
fullPlot <- linePlot + geom_point(
data = mc5_points,
aes(x = x, y = y, fill = `Potency Estimates`), shape = 21, cex = 2.5
) +
scale_fill_manual(values = mc5_points %>% arrange(`Potency Estimates`) %>% pull(mc_color))
## Display the Compiled Plot ##
fullPlot
Additional information on derivations on potency estimates is found in Data Interpretation>Potency Estimates. After curve fitting, all concentration series are also assigned a fit category (fitc) based on similar characteristics and shape. See the Data Interpretation>Fit Category section for more details.
In addition to the continuous hitc and the fitc, cautionary flags on curve-fitting can provide context to interpret potential false positives (or negatives) in ToxCast data, enabling the user to decide the stringency with which to filter these targeted in vitro screening data. These flags are programmatically generated and indicate characteristics of a curve that need extra attention or potential anomalies in the curve or data. See the Data Interpretation>Flags section for more details.
For invitrodb v4.2 onward, a new mc7 table contains pre-generated AED values using several potency metrics from invitrodb and a subset of models from the High-throughput Toxicokinetics R package httk. AEDs are generated in a separate script using the httk R package. This is done separately due to the resource-intensive nature of running the Monte Carlo simulations to get estimates of plasma concentration for the median (50th %-ile) and most sensitive (95th %-ile) toxicokinetic individuals. Moreover, this is applied to both the 3-compartment steady state (3compartments) model and the physiologically-based toxicokinetic (pbtk) model for all chemicals included in invitrodb v4.2 (generation of the table as configured in the current code took 24h using 40 cores). See the Administered Equivalent Dose section.
This section includes a practical applications for single- and multiple-concentration data, from methods assignment through data processing.
After SC0 is loaded into the database, SC1 and SC2 methods can be assigned. Once assigned, data can be processed.
After MC0 is loaded into the database, methods can be assigned. Once assigned, data can be processed. This can be done in two ways: (A) from start to finish (i.e. Level 0 to 5) with the assay component ID (acid) or (B) Level 0 to 3 with the assay component ID (acid) and Level 4 to 5 with the assay endpoint ID (aeid). Option A may be helpful if data needs to be processed completely, whereas Option B or derivatives of Option B may be helpful when adjusting methods.
## Methods Assignment
# No MC1 methods needed
tcplMthdAssign(lvl = 2, id = 1, mthd_id = c(3,4,2), ordr = 1:3, type = "mc")
tcplMthdAssign(lvl = 3, id = 2, mthd_id = 1, ordr = 1, type = "mc")
tcplMthdAssign(lvl = 4, id = 2, mthd_id = 1, ordr = 1:2, type = "mc")
tcplMthdAssign(lvl = 5, id = 2, mthd_id = c(1,3,5), ordr = 1:3, type = "mc")
tcplMthdAssign(lvl = 5, id = 2, mthd_id = c(1,3,5), ordr = 1:3, type = "mc")
## Assign the Number of Processing Cores.
mycores <- 1 # If users do NOT want to leverage parallel computing.
# Users that want to leverage parallel computing set to > 1, but less than the total number of cores
# (i.e. need at least 1 core open for overhead). If not provided, this will be assumed.
# "parallel::detectCores()" can be run to understand the maximum allowed number of cores.
## Option A: MC0-5 Processing by acid
tcplRun(id = 80, type = "mc", slvl = 0L, elvl = 5L, mc.cores = 20)
##Option B: MC0-3 Processing by acid, followed by MC4-6 by aeid
tcplRun(id = list$acid, type = "mc", slvl = 0L, elvl = 3L)
tcplRun(id = list$aeid, type = "mc", slvl = 4L, elvl = 5L)
After fitting, a continuous hit call (hitc) is calculated as the product of three proportional weights. Several potency estimates are also calculated for the winning model, including activity concentrations at specified levels of response, such as concentration at 50% of maximal activity (AC50) and concentration at activity observed at the cutoff (ACC), and a benchmark dose (BMD) at a specified benchmark response (BMR). This section will review the statistics behind the activity and potency estimates available for each concentration-response series and how the user may interpret these values alongside other available information, such as fit categories, representative samples, and cytotoxicity burst thresholds. Overall, this Data Interpretation section seeks to enhance user confidence in reviewing and using ToxCast data for different tasks.
In tcpl v2, activity hit calls (hitc) were binary, where 0 was negative, 1 was positive, and −1 corresponded to concentration-response series that tcpl was “unable to fit” (e.g., <4 concentrations). In tcpl v3 onwards, the hitc is the product of three proportional weights, and the resulting continuous value is between 0 and 1, though the values are not normally distributed and tend to approach 0 or 1. Hitcalls that approach 1 indicate concentration-response series with biological activity in the measured response (i.e. ‘active’ hit). Hitc is typically binarized into active or inactive designations, depending on the level of stringency required by the user. For current ToxCast work, a hitc greater than or equal to 0.90 is labeled active, whereas anything less was considered inactive. This threshold of 0.90 was based on other tcplfit2 implementations with in vitro screening data (Nyffeler et al., 2023) and reflects the apparent bimodal nature of the hitc distribution, where a preponderance of the hitc fall between 0 and 0.1 and 0.9 and 1.0. Users may interpret the continuous hitc into active or inactive designations based on different thresholds. Further testing through implementation of this new functionality may reveal appropriate thresholds for different applications or assays. The “unable to fit” series now appear as model “none” with a hitc of 0 (inactive). Negative hitc in tcpl v3.2 correspond to curves that suggest biological activity in an unintended direction, as further described below.
Continuous hitc as defined in tcplfit2 R package is calculated as the product of three proportional weights representing the confidence that:
See Sheffield et al., 2021 for more information on tcplfit2.
The cutoff is a user-defined level of efficacy that corresponds to statistical and/or biological relevant change from baseline for each assay endpoint. All versions of tcpl provide methods for estimation of the baseline sampling variability, or noise around the assay controls, including calculation of the median absolute deviation over all response values given by wells that may represent baseline response (the BMAD), such as the neutral or vehicle control or the first two concentrations in the concentration series for all chemicals screened as defined by Level 4 methods. Users define mc5 methods depending on assay and data type, with some common cutoff thresholds used to establish a cutoff including 3 * BMAD, 20% percent change, or 1.2*log10 fold-change. Operationally in tcpl, the efficacy cutoff value (coff) is defined as the maximum of all values given by the methods assigned at level 5. When two or more methods (i.e. cutoff values) are applied for processing, the largest cutoff value is always selected as the cutoff for the endpoint. In the event only one method is applied, then that will serve as the efficacy cutoff for the endpoint. Failing to assign a level 5 method will result in every concentration series being called active. For a complete list of level 5 methods, see tcplMthdList(lvl = 5) or ?MC5_Methods.
Curve-fitting enables determination of various metrics of potency,
i.e., concentrations at which some amount of in vitro
bioactivity is expected to occur, as illustrated above. This includes Activity Concentrations at
Specified Response and Benchmark Dose (BMD), which vary in the
mathematical approach for computing these values, noting that logic for
computation of the BMD is controlled in the R package
tcplfit2
.
An activity concentration is the estimated concentration inducing a specified level of response (activity). A common potency metric used from tcpl is the activity concentration at 50% of maximal activity, or AC50. The default baseline region is defined as ± 3 * BMAD, and the ACB is the concentration at which the model first reaches a default of 3 * BMAD, whereas ACC is defined as the concentration at which the model reaches the user-defined cutoff. Note that potency metrics such as AC50 are reported even if the hit call is inactive if a non-constant model could be fit to the data.
All versions of tcpl output the activity concentrations as described in Table 2.
Activity_Concentration_uM | Specified_Level_of_Response |
---|---|
AC5 | Concentration at 5% of the maximal response |
AC10 | Concentration at 10% of the maximal response |
AC20 | Concentration at 20% of the maximal response |
AC50 | Concentration at 50% of the maximal response |
ACB | Concentration at baseline of 3*BMAD |
ACC | Concentration at the user-defined cutoff |
AC1SD | Concentration at 1 standard deviation from baseline |
A Benchmark Dose (BMD) is the activity concentration observed at the Benchmark Response (BMR) level. In the current implementation of tcpl and tcplfit2, BMR is only defined as 1.349 standard deviations of baseline response in the two lowest concentrations of treatment wells or neutral controls wells, as defined by Level 4 methods. tcpl uses the following definitions and assumptions for setting the BMR:
BMR is a change from the mean response at baseline (𝜇(𝑏)) by some multiple (𝑐) of the standard deviation of the baseline (𝑠𝑑(𝑏)).
𝜇(𝑏) + 𝑐 ∗ 𝑠𝑑(𝑏) = 𝐵𝑀𝑅 = 𝜇(𝐵𝑀𝐷)
Here, the baseline (𝑏) is defined as samples from the two lowest concentrations across chemicals within an assay endpoint and the 𝑐 = 1.349a (Yang et al., 2017).
A 90% confidence interval around the BMD, bounded by the benchmark dose lower bound (BMDL) and the benchmark dose upper bound (BMDU), is also computed and provided to reflect the uncertainty in the BMD estimate. The calculation of these confidence intervals will occasionally fail due to a singular matrix inverse, and in these cases, BMDU and BMDL will not be reported. This case occurs when the data are especially noisy and the confidence interval around the BMD approaches infinity. The winning model may return a bmd estimate that falls outside of the tested concentration range, so bounds are placed to censor the estimate values. The lower and upper bounds for bmd estimates are 0.1 * the lowest test concentration and 10 * the the highest test concentration, respectively. If the calculated bmd estimate is below or above the lower or the upper bounds, the value at the bound will be returned as the bounded bmd estimate instead.
A hierarchical fit category (fitc) decision tree is used to bin each fit as shown in Figure 2. Each fit falls into one leaf of the tree using the described logic with the final fitc indicated with gray boxes. Abbreviations are defined as: conc = concentration; hitc = hit call; |top| = absolute value of the modeled curve top; coff = cutoff; logc(min) = minimum log10 concentration tested; logc(max) = maximum log10 concentration tested; AC50 = 50% activity concentration; AC95 = 95% activity concentration.
After curve fitting, all concentration series are assigned a fit category (fitc) based on similar characteristics and shape. Logic is based on relative activity, efficacy, and potency comparisons as shown in Figure 5. For continuity purposes, fitc numbering has been conserved from past tcpl versions. Grouping all series into fitc enables quality control and can be useful in data cleaning applications, especially when considered with Level 6 flags. In invitrodb v3-3.5, a common filtering approach removed the least reproducible curve-fits, i.e. those with very low AC50 (below the screened conc range) and low efficacy (within 1.2-fold of the cutoff) as well as 3+ flags. However, preliminary investigation into invitrodb v4.1-4.2 has suggested that removing curve fits with 4 or more flags, or possibly filtering based on specific flags in combination with fitc such as fitc 36, may be a more appropriate filtering approach due to changes in curve fitting and flags in invitrodb v4 and beyond. The stringency of filtering for flags should be explored in a fit-for-purpose way.
Fit category is largely based upon the relative efficacy and, in the case of actives, the location of the AC50 and concentration at 95% activity (an estimate of maximum activity concentration, AC95) compared to the tested concentration range. All concentration response curves are first split into active, inactive, or cannot determine. “Cannot determine” is indicative of exceptions that cannot be curve-fit, e.g. a concentration series with fewer than 4 concentrations. Active designations are determined for fitc based on whether the hitc surpasses the 0.90 threshold. For those series that are designated inactive with a hitc less than 0.90, fitc can be used to indicate to what extent the curve represents borderline inactivity via comparison of top modeled efficacy to the cutoff (i.e, the absolute value of the modeled top is less than 0.8 times the cutoff).
For active curves, efficacy, as represented by the modeled top, is compared to 1.2 times the cutoff (less than or equal to, or greater than), thereby differentiating curves that may represent borderline activity from moderate activity. Active curves also have potency metrics estimated, e.g., AC50 and AC95 values, that can be compared to the range of concentrations screened to indicate curves for which potency estimates are more quantitatively informative. Curves for which the AC50 is less than or equal to the minimum concentration tested (fitc = 36, 40) may indicate AC50 values that are less quantitatively informative than AC50 values within the concentration range screened. When the AC50 is greater than the minimum concentration tested but the AC95 is greater than or equal to the maximum concentration tested (fitc = 38, 42), it is possible the maximum activity was not fully observed in the concentration range screened. Fitc for curves where the AC50 and AC95 are both within the concentration range screened (fitc = 37, 41) represent the most quantitatively informative AC50 values.
Fitc 36 describes a curve that is of low efficacy and with a low AC50, below the concentration range screened. These are more likely to be noise or less reproducible fits. Fitc 41 and 42 are the ideal fitc for reproducible curves, as demonstrated by these two fitc comprising the majority of positive (hitc > 0.9) curves in invitrodb v4.1. Fitc 40 indicates a curve with at least moderate efficacy, but an AC50 below the concentration range screened. These chemicals may be positive or reference chemicals screened in the incorrect concentration window to observe their minimum activity. These curves may also represent high-confidence positives for which we have limited understanding of the slope of the concentration-response curve, and as such, the AC50 may be associated with more uncertainty.
In addition to the continuous hitc and the fitc, cautionary flags on curve-fitting can provide context to interpret potential false positives (or negatives) in ToxCast data, enabling the user to decide the stringency with which to filter these targeted in vitro screening data. Cautionary flags on fitting were developed in previous versions of tcpl and have been stored at Level 6. These flags are programmatically generated and indicate characteristics of a curve that need extra attention or potential anomalies in the curve or data.
For example, a curve may be considered a single point hit with activity not at the highest concentration tested, but re-inspection of the flagged curve could indicate a potential false positive. Other flags may suggest borderline activity, overfitting, or cell viability assays that are fit with gain-loss as the winning model. It is important to note that flags have no effect on the hitc or potency estimates, but they may indicate that a curve requires further examination to aid in data interpretation. A full list of flags implemented and descriptions is presented below:
Method | FlagNames | FlagDescription |
---|---|---|
5 | modl.directionality.fail | Flag series if model directionality is questionable, i.e. if the winning model direction was opposite, more responses (resp) would have exceeded the cutoff (coff). If loss was winning directionality (top < 0), flag if count(resp < −1 * coff) < 2 * count(resp > coff). If gain was winning directionality (top > 0), flag if count(resp > coff) < 2 * count(resp < −1 * coff). |
6 | singlept.hit.high | Flag single-point hit that’s only at the highest conc tested, where series is an active hit call (hitc > = 0.9) with the median response observed above baseline occurring only at the highest tested concentration tested. |
7 | singlept.hit.mid | Flag single-point hit that’s not at the highest conc tested, where series is an active hit call (hitc > = 0.9) with the median response observed above baseline occurring only at one concentration and not the highest concentration tested. |
8 | multipoint.neg | Flag multi-point miss, where series is an inactive hit call (hitc < 0.9) with multiple median responses observed above baseline. |
9 | bmd.high | Flag series if modeled benchmark dose (BMD) is greater than AC50 (concentration at 50% maximal response). This is indicates high variability in baseline response in excess of more than half of the maximal response. |
10 | noise | Flag series as noisy if the quality of fit as calculated by the root mean square error (rmse) for the series is greater than the cutoff (coff); rmse > coff |
11 | border | Flag series if borderline activity is suspected based on modeled top parameter (top) relative to cutoff (coff); |top| < = 1.2 * coff or |top| > = 0.8 * coff. |
13 | low.nrep | Flag series if the average number of replicates per concentration is less than 2; nrep < 2. |
14 | low.nconc | Flag series if 4 concentrations or less were tested; nconc < = 4. |
15 | gnls.lowconc | Flag series where winning model is gain-loss (gnls) and the gain AC50 is less than the minimum tested concentration, and the loss AC50 is less than the mean tested concentration. |
17 | efficacy.50 | Flag low efficacy hits if series has an active hit call (hitc > = 0.9) and efficacy values (e.g. top and maximum median response) less than 50%; intended for biochemical assays. If hitc > = 0.9 and coff > = 5, then flag when top < 50 or max_med < 50. If hitc > = 0.9 and coff < 5, then flag when top < log2(1.5) or max_med < log2(1.5). |
18 | ac50.lowconc | Flag series with an active hit call (hitc > = 0.9) if AC50 (concentration at 50% maximal response) is less than the lowest concentration tested; if hitc > = 0.9 and AC50 < 10logc(min), then flag. |
19 | viability.gnls | Flag series with an active hit call (hitc > = 0.9) if denoted as cell viability assay with winning model is gain-loss (gnls); if hitc > = 0.9, modl = gnls and cell_viability_assay = 1, then flag. |
20 | no.med.gt.3bmad | Flag series where no median response values are greater than baseline as defined by 3 times the baseline median absolute deviation (bmad) or less than baseline as defined by -3 times bmad; both nmed_gtbl_pos and nmed_gtbl_neg = 0, where nmed_gtbl_pos is the number of median response values > 3 * bmad and nmed_gtbl_neg is the number of median response values < −3 * bmad. |
Multiple samples of the same chemical (many spids per chid) may be tested for a given assay endpoint. The tcplSubsetChid function subsets multiple-concentration Level 5 (MC5) or single-concentration Level 2 (SC2) data to select the “best” single tested sample to represent that chemical for a given endpoint. The function uses a series of logic to select the “representative sample” in MC and SC. Representative sample designation for a tested chemical are stored in the MC5_chid and SC2_chid tables, where 1 indicates if ID (M5ID or S2ID, respectively) is a representative sample, else 0.
To select a representative sample in MC, a “consensus hitc” is made by taking the mean of all binarized sample hitc, with ties defaulting to active. After the chemical-wise hitc is made, the samples corresponding to chemical-wise hit call are logically ordered using the fit category, the number of the flags, and AC50, then the first sample for every chemical is selected. This logic prioritizes active over inactive hit calls when there is a disagreement among samples, but then attempts to pick the representative sample with the highest quality curve, with the goal of minimizing false negatives. Logic encompasses the following:
To select a representative sample in SC, a “consensus hitc” is made by taking the mean of all sample hitc, with ties defaulting to active. After the chemical-wise hitc is made, the samples in the consensus hitc are ordered based on lowest concentration tested and the sample with the maximum median response is selected.
The tcplVarMat function creates chemical-by-endpoint matrices, combining summary information for single-concentration and/or multiple-concentration screening. Such a matrix can be useful for understanding which chemicals were tested, and what hit calls and potency values resulted from this screening. For example, one might ask, “Which chemicals were screened in either single-concentration or multi-concentration for some assay endpoint, and if MC screening was performed, what potency was estimated for any observed activity?”
For variable matrices from tcplVarMat , a
representative sample is selected using the tcplSubsetChid function in
cases where a chemical-by-endpoint was tested in multiple samples
(performed separately for multiple-concentration or single-concentration
screening). When multiple sample-assay series exist for one chemical, a
single series is selected by the tcplSubsetChid
function. See the Representative Samples
section for more information.
tcplVarMat produces matrices of combined sc-mc output. For the AC50 and ACC matrices specifically, values are inserted in place to show complete views of what was tested and what the results. Further, “ac50_verbose” and “acc_verbose” replace such values with text. AC50 and ACC are:
Both SC and MC data are required for tcplVarMat. As a result, the “API” driver is not currently supported for invitrodb v4.1 since it does not return SC data.
In addition, additional matrices can be defined by the ‘add.vars’ parameter. The ‘add.vars’ parameter will take any Level 4 or 5 field and create the respective matrix.
# create matrices with all chemicals and assays; the entire database
varmat <- tcplVarMat()
# create matrices using a subset of chemicals and/or aeids
aeids <- c(80)
dtxsid <- c("DTXSID80379721", "DTXSID10379991", "DTXSID7021106", "DTXSID1026081")
varmat <- tcplVarMat(aeid = aeids, dsstox_substance_id = dtxsid)
# create extra matrices by adding vars
varmat <- tcplVarMat(aeid = aeids, add.vars = c("m4id", "resp_max", "max_med"))
tcplVarMat returns a list of chemical by assay matrices (data.tables) where the rows are given by the dsstox_substance_id and corresponding chnm (chemical name) columns and the colnames are given by assay endpoint name (aenm). To export this list, save to a .xlsx. Each matrix will have its own sheet.
Estimates of chemical concentrations that elicit cytotoxicity and/or cell stress have been informative for contextualizing bioactivity screening data in ToxCast by providing information on the likelihood that these data may be confounded by assay interference resulting from cytotoxicity and/or cell stress, particularly when a parallel or in-well estimate of cell viability is unavailable. As such, general estimates of the median and lower bound concentrations that might elicit cytotoxicity and/or cell stress in vitro have previously been calculated using the tcplCytoPt function, which considers activity across a suite of cell-based assays based on updates to previous work (Judson et al., 2016).
The burst threshold can be used to infer activity above or below an estimated threshold of generalized cell stress and/or cytotoxicity, where users can define the degree of difference between some observed bioactivity and estimates of cell stress and/or cytotoxicity needed in order to discern “selective” activity (i.e., bioactivity thought to occur independently of cell stress and/or cytotoxicity) vs. “non-selective” activity (i.e., bioactivity that appears to occur concomitantly with estimates of the concentration needed for cell stress and/or cytotoxicity).
These estimated concentration threshold values have been released in the “cytotox” table of invitrodb and are also provided on the CompTox Chemicals Dashboard (CCD) Bioactivity Summary Plot, as shown in the BPA example below.
tcplCytoPt function uses the assay endpoints listed in the burst_assay field of the “assay_component_endpoint” table, where 1 indicates including the assay endpoint in the calculation. The “burst” assay endpoints can be identified by running tcplLoadAeid(fld = “burst_assay”, val = 1) .
The cytotoxicity point is the median AC50 for a set of assay endpoints defined within the tcpl database. tcplCytoPt outputs estimates in logged and unlogged concentration units. In addition to the cytotoxicity point (cytotox_median), a lower bound estimate (cytotox_lower_bnd) is defined by the cytotoxicity point minus 3 times the calculated global median absolute deviation (cytotox_median_um −3 * globalMAD). This global MAD represents the typical amount of variance observed in AC50 values for chemicals screened in many cell stress/cytotoxicity assays, giving us a means of estimating a lower bound on the concentration window that may result in cell stress/cytotoxicity across a number of cell lines and assay technologies.
A cytotox_median is computed for any chemical in the database screened with at least two active hit calls and an active hit in at least 5% of burst assay endpoints screened from the set of assay endpoints defined as “burst” related (90 assay endpoints in invitrodb v4.2). If a chemical is a hit in fewer than 5% of burst assay endpoints screened or is a hit in only 1 assay endpoint, the default cytotox_median (3 on the log10 scale or 1000 micromolar on the arithmetic scale) will be assigned because we lack enough data to compute a median and assume some estimate of variance in cell stress/cytotoxicity data. 103 = 1000, therefore, when using micromolar units, 3 is equivalent to 1 millimolar. 1 millimolar was chosen as an arbitrary high concentration given it’s outside the typical testing range for ToxCast data and based on the principle that all compounds are toxic if given in high enough concentration.
In contrast, for a chemical to be included in a computation of the global median absolute deviation (global_MAD), the chemical must be screened in a higher number of burst assays. The global MAD is an estimate of the variance expected for a chemical tested in many cytotoxicity and cell stress assays, currently defined as greater than or equal to 60 assay endpoints annotated as “burst” endpoints and with an active hit call in at least 5% of “burst” assay endpoints tested. This typically includes chemicals that were highly screened in a multitude of assays, based on inclusion in initial ToxCast Phase I and II chemical screening. Additional filtering of burst assay data was also required to ensure only losses in cell viability were included and any cell proliferation responses were excluded (for a subset of bidirectional endpoints in the set of burst assay endpoints).
Given ToxCast includes a heterogeneous set of assays across a diverse biological space, annotations in the database help flexibly aggregate and differentiate processed data whereas assay documentation aligned with international standardization efforts can make ToxCast data more useful and interpretable for use in decision-making. The OECD Guidance Document 211 (GD211) is a standard for comprehensive assay documentation describing non-guideline in vitro test methods and their interpretation. This template is intended to harmonize non-guideline, in vitro method descriptions to allow assessment of the relevance of the test method for biological responses of interest and the quality of the data produced. Unlike the assay element annotations which are often short in a standardized format or use a controlled term list, the assay_descriptions fields have no character limit for text. A compiled report of these assay description documents are available on the ToxCast Downloadable Data page.
The highest level assumption in the in vitro to in vivo extrapolation (IVIVE) approach employed here is that the in vitro bioactive concentration in a ToxCast assay endpoint is roughly equivalent to a human plasma concentration in vivo. For a review of IVIVE and httk models for it, please see: Breen et al, 2021
For invitrodb v4.2 onward, a new MC7 table contains pre-generated AED values using several potency metrics from invitrodb and a subset of models from the High-throughput Toxicokinetics R package httk . As implemented, this MC7 table provides users with pre-calculated estimates of the in vivo human administered dose (mg/kg/day) based on the in vitro bioactive concentrations as seen in ToxCast screening data.
AEDs are generated in a separate .R script using the httk R package because of the resource-intensive nature of running the Monte Carlo simulations to get estimates of plasma concentration for the median (50th %-ile) and most sensitive (95th %-ile) toxicokinetic individuals for both the 3-compartment steady state (3compartmentss) model and the physiologically-based toxicokinetic (pbtk) model. Given the large number of chemicals and endpoints included in invitrodb v4.2, generation of the MC7 table as configured with the options below took 24 hours using 40 cores.
Options Applied when extrapolating Adminstered Equivalent Doses. | |
Parameter | Options_Applied |
---|---|
Library(httk) | Version 2.3.1 |
httk::calc_mc_oral_equiv() | species = ‘Human’ restrictive.clearance=T output.units=’mgpkgpday’ Caco2.options, which revise the fraction bioavailable using estimates of absorption and gut permeability, were kept as default (Caco2.options = list(Caco2.Pab.default = 1.6, Caco2.Fabs=TRUE, Caco2.Fgut=TRUE, overwrite.invivo=FALSE, keepit100=FALSE)) |
httk models used | 3compartmentss: employs 3 compartments and steady-state assumption with 1 mg/kg/day dosing, assumes clearance = 1/plasma concentration at steady state. When fraction unbound is unavailable, model assumes it is just a very small number pbtk: multi-compartment model that does not assume steady-state kinetics. Requires estimates of intrinsic clearance and fraction unbound; not available for quite as many chemicals as 3compartmentss |
httk and QSPRs | Quantitative structure property relationships is loaded via load_sipes2017(), load_pradeep2020(), and load_dawson2021() to be able to make AED estimates for as many chemicals as possible. |
Potency metrics used for httk::calc_mc_oral_equiv() | ac50, acc, bmd |
Filters on m4id | Hitc >= 0.9 Number of mc6 flags is < 4 Fit category is not 36. This removes borderline responses resulting in ac50 below the concentration range screened, which is not considered to be quantitatively informative. . |
There is high value in understanding the outcomes of solubilization and chemical stability in the vehicle chosen to solubilize the chemical, i.e. a chemical’s applicability domain for in vitro screening. This informs what chemicals and samples should screened in future experiments. It also helps inform future structural models to understand which chemicals will be stable and detectable in solubilization, and further provide insight into possible degradation products that could be synthesized or purchased. Most critically, this information promotes understanding of uncertainty in estimates of initial experimental concentration of chemicals.
To establish a resource of applicability domain information at the substance and sample level, a retrospective analysis of the analytical QC data for the ToxCast/Tox21 chemical library was conducted. This involved reviewing legacy reports from gas chromatography-mass spectrometry (GCMS), liquid chromatography-mass spectrometry (LCMS), and Nuclear Magnetic Resonance (NMR) experiments. Additional Analytical QC, such as for the PFAS chemical library, and integration efforts are ongoing.
Table 4: Fields in the Chemical_Analytical_QC Table. | |
Field | Description |
---|---|
dtxsid | Unique identifier from U.S. EPA Distributed Structure-Searchable Toxicity (DSSTox) Database |
chnm | Chemical name |
spid | Sample ID |
qc_level | Level of analytical QC: substance or sample |
pass_or_caution | Indication of analytical QC pass or caution. Considered caution unless T0 or T4 in (A, B), or T0 and T4 are I with S call |
t0 | Grade at T0 (Time zero: Compounds freshly taken out of freezer). Grade options include: A: Molecular Weight (MW) Confirmed, Purity >90% B: MW Confirmed, Purity 75-90% C: MW Confirmed, Purity 50-75% D: CAUTION Purity <50% Ac: Purity > 90% CAUTION Low Conc. 5-30% of expected value Bc: Purity 75-90% CAUTION, Low Conc. 5-30% of expected value Cc: Purity 50-75% CAUTION, Low Conc. 5-30% of expected value Fc: CAUTION Very Low Conc. <5% of expected value. Biological Activity Unreliable Z: MW Confirmed, No Purity Info I: ISOMERS Two or more isomers detected M: DEFINED MIXTURE Two or more components F: CAUTION Incorrect MW. Biological Activity Unreliable Fns: CAUTION No Sample Detected. Biological Activity Unreliable U: Unknown/Inconclusive ND: Not Determined W: Sample Withdrawn |
t4 | Grade at T4 (Time 4 months: Compounds kept at room temperature for 4 months). Same options at T0. |
call | Call options include: S: Stable T: CAUTION Chemical transformation L: CAUTION Physical loss X: CAUTION Unstable, reason undetermined |
annotation | Annotation note from analytical QC manual curation |
flags | Interpretative flags set based on observed substance or sample level QC (T0, T4, calls) or physicochemical properties. Flag options incude: Room temperature stability decreases over time: T0 in (A, B, C) AND call is in (T,L,X)) Low concentration possible: T0 OR T4 in (Ac,Bc,Cc). Call may be pass or caution depending on T0 Extreme loss at room temperature over time: T0 in (A, B, C) AND t4 in (Fc, Fns) Missing data for room temperature stability: T4 in (U,ND,NA,Z,W) Initial purity between 75-90%: T0 in (B, Bc) Initial purity between 50-75%: T0 in (C,Cc) Likely fail: t0 in (D, F, Fns, W) OR T0 in (Z) & T4 in (D, F,Fc,Fns) Examine physicochemical properties: Predicted log10-VP >= 1 OR logKow >= 6.5 Confirmed isomer or mixture: T0 in (M,I) AND T4 in (M,ND,I) Missing purity information: T0 in (Z) AND T4 in (A,Ac,B,C) |
average_mass | Mass |
log10_vapor_pressure_OPERA_pred | OPERA predicted log10 vapor pressure |
logKow_octanol_water_OPERA_pred | OPERA predicted Octanol-water partition coefficient |
The tcplLoadAsid, tcplLoadAid, tcplLoadAcid, and tcplLoadAeid functions load relevant assay ids and names for the respective assay elements based on the user specified parameters.
Assay source, assay, assay component, and assay endpoint are registered via tcpl scripting into a collection of tables. The database structure takes the annotations and organizes them as attributes of the assay conductors, the assays (i.e., experiments), the assay components (i.e., raw readouts), or the assay endpoints (i.e., normalized component data) enabling aggregation and differentiation of the data generated through ToxCast and Tox21 (or other relevant partners). The annotations capture four types of information:
# Select annotation and subset by ids or name, ex.
assay <- tcplQuery("SELECT * FROM invitrodb.assay where aid=1;")
component <- tcplQuery("SELECT * FROM invitrodb.assay_component;")
component <- subset(component, acid %in% source$acid)
endpoint <- tcplQuery("SELECT * FROM invitrodb.assay_component_endpoint;")
endpoint <- endpoint[grepl("ATG", endpoint$assay_component_endpoint_name),]
# Or select all annotations by joining multiple tables
annotations <- tcplQuery("SELECT * FROM invitrodb.assay
INNER JOIN invitrodb.assay_source on assay.asid=assay_source.asid
INNER JOIN invitrodb.assay_component on assay_component.aid=assay.aid
INNER JOIN invitrodb.assay_component_endpoint on assay_component_endpoint.acid=assay_component.acid;")
The tcplLoadChem function returns all chemical information or can be filtered for user specified parameters, e.g. the chemical name (chnm) and chemical id (chid). The tcplLoadChemList function allows the user to subdivide the chemical IDs based on presence in different chemical lists. These chemical lists are curated by the US EPA in the Distributed Structure-Searchable Toxicity (DSSTox) database. Chemicals can belong to more than one chemical list, and will be listed as separate entries when loading chemical list information.
The tcplMthdList function returns methods available for processing at a specified level (i.e. step in the tcpl pipeline). The user defined function in the following code chunk retrieves and outputs all available methods for both the SC and MC data levels.
# Create a function to list all available methods function (SC & MC).
method_list <- function() {
# Single Concentration
## Level 1
sc1 <- tcplMthdList(1, 'sc')
sc1[, lvl := "sc1"]
setnames(sc1, c("sc1_mthd", "sc1_mthd_id"), c("mthd", "mthd_id"))
## Level 2
sc2 <- tcplMthdList(2, 'sc')
sc2[, lvl := "sc2"]
setnames(sc2, c("sc2_mthd", "sc2_mthd_id"), c("mthd", "mthd_id"))
# Multiple Concentration
## Level 2
mc2 <- tcplMthdList(2, 'mc')
mc2[, lvl := "mc2"]
setnames(mc2, c("mc2_mthd", "mc2_mthd_id"), c("mthd", "mthd_id"))
## Level 3
mc3 <- tcplMthdList(3, 'mc')
mc3[, lvl := "mc3"]
setnames(mc3, c("mc3_mthd", "mc3_mthd_id"), c("mthd", "mthd_id"))
## Level 4
mc4 <- tcplMthdList(4, 'mc')
mc4[, lvl := "mc4"]
setnames(mc4, c("mc4_mthd", "mc4_mthd_id"), c("mthd", "mthd_id"))
## Level 5
mc5 <- tcplMthdList(5, 'mc')
mc5[, lvl := "mc5"]
setnames(mc5, c("mc5_mthd", "mc5_mthd_id"), c("mthd", "mthd_id"))
# Compile the Output
mthd.list <- rbind(sc1, sc2, mc2, mc3, mc4, mc5)
mthd.list <- mthd.list[, c("lvl", "mthd_id", "mthd", "desc")]
# Return the Results
return(mthd.list)
}
# Run the 'method_list' functions and store output.
amthds <- method_list()
# Print the available methods list.
amthds
The tcplMthdLoad function returns the method assignments for specified id(s). Later sections provide more detailed examples for utilizing the tcplMthdLoad function for individuals ids.
The tcplQuery function allows a user to provide an SQL query to load data from the MySQL database into the R session. In the following chunk we provide an example, but any valid SQL query can replace the one provided. Please review Database Structure section to help construct these queries.
The tcplLoadData function can be used to load the data from the MySQL database into the R session. Further, the tcplPrepOtpt function can be used in combination with tcplLoadData to add useful chemical and assay annotation information, mapped to the retrieved data.
When loading data, the user must indicate the applicable fields and ids for the corresponding data level of interest. Loading level 0 (SC0 and MC0), MC1, and MC2 data the assay component id (acid) will always be used. As described in Data Processing sections, SC1 and MC3 processing levels perform data normalization where assay component ids (acid) are converted to assay endpoint ids (aeid). Thus, the SC1 and MC3 data tables contain both acid and (aeid) ID’s. Data can be loaded using either id as long as it is properly specified. Loading SC2, MC4, 5, and 6 always use (aeid). Selected id(s) are based on the primary key within each table containing data.
NOTE: There is no need to use earlier versions of tcpl to load data from an earlier version of invitrodb. In tcpl v3.2 onwards, tcplLoadData is fully backwards compatible with any version of invitrodb.
Prior to the pipeline processing provided in this package, all the data must go through pre-processing, i.e. raw data to database level 0 data. The standard level 0 format is identical for both testing paradigms, SC or MC, as described here. Users can inspect the level 0 data and calculate assay quality metrics prior to running the processing pipeline.
# Load Level 0 single concentration (SC0) data for a single acid to R.
sc0 <- tcplLoadData(lvl = 0, fld = "acid", val = 1, type = "sc") # data type - single concentration
# Alternatively, load data in and format with tcplPrepOtpt.
sc0 <- tcplPrepOtpt(tcplLoadData(lvl = 0, fld = "acid", val = 1, type = "sc"))
The goal of this section is to provide example quantitative metrics, such as z-prime and coefficient of variance, to evaluate assay performance relative to controls.
# Create a function to review assay quality metrics using indexed Level 0 data.
aq <- function(ac){
# obtain level 1 multiple concentration data for specified acids
dat <- tcplPrepOtpt(tcplLoadData(1L, "acid", aeids$acid, type = "mc"))
# keep only observations with good well quality (wllq = 1)
dat <- dat[wllq == 1]
# obtain summary values for data and remove missing data (i.e. NA's)
agg <- dat[ ,
list(
# median response values (rval) of neutral wells (wllt = n)
nmed = median(rval[wllt == "n"], na.rm = TRUE),
# median absolute deviation (mad) of neutral wells (wllt = n)
nmad = mad(rval[wllt == "n"], na.rm = TRUE),
# median response values of positive control wells (wllt = p)
pmed = median(rval[wllt == "p"], na.rm = TRUE),
# median absolute deviation of positive control wells (wllt = p)
pmad = mad(rval[wllt == "p"], na.rm = TRUE),
# median response values of negative control wells (wllt = m)
mmed = median(rval[wllt == "m"], na.rm = TRUE),
# median absolute deviation of negative control wells (wllt = m)
mmad = mad(rval[wllt == "m"], na.rm = TRUE)
),
# aggregate on assay component id, assay component name,
# and assay plate id
by = list(acid, acnm, apid)]
# Z prime factor: separation between positive and negative controls,
# indicative of likelihood of false positives or negatives.
# - Between 0.5 - 1 are excellent,
# - Between 0 and 0.5 may be acceptable,
# - Less than 0 not good
# obtain the z-prime factor for positive controls and neutral
agg[ , zprm.p := 1 - ((3 * (pmad + nmad)) / abs(pmed - nmed))]
# obtain the z-prime factor for negative controls and neutral
agg[ , zprm.m := 1 - ((3 * (mmad + nmad)) / abs(mmed - nmed))]
agg[ , ssmd.p := (pmed - nmed) / sqrt(pmad^2 + nmad^2)]
agg[ , ssmd.m := (mmed - nmed) / sqrt(mmad^2 + nmad^2)]
# Coefficient of Variation (cv) of neutral control
# - Ideally should be under 25%
agg[ , cv := nmad / nmed]
agg[ , sn.p := (pmed - nmed) / nmad]
agg[ , sn.m := (mmed - nmed) / nmad]
agg[ , sb.p := pmed / nmed]
agg[ , sb.m := mmed / nmed]
agg[zprm.p<0, zprm.p := 0]
agg[zprm.m<0, zprm.m := 0]
acqu <- agg[ , list( nmed = signif(median(nmed, na.rm = TRUE)),
nmad = signif(median(nmad, na.rm = TRUE)),
pmed = signif(median(pmed, na.rm = TRUE)),
pmad = signif(median(pmad, na.rm = TRUE)),
mmed = signif(median(mmed, na.rm = TRUE)),
mmad = signif(median(mmad, na.rm = TRUE)),
zprm.p = round(median(zprm.p, na.rm = TRUE), 2),
zprm.m = round(median(zprm.m, na.rm = TRUE), 2),
ssmd.p = round(median(ssmd.p, na.rm = TRUE), 0),
ssmd.m = round(median(ssmd.m, na.rm = TRUE), 0),
cv = round(median(cv, na.rm = TRUE), 2),
sn.p = round(median(sn.p, na.rm = TRUE), 2),
sn.m = round(median(sn.m, na.rm = TRUE), 2),
sb.p = round(median(sb.p, na.rm = TRUE), 2),
sb.m = round(median(sb.m, na.rm = TRUE), 2)
), by = list(acid, acnm)]
# Return the Results.
return(acqu)
} #per acid
# Run the 'aq' function & store the output.
assayq <- aq(ac)
The goal of SC processing is to identify potentially active compounds from a large screen at a single concentration. After processing, users can inspect SC activity hit calls and the applied methods.
# Load Level 2 single concentration data for a single aeid.
sc2 <- tcplPrepOtpt(tcplLoadData(lvl = 2, fld = "aeid", val = 3, type = "sc"))
# Alternatively, data for a set of aeids can be loaded with a vector of ids.
sc2 <- tcplPrepOtpt(tcplLoadData(lvl = 2, fld = "aeid", val = aeids$aeid, type = "sc"))
# Create a function to load methods for single concentration data processing
# steps for given aeids.
sc_methods <- function(aeids) {
# load the level 1 methods assigned for the single concentration aeid's
sc1_mthds <- tcplMthdLoad(lvl = 1, type = "sc", id = aeids$aeid)
# aggregate the method id's by aeid
sc1_mthds<- aggregate(mthd_id ~ aeid, sc1_mthds, toString)
# reset the names of the sc1_mthds object
setnames(sc1_mthds, "mthd_id", "sc1_mthd_id")
# load the level 2 methods assigned for the single concentration aeid's
sc2_mthds <- tcplMthdLoad(lvl = 2, type = "sc", id = aeids$aeid)
# aggregate the method id's by aeid
sc2_mthds<- aggregate(mthd_id ~ aeid, sc2_mthds, toString)
# reset the names of the sc2_mthds object
setnames(sc2_mthds, "mthd_id", "sc2_mthd_id")
# Compile the Output
methods <- merge( merge(aeids, sc1_mthds, by = "aeid", all = TRUE),
sc2_mthds, by = "aeid", all = TRUE )
# Return the Results
return(methods)
}
# Run the 'sc_methods' function and store the output.
smthds <- sc_methods(aeids)
The goal of MC processing is to estimate the hitcall, potency, efficacy, and other curve-fitting parameters for sample-assay endpoint pairs. After processing, users can inspect the activity hitcalls, model parameters, concentration-response plots, and the applied methods for the MC data.
# Create a function to load methods for MC data processing
# for select aeids.
mc_methods <- function(aeids) {
# acid
## load the methods assigned to level 2 for given acids
mc2_mthds <- tcplMthdLoad(2, aeids$acid)
## aggregate the assigned methods by acid
mc2_mthds<- aggregate(mthd_id ~ acid, mc2_mthds, toString)
## rename the columns for the 'mc2_mthds' object
setnames(mc2_mthds, "mthd_id", "mc2_mthd_id")
# aeid
## load the methods assigned to level 3 for given aeids
mc3_mthds <- tcplMthdLoad(3, aeids$aeid)
## aggregate the assigned methods by aeid
mc3_mthds<- aggregate(mthd_id ~ aeid, mc3_mthds, toString)
## rename the columns for the 'mc3_mthds' object
setnames(mc3_mthds, "mthd_id", "mc3_mthd_id")
## load the methods assigned to level 4 for given aeids
mc4_mthds <- tcplMthdLoad(4, aeids$aeid)
## aggregate the assigned methods by aeid
mc4_mthds<- aggregate(mthd_id ~ aeid, mc4_mthds, toString)
## rename the columns for 'mc4_mthds' object
setnames(mc4_mthds, "mthd_id", "mc4_mthd_id")
## load the methods assigned to level 5 for given aeids
mc5_mthds <- tcplMthdLoad(5, aeids$aeid)
## aggregate the assigned methods by aeid
mc5_mthds<- aggregate(mthd_id ~ aeid, mc5_mthds, toString)
## rename the columns for 'mc5_mthds' object
setnames(mc5_mthds, "mthd_id", "mc5_mthd_id")
# Compile the Results.
## merge the aeid information with the level 2 methods by acid
acid.methods <- merge(aeids, mc2_mthds, by.x = "acid", by.y = "acid")
## merge the level 3, 4, and 5 methods by aeid
mthd35 <- merge(
merge(mc3_mthds, mc4_mthds, by = "aeid", all = TRUE),
mc5_mthds, by = "aeid", all = TRUE
)
## merge all methods information by aeid
methods <- merge(acid.methods, mthd35, by.x = "aeid", by.y = "aeid")
# Print the Results.
print(methods)
# Return the Results.
return(methods)
}
# Run the 'methods' function and store the output.
mmthds <- mc_methods(aeids)
tcplPlot is tcpl’s single
flexible plotting function, allowing for interactive yet consistent
visualization of concentration-response curves via customizable
parameters. The standalone plotting utility is built with the R
libraries plotly and
ggplot2 to display the additional
curve-fitting models. The tcplPlot function
requires the selection of a field (fld
), and value
(val
) to load the necessary data and display the associated
plots. Level lvl
selection is no longer required and is
replaced by type
. Customization of output is possible by
specifying the following parameters:
Field | Description |
---|---|
type | ‘MC’ assumed as default. type = ‘mc’ plots available MC data fit by all models and highlights the winning model with activity hit call presented whereas type = ‘sc’ plots available SC data including response values, maximum median, and cutoff with activity hit call presented. |
fld | Required parameter for field to query on |
val | Required parameter for values to query on that must be listed for each corresponding ‘fld’ |
compare.val |
Parameter is used to generate comparison or dual plots. Using the same
field(s) as val , supply a list or vector of values for each
field to be plot one-to-one alongside val. Since tcplPlot matches ids
between val and compare.val, compare.val must be the same
length as val and the order val and
compare.val are given will be maintained in the output. The
default value is compare.val = NULL where the plots will be
individual; if it is set, tcplPlot will attempt to generate comparison
plots. For example, if fld = m4id and the user supplies three m4ids to
val , compare.val must also contain three
m4ids, where the first element of each val parameter are
plot together, the second elements together, etc.
|
output |
Parameter indicates how the plots will be presented. In addition to
outputs viewable with the R console , tcplPlot supports a
variety of publication-quality file type options, including raster
graphics (PNG , JPG , and TIFF ) to
retain color quality when printing to photograph and vector graphics
(SVG and PDF ) to retain image resolution when
scaled to large formats. For a more customizable option, an indivdiual
plot can be output in environment as a ggplot
|
verbose |
Parameter results in a plot that includes a table containing potency and
model performance metrics; verbose = FALSE is default and
the only option in console outputs. When verbose = TRUE the
model aic values are listed in descending order and generally the
winning model will be listed first.
|
multi |
Parameter allows for single or multiple plots per page.
multi = TRUE is the default option for PDF outputs, whereas
multi = FALSE is the only option for other outputs. If
using the parameter option multi = TRUE , the default number
of plots per page is set by the verbose parameter. The
default number of plots per page is either 6 plots per page
(verbose = FALSE ) or 4 plots per page
(verbose = TRUE ).
|
by | Parameter indicates how files should be divided, typically by aeid or spid |
fileprefix |
Parameter allows the user to set a custom filename prefix. The standard
filename is tcplPlot_sysDate().output (example: tcplPlot_2023_08_02.jpg)
or, if by parameter is set, tcplPlot_sysDate()_by.output
(example: tcplPlot_2023_08_02_aeid_80.pdf). When a
fileprefix is assigned the default tcplPlot prefix
is replaced with the new filename.
|
‘nrow’ and ‘ncol’ |
The ‘nrow’ parameter specifies the number of rows for the multiple plots
per page; this is 2 by default. The ncol parameter
specifies the number of columns for the multiple plots per page; this is
3 by default. If verbose = FALSE , ncol is 2.
nrow and ncol can customize the number of
plots included per page. Both nrow and ncol
must be greater than 0. While there is no hard coded upper limit to the
number of rows and columns, the underlying technology has a dimension
limitation of nrow = 9 and ncol = 7 .
|
dpi | Parameter specifies image print resolution for image file output types (PNG, JPG, TIFF, SVG); this is 600 by default. |
flag |
Parameter is used for toggling the output of Level 6 flags. The default
option is flag = FALSE . If type = 'sc , setting
flag = TRUE will result in warning, since invitrodb does
not store flags for single-concentration data.
|
yuniform |
Parameter is used for toggling automatic uniform y-axis scaling. The
default option is yuniform = FALSE . If set to
TRUE , tcplPlot will set each plot’s y-axis range to be the
minimum and maximum of response values and cutoffs across every
requested plot. For example, when plotting a percent activity endpoint
if the maximal response was 100% and minimal was -50%, while the cutoff
was 20%, the y-axis range for every plot will be set to be from -50% to
100%. This is most useful for across-plot interpretation.
|
yrange |
Parameter is used for toggling user-specified uniform y-axis scaling.
yrange is required to be an integer of length 2:
c(min,max). By default, c(NA,NA) will not set any uniform range. For
example, when plotting a percent activity endpoint, the user may wish to
set the range to c(-100,100) so every plot is contained to -100% and
100%.
|
dat | Parameter permits the user to supply plot-ready data rather than automatically loading it within tcplPlot. Use cases include plotting across multiple database connections or using tcplPlot to plot other tcplfit2-fit data. See the advanced comparison plotting section and ?tcplPlotLoadData for more information. |
The following examples demonstrate tcplPlot functionality through available the variety of customization options:
The following two examples produce plots of MC data for the selected aeids. A new pdf is generated for each endpoint. Filtering can be applied if only plots for a subset of samples (spids) are desired.
# Plot MC data for aeids 3157-3159 and outputs plots separate pdfs by aeid.
tcplPlot(type = "mc", # not required; "mc" is default
fld = "aeid", # field to query on
val = 3157:3159, # values should match their corresponding 'fld'
by = "aeid", # parameter to divide files
multi = TRUE, # multiple plots per page - output 4 per page
verbose = TRUE, # output all details if TRUE
output = "pdf") # output as pdf
# Loading required mc_vignette data for example below
data(mc_vignette, package = 'tcpl')
mc5 <- mc_vignette[["mc5"]]
# Plot MC data from the mc_vignette R data object for a single aeid 80 and
# spids "TP0001652B01", 01504209", "TP0001652D01", "TP0001652A01", and "1210314466"
tcplPlot(fld = c("aeid", "spid"), # field to query on
val = list(mc5$aeid, mc5$spid), # values must be listed for each corresponding 'fld'
by = "aeid", multi = TRUE, verbose = TRUE, flags = TRUE, yrange = c(-0.5, 1.5),
output = "pdf", fileprefix = "output_pdf")
This example illustrates an MC verbose plot for a single endpoint-sample of output type “jpg”.
# Plot a verbose plot of MC data for single aeid 80 and spid 01504209 and
# output as jpg.
tcplPlot(type = "mc", fld = c('aeid','spid'), val = list(80,'01504209'), multi = FALSE,
verbose = TRUE, flags = TRUE, output = "jpg", fileprefix = "output_jpg")
Due to the dynamic nature of m# ids, the first example code chunk does not include a corresponding plot. Here, the m4id value (482273) corresponds with the mc_vignette R data object. To run test this code, a valid m4id value must be supplied.
The second example includes a level 5 plot for one endpoint and one sample of output type “console”. Only 1 concentration series can be output in console at a time.
# Create MC plot for a single m4id.
tcplPlot(type = "mc", fld = "m4id", val = 482273,
multi = FALSE, verbose = FALSE, output = "console")
# Plot of MC data for single aeid (80) and spid (01504209)
# and output to console.
tcplPlot(type = "mc", fld = c('aeid','spid'), val = list(80, '01504209'),
multi = FALSE, verbose = FALSE, output = "console")
Single concentration plotting is enabled by setting
type = "sc"
. The example below plots all samples for one
endpoint.
# Plot SC data for aeid 704 and outputs plots separate pdfs by aeid.
tcplPlot(type = "sc", fld = "aeid", val = 704, multi = TRUE,
verbose = TRUE, output = "pdf", fileprefix = "sc_output")
The “compare”/dual-plot feature can be used by supplying a list or
vector of the same length to val
and
compare.val
. In the example below, plots are generated for
4 samples that appear in two different endpoints. These 4 samples are
compared one-to-one.
spids <- c("EPAPLT0108M13", "EPAPLT0108H01", "EPAPLT0108C17", "EPAPLT0106J20")
# default parameters used here: type = "mc"
tcplPlot(fld = c("spid", "aeid"), # field(s) to query on
val = list(spids, 3074), # values must be listed for each corresponding `fld`
compare.val = list(spids, 3076), # length must equal that of 'val'
output = "pdf", verbose = TRUE, multi = TRUE, flags = TRUE,
yuniform = TRUE, fileprefix = "plot_compare")
The “compare”/dual-plot feature can also be used with single concentration plotting. In the example below, plots are generated for 6 samples that appear in two different endpoints. These 4 samples are compared one-to-one.
spids <- c("MLS", "DMSO", "Tox21_400088", "Tox21_200265", "Tox21_200001", "Tox21_200266")
# Plot comparison across two different endpoints with same samples
tcplPlot(type = "sc", fld = c("spid","aeid"), val = list(spids, 3017),
compare.val = list(spids, 3018), # length must equal that of 'val'
output = "pdf", verbose = TRUE, multi = TRUE, fileprefix = "sc_plot_compare")
tcplPlot supports advanced comparison plotting across data connections. If working on a local invitrodb instance, additional dose-response data may be available, data reprocessed, methods adjusted, etc. Users may wish to compare data released in different versions, such as comparing CTX Bioactivity API data or versioned database (invitrodb v4.1 and later) to one’s local invitrodb database.
Using the utility function
tcplPlotLoadData while
connected to one data source, users can switch to a new data
configuration and pass along data via
tcplPlot’s dat
parameter and use
compare.val
for their new connection’s data.
# tcplConf() configured with some connection like invitrodb v4.1 or CTX APIs
plot_data <- tcplPlotLoadData(lvl = 5, fld = "aeid", val = 704, type = "mc", flags = TRUE)
# fill with different database connection information
tcplConf(user = "", pass = "", db = "invitrodb", drvr = "MySQL", host = "")
# Plot comparisons of aeid 704 from one database version to another and output to pdf
tcplPlot(dat = plot_data, # previously loaded data from tcplPlotLoadData()
fld = "aeid", val = 704, # include as copy of 'val' from tcplPlotLoadData()
compare.val = 704, # length must equal that of 'val', assumes aeid has not gained any samples
output = "pdf", verbose = TRUE, multi = TRUE, flags = TRUE, fileprefix = "plot_compare")
Below are some additional example code chunks for retrieving various bits of information from the database.
In this example, steps for extracting information about the compound Bisphenol A found within the database are illustrated. The user will define the chemical of interest, isolate all associated sample ids (spids), and then load all data for the given chemical.
# Provide the chemical name and assign to 'chnm'. Synonyms will not be matched, so other chemical identifiers may be more appropriate to query.
chnm <- 'Bisphenol A'
# Load the chemical data from the database.
chem <- tcplLoadChem(field = 'chnm', val = chnm)
# Load mc5 data from the database for the specified chemical.
BPA.mc5 <- tcplLoadData(lvl = 5, fld = 'spid', val = chem[, spid], type = 'mc')
In this example, plotting by endpoint for a sample subset, as opposed to plotting all samples tested within an endpoint, is illustrated. The user will load data for the select endpoints, isolate the samples of interest, and then plot by endpoint for the sample subset.
# Load MC% data summary values for select aeids
mc5 <- tcplPrepOtpt(tcplLoadData(lvl = 5, fld = 'aeid', val = tcplLoadAeid(fld = "asid", val = 25)$aeid, type = 'mc', add.fld = TRUE))
# Identify sample subset.
spid.mc5 <- mc5[spid %in% c("EPAPLT0018N08", "EPAPLT0023A16", "EPAPLT0020C11", "EPAPLT0018B13", "EPAPLT0018B14", "EPAPLT0018B15"),]
# Plot by endpoint for sample subset.
tcplPlot(fld = c("spid", "aeid"), val = list(spid.mc5$spid, spid.mc5$aeid), by = "aeid", multi = TRUE, verbose = TRUE, output = "pdf", fileprefix = "output/upitt")
To support different ToxCast data retrieval needs, there are a number of tcpl functions that can be used to query the API and return information to a local R environment. Abbreviations may be used to refer to processing steps or data. Single-concentration “SC” assay data is not currently available via API. “MC” describes multiple-concentration assay data. A particular data or processing level is indicated by appending the level id/number to the end of the SC or MC designation. For example, multiple concentration data from level 3 processing uses the abbreviation MC3.
The tcplLoadAsid, tcplLoadAid, tcplLoadAcid, and tcplLoadAeid functions load relevant ids and names for the respective assay elements based on the user specified parameters whether user has database or API connection.
tcplQueryAPI is a general querying function which is flexible enough to handle most kinds of queries users may have for the API. Unlike tcplQuery, tcplQueryAPI does not accept a MySQL query but instead has a few arguments which can be set to mimic a request to the various API endpoints. tcplQueryAPI is used mostly as a helper function to other tcplLoad functions, but is available to users for more specific and/or personalized requests.
# Request and load all assays+annotations for specified asid
data <- tcplQueryAPI(resource = "data", # resource to query from API, either 'data' or 'assay'
fld = "aeid", val = 891, # field and val to query on
return_fld = c("spid", "chnm", "hitcall")) # specify the return fields, leave NULL for all fields
The tcplLoadData function
can be used to load data from the CTX APIs into the R environment.
Only MC levels 3, 4, 5, 6, and ‘agg’ are currently available via
the API. To add chemical and assay annotation information, and
data from every level mapped to the retrieved data, set
add.fld = TRUE
(this is the default). The output with
add.fld = FALSE
will look as similar to the tcplLoadData
output by level from a database connection, though some less known
columns may not be available and the column order may differ.
When loading data, the user must indicate the applicable fields and
ids which are query-able via the CTX Bioactivity API. These such fields
are limited to “aeid”, “m4id”, “spid”, and “dtxsid”, and any other
fields will result in error. While supplying multiple ids (through
val
) are valid, multiple fields are not. Combinations of
fields are currently not supported with
tcplLoadData using an API connection.
tcplLoadData will return whatever data is
found, and list in output if any val(s) were not found or contain no
data to be returned. Examples of loading data are detailed in later
sections.
Assay source, assay, assay component, and assay endpoint are registered via tcpl scripting into a collection of database tables within invitrodb. For the API, these tables were joined to create a singular “assay” annotations view, which can be returned using tcplQueryAPI.
As described in greater detail within the Data Processing sections, a goal of MC processing is to derive the efficacy and potency estimates for for each modeled endpoint-sample dose response. API data is available for levels 3 through 6, herein users can inspect the MC data including efficacy and potency estimates, model parameters, raw concentration response values, cautionary flags, and applied methods.
Loading data is completed for a given endpoint (aeid), sample (spid), chemical (dtxsid), or endpoint-sample (m4id) by specifying “fld”. Set add.fld = FALSE is the option used to limit fields to those which are defaults for each level when loading from invitrodb directly. Leaving add.fld = TRUE (default) will return all available fields, i.e. all information from levels 3 through 6.
# Load MC5 data by aeid
mc5 <- tcplLoadData(lvl = 5, # data level
fld = "aeid", # fields to query on
val = 704, # values should match their corresponding 'fld'
type = "mc", # default. Note: sc data is not available on APIs yet
add.fld = FALSE) # restrict to just level 5 parameters
aeid | m4id | spid | bmad | resp_max | resp_min | max_mean | max_mean_conc | max_med | max_med_conc | logc_max | logc_min | nconc | npts | nrep | nmed_gtbl | m5id | modl | hitc | fitc | coff | model_type |
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
704 | 1842410 | LEGTV002G10 | 3.763468 | 22.88978 | -26.552798 | 22.88978 | 0.7781513 | 22.88978 | 0.7781513 | 1.69897 | -1.69897 | 8 | 8 | 1 | 4 | 5037056 | exp4 | 0.2411318 | 15 | 22.58081 | 2 |
704 | 1842408 | LEGTV002F02 | 3.763468 | 30.30548 | -15.782463 | 30.30548 | 1.6989700 | 30.30548 | 1.6989700 | 1.69897 | -1.69897 | 8 | 8 | 1 | 2 | 5037054 | poly1 | 0.6460865 | 15 | 22.58081 | 2 |
704 | 1842409 | LEGTV002F07 | 3.763468 | 11.82202 | -15.519376 | 11.82202 | -1.6989700 | 11.82202 | -1.6989700 | 1.69897 | -1.69897 | 8 | 8 | 1 | 1 | 5037055 | poly1 | 0.0000143 | 13 | 22.58081 | 2 |
704 | 1842411 | LEGTV002H08 | 3.763468 | 73.67532 | -5.395088 | 73.67532 | 0.7781513 | 73.67532 | 0.7781513 | 1.69897 | -1.69897 | 8 | 8 | 1 | 6 | 5037057 | exp4 | 0.9999995 | 42 | 22.58081 | 2 |
704 | 1842412 | LEGTV003B07 | 3.763468 | 77.95706 | -28.220282 | 77.95706 | 1.6989700 | 77.95706 | 1.6989700 | 1.69897 | -1.69897 | 8 | 8 | 1 | 5 | 5037058 | hill | 0.9999987 | 41 | 22.58081 | 2 |
704 | 1842413 | Strychnine nitrate | 3.763468 | 101.90303 | -13.656252 | 90.10575 | 0.0000000 | 92.78949 | 1.0000000 | 1.00000 | -2.00000 | 5 | 120 | 28 | 4 | 5037059 | hill | 1.0000000 | 41 | 22.58081 | 2 |
## Load MC5 data by spid
mc5 <- tcplLoadData(lvl=5, fld = "spid", val = "TP0000904H05", type = "mc", add.fld = FALSE)
## Load MC5 data by m4id
mc5 <- tcplLoadData(lvl=5, fld = "m4id", val = 1842443, type = "mc", add.fld = FALSE)
## Load MC data by dtxsid
mc5 <- tcplLoadData(lvl=5, fld = "dtxsid", val = "DTXSID30944145", type = "mc", add.fld = FALSE)
In addition to level 5 data, levels 3, 4, 6, and ‘agg’ are available to pull from the API.
## Load MC3. This returns m4id, spid, conc, aeid, logc, resp
mc3 <- tcplLoadData(lvl = 3, fld = "m4id", val = 1842443, type = "mc", add.fld = FALSE)
## Load MC4. This returns m4id, spid, bmad, resp_max, resp_min, max_mean, max_mean_conc,
# max_med, max_med_conc, logc_max, logc_min, nconc, npts, nrep, nmed_gtbl
mc4 <- tcplLoadData(lvl = 4, fld = "m4id", val= 1842443, type = "mc", add.fld = FALSE)
## Load MC6. This returns mc5 parameters plus flags
mc6 <- tcplLoadData(lvl = 6, fld = "m4id", val = 1842443, type = "mc", add.fld = FALSE)
## Load MC4 agg. This returns mc3 and mc4 parameters
agg <- tcplLoadData(lvl = "agg", fld = "m4id", val = 1842443, type = "mc", add.fld = FALSE)
## Load data with add.fld = TRUE to return all available processed data fields
all_fields <- tcplLoadData(lvl = 3, val = 1842443, type = "mc", add.fld = TRUE)
The tcplLoadAsid, tcplLoadAid, tcplLoadAcid, and tcplLoadAeid functions load relevant assay ids and names for the respective assay elements based on the user specified parameters.
tcplLoadAeid is used to load endpoint id (aeid) and endpoint name (aenm) as well any mapped annotations. Any annotations field can be used in the “fld” and “val” parameters to produce an annotations subset. Users may consider reviewing all annotations via tcplQueryAPI(resource = “assay”) if unsure which values to supply.
## acid aeid aenm
## <int> <int> <char>
## 1: 400 663 NVS_GPCR_rAdra1_NonSelective
Users may subset on as many fields as desired.
tcplLoadAeid joins the criteria with multiple
fld
and val
as an “AND” rather than “OR”,
meaning the subset returns rows where all are TRUE. val
has
the same length that fld
. To combine fields of different
types (i.e. numeric and string), or of different element lengths, ensure
all values are provided in appropriate length lists.
# subset all aeids by using multiple fields -- val must be same length in list form!
aeids <- tcplLoadAeid(fld = c("intended_target_type", "detection_technology_type"),
val = list("protein", c("Colorimetric", "Fluorescence"))) # list length == 2!
The above example subsets to endpoints where intended target type is “protein” and detection_technology_type is “colorimetric” or “fluorescence”.
Similar to tcplLoadAeid,
tcplLoadAcid loads assay component id (acid)
and assay component name (acnm) as well as any specified annotation
fields. Like tcplLoadAeid, output can be
subset with as many fld
and val
as
desired.
# load acid and acnm for given aeid
acid <- tcplLoadAcid(fld = "aeid", val = c(663,891))
# subset all acids by using multiple fields -- val must be same length in list form!
acids <- tcplLoadAcid(fld = c("organism", "tissue"),
val = list("rat", "liver"),
add.fld = c("aeid", "aid", "asid", "signal_direction"))
Similar to tcplLoadAeid,
tcplLoadAid loads assay id (aid) and assay
name (anm) as well as any specified annotation fields. Like
tcplLoadAeid, output can be subset with as
many fld
and val
as desired.
Similar to tcplLoadAeid,
tcplLoadAsid loads assay source id (asid) and
assay source name (asnm) as well as any specified annotation fields.
Like tcplLoadAsid, output can be subset with
as many fld
and val
as desired.
# Load asid and asnm for given aeid
asid <- tcplLoadAsid(fld = "aeid", val = 663)
# Subset all asids by using multiple fields -- val must be same length in list form!
asids <- tcplLoadAsid(fld = c("organism", "tissue"),
val = list("rat", "liver"),
add.fld = c("aeid", "acid", "asid", "signal_direction"))
To load the normalized data type, or response unit, use
tcplLoadUnit. This function does not require
fld
and val
parameters, but uses
aeid
as input. tcplLoadUnit is
typically used as an internal function for plotting.
tcplLoadConcUnit is used to load the
concentration unit for a specific spid
or multiple
spid
s. This is typically used as an internal function for
plotting.
# Load conc_unit for given spid
conc_unit <- tcplLoadConcUnit(spid = "TP0000904H05")
print(conc_unit)
## spid conc_unit
## <char> <char>
## 1: TP0000904H05 uM
tcplLoadChem is used to load the chemical
information for a specific spid
or multiple
spid
s. Notice tcplLoadChem uses field
instead
of fld
.
# Load chem_info for given spid
chem_info <- tcplLoadChem(field = "spid", val = "TP0000904H05")
print(chem_info)
## spid chid casn chnm dsstox_substance_id code
## <char> <int> <char> <char> <char> <char>
## 1: TP0000904H05 45232 51-24-1 Tiratricol DTXSID2045232 C51241
tcplPlot is tcpl’s single flexible plotting function, allowing for interactive and consistent visualization of concentration-response curves via customizable parameters. For more details on implementation, parameters and specific customization instructions, refer to the main Data Retrieval section. This section will instead focus on the limitations of plotting using the CTX APIs as a data connection.
As with loading data via tcplLoadData, the user must indicate the applicable fields and ids which are query-able via the CTX Bioactivity API. These such fields are limited to “aeid”, “m4id”, “spid”, and “dtxsid”, and any other fields input will result in error.
While supplying multiple ids (through val) is valid, supplying multiple fields is not. For example, if fld = “spid”, no aeid can be specified, meaning every matching spid will be plotted. If fld = “aeid”, every sample within the given endpoint(s) will be plotted. If fld = “m4id”, only one plot will output for every each m4id input. Combinations of fields are currently not supported with tcplPlot using API connections. Therefore, if looking for a specific aeid and spid combo, one should determine the corresponding m4id, like so:
# Load all matching spids and then subset using the aeid desired to find m4id
mc5 <- tcplLoadData(lvl = 5,
fld = "spid",
val = "TP0000904H05",
type = "mc",
add.fld = FALSE) # 8 rows of data
m4id <- mc5[aeid == 714]$m4id # subset to 1 aeid extract m4id
# Default parameters used here: fld = "m4id", type = "mc" (type can never be "sc" when connected to API)
tcplPlot(val = m4id, output = "jpg", verbose = TRUE, flags = TRUE)
The “compare”/dual-plot feature can be used by supplying a list of
m4ids to val
and compare.val
. These lists must
be the same length!
# Using the data pulled in the previous code chunk 'mc5'
m4id <- mc5$m4id # create m4id vector length == 8
# Default parameters used here: fld = "m4id", type = "mc" (type can never be "sc" when connected to API)
tcplPlot(val = m4id[1:4], compare.val = m4id[5:8], output = "pdf", verbose = TRUE, multi = TRUE,
flags = TRUE, yuniform = TRUE, fileprefix = "API_plot_compare")
Supply fld = "aeid"
to plot every curve available in the
API for the given endpoint(s).
# plot all curves across endpoint(s)
tcplPlot(fld = "aeid", val = 704, output = "pdf", verbose = TRUE,
multi = TRUE, yrange = c(-100,100), fileprefix = "API_plot_704")
tcplPlot has been updated for advanced
comparison plotting across data connections. If working on a local
invitrodb instance, additional dose-response data may be available, data
reprocessed, methods adjusted, etc. Users may wish to compare data
released in different versions, such as comparing API data or versioned
database (invitrodb v4.1 and later) to one’s local invitrodb database.
Using the utility function tcplPlotLoadData
while connected to the API, users can pass along data via
tcplPlot’s dat
parameter and use
compare.val
for their database data as described above.
This section will explore how one can compare in vivo Points of Departure (PODs) from the Toxicity Reference Database (ToxRefDB) with administered equivalent doses (AEDs) from ToxCast in vitro bioactivity data from invitrodb. The process can be adapted for any given chemical and target depending on available data in either database.
The following example will consider “Pentachlorophenol (PCP, DTXSID7021106)” and “liver toxicity”. This pesticide was selected at random to showcase workflow, but the process can be adapted for any given chemical and target depending on available data in either database.
First, export ToxRefDB batch download results for any chemical from the CompTox Chemicals Dashboard Batch Search or CTX Hazard APIs. Check out the ctxR R Client package for more guidance on interacting with the CTX APIs.
After loading all chemical-specific data for “Pentachlorophenol”, filter results to only include “liver”-related effects. Next identify the observed lowest effect (significantly different from control in source document i.e. treatment related=1) and lowest observed adverse (deemed adverse by study reviewer in source document i.e. critical_effect=1) effect levels at minimum dose_adjusted (mg/kg/day) value.
First, query the invitrodb database for all assay annotations, and filter results to consider only “liver” derived tissue-based endpoints.
For this subset of endpoints of targeted interest, pull assay results (mc5-mc6) for the chemical “Pentachlorophenol”
# Load the chemical data from the database
chnm <- 'Pentachlorophenol'
chem <- tcplLoadChem(field = 'chnm', val = chnm)
# Load mc5 data from the database for the specified chemical
mc5 <- tcplLoadData(lvl = 5, fld = 'spid', val = chem[, spid], type = 'mc')
#Join with level 6 flag information
mc6 <- tcplPrepOtpt(tcplLoadData(lvl = 6, fld = 'm4id', val = mc5$m4id, type = 'mc'))
setDT(mc6)
mc6_mthds <- mc6[ , .( mc6_mthd_id = paste(mc6_mthd_id, collapse=",")), by = m4id]
mc6_flags <- mc6[ , .( flag = paste(flag, collapse=";")), by = m4id]
mc5$mc6_flags <- mc6_mthds$mc6_mthd_id[match(mc5$m4id, mc6_mthds$m4id)]
mc5[, flag.length := ifelse(!is.na(mc6_flags),
count.fields(textConnection(mc6_flags), sep = ','), NA)]
# filter the potency and activity using coarse filters related to hitc, flags, fitc
mc5[hitc >= 0.9 & flag.length < 3, use.me := 1]
mc5[hitc >= 0.9 & is.na(flag.length), use.me := 1]
mc5[hitc >= 0.9 & flag.length >= 3, use.me := 0]
mc5[fitc %in% c(36, 45), use.me := 0]
mc5[hitc < 0.9, use.me := 0]
mc5[use.me == 0, ac50 := as.numeric(NA)]
mc5[use.me == 0, hitc := 0]
mc5[hitc == 0, ac50 := as.numeric(NA)]
mc5[hitc >= 0.9, ac50_uM := ifelse(!is.na(ac50), ac50, NA)]
#Filter to only liver endpoints
toxcast_mc5_liver <- mc5[aeid %in% toxcast_annotations_subset$aeid,]
Obtain a summary of the ToxCast AC50 values with the 5th and 50th percentiles, as well as the mean.
# Calculating summary statistics for ac50 values for httk processing to calculate AED
toxcast_mc5_liver_summary <- toxcast_mc5_liver[,list(
p5.ac50uM = quantile(ac50_uM, probs = c(0.05), na.rm = T),
p50.ac50uM = quantile(ac50_uM, probs = c(0.50), na.rm = T),
mean.ac50uM = mean(ac50_uM, na.rm = T))]
Administered Equivalent Doses can be accessed from Level 7, or calculated ad hoc using the High-throughput Toxicokinetics R package (httk). Potency estimates and model options can be adjusted based on use case. In this example, modeling assumptions when estimating the AEDs with httk were:
# Generate AEDs
toxcast_aed_liver_summary <- toxcast_mc5_liver_summary %>%
summarize(aed.p5ac50.hu.css.50 = calc_mc_oral_equiv(conc=p5.ac50uM,
dtxsid = 'DTXSID7021106', which.quantile = c(0.95),
species ='Human', restrictive.clearance = T,
output.units = 'mgpkgpday', model = '3compartmentss'),
aed.p50ac50.hu.css.50 = calc_mc_oral_equiv(conc = p50.ac50uM,
dtxsid = 'DTXSID7021106', which.quantile = c(0.95),
species = 'Human', restrictive.clearance = T,
output.units = 'mgpkgpday', model = '3compartmentss'),
aed.meanac50.hu.css.50 = calc_mc_oral_equiv(conc = mean.ac50uM,
dtxsid = 'DTXSID7021106', which.quantile = c(0.95),
species = 'Human', restrictive.clearance = T,
output.units = 'mgpkgpday', model = '3compartmentss'))
ToxRefDB LEL | ToxRefDB LOAEL | ToxCast AED at 5th percentile AC50 | ToxCast AED at 50th percentile/median AC50 | ToxCast AED at mean AC50 |
---|---|---|---|---|
1.5 | 1.5 | 2.273744 | 7.666872 | 16.09772 |
For the “Pentachlorophenol liver toxicity” example provided here, the POD estimated from ToxRefDB (POD-Traditional) is more protective compared to the lowest summary estimate from ToxCast (POD-NAM)
ToxCast data are predominantly based on mammalian models, but still may have value in ecological risk assessments. This section will explore how one may review ToxCast derived values in combination with curated values from Ecotoxicology (ECOTOX) Knowledgebase as well as cross-species applicability through Sequence Alignment to Predict Across Species Susceptibility (SeqAPASS) tool. The process can be adapted for any given chemical and target depending on available data in either database.
Repeat steps outlined above. This example will utilize a new chemical of interest: 17alpha-Ethinylestradiol (EE2, DTXSID5020576). Consider ToxCast and ToxRefDB to set POD-NAM and POD-Traditional, respectively.
# identify the lel and loaels from toxref chemical subset
toxref_chnm_POD<-toxref_chnm_EE2 %>%
summarise(lel=min(dose_adjusted[treatment_related==1]),
loael=min(dose_adjusted[critical_effect==1]))
# Load the chemical data from the database
chem <- tcplLoadChem(field = 'dsstox_substance_id',val = "DTXSID5020576")
# Load mc5 data from the database for the specified chemical
mc5 <- tcplLoadData(lvl = 5, # data level
fld = 'spid', # field to query on
val = chem[,spid], # value for each field (fld)
type = 'mc') # data type - MC
#Join with level 6 flag information
mc6 <- tcplPrepOtpt(tcplLoadData(lvl = 6, fld = 'm4id', val = mc5$m4id, type = 'mc'))
setDT(mc6)
mc6_mthds <- mc6[ , .( mc6_mthd_id = paste(mc6_mthd_id, collapse = ",")), by = m4id]
mc6_flags <- mc6[ , .( flag = paste(flag, collapse = ";")), by = m4id]
mc5$mc6_flags <- mc6_mthds$mc6_mthd_id[match(mc5$m4id, mc6_mthds$m4id)]
mc5[, flag.length := ifelse(!is.na(mc6_flags),
count.fields(textConnection(mc6_flags), sep =','), NA)]
# filter the potency and activity using coarse filters related to hitc, flags, fitc
mc5[hitc >= 0.9 & flag.length < 3, use.me := 1]
mc5[hitc >= 0.9 & is.na(flag.length), use.me := 1]
mc5[hitc >= 0.9 & flag.length >= 3, use.me := 0]
mc5[fitc %in% c(36,45), use.me := 0]
mc5[hitc < 0.9, use.me := 0]
mc5[use.me == 0, ac50 := as.numeric(NA)]
mc5[use.me == 0, hitc := 0]
mc5[hitc == 0, ac50 := as.numeric(NA)]
mc5[hitc >= 0.9,ac50_uM := ifelse(!is.na(ac50), ac50, NA)]
# Calculating summary statistics for ac50 values for httk processing to calculate AED
toxcast_mc5_EE2_summary <- mc5[,list(
p5.ac50uM = quantile(ac50_uM, probs = c(0.05), na.rm=T),
p50.ac50uM = quantile(ac50_uM, probs = c(0.50), na.rm=T),
mean.ac50uM = mean(ac50_uM, na.rm=T))]
# Generate AEDs
toxcast_aed_EE2_summary <- toxcast_mc5_EE2_summary %>%
summarize(aed.p5ac50.hu.css.50 = calc_mc_oral_equiv(conc = p5.ac50uM,
dtxsid = 'DTXSID5020576', which.quantile = c(0.95),
species ='Human', restrictive.clearance = T,
output.units = 'mgpkgpday', model = '3compartmentss'),
aed.p50ac50.hu.css.50 = calc_mc_oral_equiv(conc = p50.ac50uM,
dtxsid = 'DTXSID5020576', which.quantile = c(0.95),
species ='Human', restrictive.clearance = T,
output.units='mgpkgpday', model='3compartmentss'),
aed.meanac50.hu.css.50 = calc_mc_oral_equiv(conc=mean.ac50uM,
dtxsid = 'DTXSID5020576', which.quantile = c(0.95),
species = 'Human', restrictive.clearance = T,
output.units = 'mgpkgpday', model = '3compartmentss'),
aed.minac50.aeid807.hu.css.50 = calc_mc_oral_equiv(conc = 0.0002448276,
dtxsid = 'DTXSID5020576', which.quantile = c(0.95),
species ='Human', restrictive.clearance = T,
output.units = 'mgpkgpday', model = '3compartmentss'))
ToxRefDB LEL | ToxRefDB LOAEL | ToxCast AED at 5th percentile AC50 | ToxCast AED at 50th percentile/median AC50 | ToxCast AED at mean AC50 |
---|---|---|---|---|
0.00012 | 0.00021 | 2.26e-07 | 0.00661 | 0.01994 |
These summary POD-NAM values are calculated using all ToxCast endpoints. Additional inspection of individual endpoints and annotations may be warranted. Utilize the SeqAPASS column to filter to endpoints annotated with SeqAPASS protein targets, i.e. enter “NP_” into SeqAPASS search box.
The SeqAPASS tool has been developed to predict a species relative intrinsic susceptibility to chemicals with known molecular targets (e.g., pharmaceuticals, pesticides) as well as evaluate conservation of molecular targets in high-throughput screening assays (i.e., ToxCast), molecular initiating events (MIEs), and early key events in the adverse outcome pathway (AOP) framework as a means to extrapolate such knowledge across species. After copying the NCBI protein Accession numbers for ToxCast endpoints of interest, visit the SeqAPASS web interface to understand potential for cross-species comparison. Note that new users will need to request a free log-in to access this resource and should review the SeqAPASS User Guide for example workflows.
The ECOTOX widget in SeqAPASS gives the user the option to create a species and chemical filter that will link out to ECOTOX. The widget allows for rapid access of curated empirical toxicity data from the ECOTOXicology (ECOTOX) Knowledgebase that can be compared to sequence-based predictions of chemical susceptibility from SeqAPASS results.
All curated endpoint data may not be relevant for comparison and weight of relevance of these species-specific endpoints may also depend on SeqAPASS percent similarity. Additionally, ECOTOX records often cannot always be easily converted into mg/kg/day internal dose values for comparison. This is especially true for the non-dietary exposures, such as the aqueous exposures, where there are no chemical concentration measurements in the organisms across different species and life stages observed. These are considerations that can be further explored by reviewing the curated information and source documents.
An example of cross species extrapolation is described in Vliet et al, 2023. Overall, this study demonstrates a framework for utilizing bioinformatics and existing data to build weight of evidence for cross-species extrapolation and provides a technical basis for extrapolating data to prioritize hazard in non-mammalian vertebrate species.