From d83c54356c5b122544ceb731fb7279fd6cf5e62c Mon Sep 17 00:00:00 2001
From: schuemie Running multiple analyses at once using the
Martijn J.
Schuemie, Marc A. Suchard and Patrick Ryan
- 2023-04-17
+ 2023-12-21
Source: vignettes/MultipleAnalyses.Rmd
MultipleAnalyses.Rmd
Preparation for the example?createConnectionDetails for the specific
settings required for the various database management systems (DBMS).
For example, one might connect to a PostgreSQL database using this
code:
-connectionDetails <- createConnectionDetails(dbms = "postgresql",
- server = "localhost/ohdsi",
- user = "joe",
+connectionDetails <- createConnectionDetails(dbms = "postgresql",
+ server = "localhost/ohdsi",
+ user = "joe",
password = "supersecret")
cdmDatabaseSchema <- "my_cdm_data"
-resultsDatabaseSchema <- "my_results"
-options(sqlRenderTempEmulationSchema = NULL)
-outputFolder <- "./CohortMethodOutput"
The last three lines define the cdmDatabaseSchema
,
-resultSchema
, and outputFolder
variables.
-We’ll use these later to tell R where the data in CDM format live, where
-we want to write intermediate tables, and where the intermediate and
-output files should be stored in the local file system. Note that for
+cohortDatabaseSchema <- "my_results"
+cohortTable <- "my_cohorts"
+options(sqlRenderTempEmulationSchema = NULL)
+
The last few lines define the cdmDatabaseSchema
,
+cohortDatabaseSchema
, and cohortTable
+variables. We’ll use these later to tell R where the data in CDM format
+live, and where we want to write intermediate tables. Note that for
Microsoft SQL Server, databaseschemas need to specify both the database
and the schema, so for example
-cdmDatabaseSchema <- "my_cdm_data.dbo"
.
We also need to prepare our exposures and outcomes of interest. The -drug_era table in the OMOP Common Data Model already contains -prespecified cohorts of users at the ingredient level, so we will use -that for the exposures. For the outcomes, we want to restrict our -analysis only to those outcomes that are recorded in an inpatient -setting, so we will need to create a custom cohort table. For this -example, we want to include GI bleed (concept ID 192671) as well as a -set of 35 negative controls. Negative controls are defined as those -outcomes where there is no evidence that either the target drug -(celexocib) or comparator drug (diclofenac) causes the outcome.
-We create a text file called VignetteOutcomes.sql with the -following content:
-/***********************************
-File VignetteOutcomes.sql
-***********************************/
-DROP TABLE IF EXISTS @resultsDatabaseSchema.outcomes;
-
-SELECT ancestor_concept_id AS cohort_definition_id,
-AS cohort_start_date,
- condition_start_date AS cohort_end_date,
- condition_end_date AS subject_id
- condition_occurrence.person_id INTO @resultsDatabaseSchema.outcomes
-FROM @cdmDatabaseSchema.condition_occurrence
-INNER JOIN @cdmDatabaseSchema.visit_occurrence
-ON condition_occurrence.visit_occurrence_id = visit_occurrence.visit_occurrence_id
- INNER JOIN @cdmDatabaseSchema.concept_ancestor
-ON condition_concept_id = descendant_concept_id
- WHERE ancestor_concept_id IN (192671, 24609, 29735, 73754, 80004, 134718, 139099,
-141932, 192367, 193739, 194997, 197236, 199074, 255573, 257007, 313459, 314658,
-316084, 319843, 321596, 374366, 375292, 380094, 433753, 433811, 436665, 436676,
-436940, 437784, 438134, 440358, 440374, 443617, 443800, 4084966, 4288310)
-AND visit_occurrence.visit_concept_id IN (9201, 9203);
This is parameterized SQL which can be used by the
-SqlRender
package. We use parameterized SQL so we do not
-have to pre-specify the names of the CDM and result schemas. That way,
-if we want to run the SQL on a different schema, we only need to change
-the parameter values; we do not have to change the SQL code. By also
-making use of translation functionality in SqlRender
, we
-can make sure the SQL code can be run in many different
-environments.
-library(SqlRender)
-sql <- readSql("VignetteOutcomes.sql")
-sql <- render(sql,
- cdmDatabaseSchema = cdmDatabaseSchema,
- resultsDatabaseSchema = resultsDatabaseSchema)
-sql <- translate(sql, targetDialect = connectionDetails$dbms)
+cdmDatabaseSchema <- "my_cdm_data.dbo"
. For database
+platforms that do not support temp tables, such as Oracle, it is also
+necessary to provide a schema where the user has write access that can
+be used to emulate temp tables. PostgreSQL supports temp tables, so we
+can set options(sqlRenderTempEmulationSchema = NULL)
(or
+not set the sqlRenderTempEmulationSchema
at all.)
+We need to define the exposures and outcomes for our study. Here, we
+will define our exposures using the OHDSI Capr
package. We
+define two cohorts, one for celecoxib and one for diclofenac. For each
+cohort we require a prior diagnosis of ‘osteoarthritis of knee’, and 365
+days of continuous prior observation. we restrict to the first exposure
+per person:
+
-In this code, we first read the SQL from the file into memory. In the
-next line, we replace the two parameter names with the actual values. We
-then translate the SQL into the dialect appropriate for the DBMS we
-already specified in the connectionDetails
. Next, we
-connect to the server, and submit the rendered and translated SQL.
-
The first group of arguments define the target, comparator, and -outcome. Here we demonstrate how to create one set, and add that set to -a list:
+osteoArthritisOfKneeConceptId <- 4079750 +celecoxibConceptId <- 1118084 +diclofenacConceptId <- 1124300 +osteoArthritisOfKnee <- cs( + descendants(osteoArthritisOfKneeConceptId), + name = "Osteoarthritis of knee" +) +attrition = attrition( + "prior osteoarthritis of knee" = withAll( + atLeast(1, condition(osteoArthritisOfKnee), + duringInterval(eventStarts(-Inf, 0))) + ) +) +celecoxib <- cs( + descendants(celecoxibConceptId), + name = "Celecoxib" +) +diclofenac <- cs( + descendants(diclofenacConceptId), + name = "Diclofenac" +) +celecoxibCohort <- cohort( + entry = entry( + drug(celecoxib, firstOccurrence()), + observationWindow = continuousObservation(priorDays = 365) + ), + attrition = attrition, + exit = exit(endStrategy = drugExit(celecoxib, + persistenceWindow = 30, + surveillanceWindow = 0)) +) +diclofenacCohort <- cohort( + entry = entry( + drug(diclofenac, firstOccurrence()), + observationWindow = continuousObservation(priorDays = 365) + ), + attrition = attrition, + exit = exit(endStrategy = drugExit(diclofenac, + persistenceWindow = 30, + surveillanceWindow = 0)) +)We’ll pull the outcome definition from the OHDSI
+PhenotypeLibrary
:
+library(PhenotypeLibrary)
+outcomeCohorts <- getPlCohortDefinitionSet(77) # GI bleed
In addition to the outcome of interest, we also want to include a +large set of negative control outcomes:
-outcomeOfInterest <- createOutcome(outcomeId = 192671,
- outcomeOfInterest = TRUE)
-
-negativeControlIds <- c(192671, 29735, 140673, 197494,
+negativeControlIds <- c(29735, 140673, 197494,
198185, 198199, 200528, 257315,
314658, 317376, 321319, 380731,
432661, 432867, 433516, 433701,
@@ -280,26 +276,80 @@ Specifying hypotheses of interest 444252, 444429, 4131756, 4134120,
4134454, 4152280, 4165112, 4174262,
4182210, 4270490, 4286201, 4289933)
+negativeControlCohorts <- tibble(
+ cohortId = negativeControlIds,
+ cohortName = sprintf("Negative control %d", negativeControlIds),
+ outcomeConceptId = negativeControlIds
+)
We combine the exposure and outcome cohort definitions, and use
+CohortGenerator
to generate the cohorts:
+library(CirceR)
+# For exposures, create a cohort definition set table as required by CohortGenerator:
+exposureCohorts <- tibble(cohortId = c(1,2),
+ cohortName = c("Celecoxib", "Diclofenac"),
+ json = c(as.json(celecoxibCohort),
+ as.json(diclofenacCohort)))
+exposureCohorts$sql <- sapply(exposureCohorts$json,
+ buildCohortQuery,
+ options = createGenerateOptions())
+allCohorts <- bind_rows(outcomeCohorts,
+ exposureCohorts)
+library(CohortGenerator)
+cohortTableNames <- getCohortTableNames(cohortTable = cohortTable)
+createCohortTables(connectionDetails = connectionDetails,
+ cohortDatabaseSchema = cohortDatabaseSchema,
+ cohortTableNames = cohortTableNames)
+generateCohortSet(connectionDetails = connectionDetails,
+ cdmDatabaseSchema = cdmDatabaseSchema,
+ cohortDatabaseSchema = cohortDatabaseSchema,
+ cohortTableNames = cohortTableNames,
+ cohortDefinitionSet = allCohorts)
+generateNegativeControlOutcomeCohorts(
+ connectionDetails = connectionDetails,
+ cdmDatabaseSchema = cdmDatabaseSchema,
+ cohortDatabaseSchema = cohortDatabaseSchema,
+ cohortTable = cohortTable,
+ negativeControlOutcomeCohortSet = negativeControlCohorts
+)
If all went well, we now have a table with the cohorts of interest. +We can see how many entries per cohort:
+
+connection <- DatabaseConnector::connect(connectionDetails)
+sql <- "SELECT cohort_definition_id, COUNT(*) AS count FROM @cohortDatabaseSchema.@cohortTable GROUP BY cohort_definition_id"
+DatabaseConnector::renderTranslateQuerySql(
+ connection = connection,
+ sql = sql,
+ cohortDatabaseSchema = cohortDatabaseSchema,
+ cohortTable = cohortTable
+)
+DatabaseConnector::disconnect(connection)
The first group of arguments define the target, comparator, and +outcome. Here we demonstrate how to create one set, and add that set to +a list:
+
+outcomeOfInterest <- createOutcome(outcomeId = 77,
+ outcomeOfInterest = TRUE)
negativeControlOutcomes <- lapply(
negativeControlIds,
function(outcomeId) createOutcome(outcomeId = outcomeId,
outcomeOfInterest = FALSE,
trueEffectSize = 1)
)
-
-
-
tcos <- createTargetComparatorOutcomes(
- targetId = 1118084,
- comparatorId = 1124300,
+ targetId = 1,
+ comparatorId = 2,
outcomes = append(list(outcomeOfInterest),
negativeControlOutcomes)
)
-
targetComparatorOutcomesList <- list(tcos)
We first define the outcome of interest (GI-bleed, concept ID -192671), explicitly stating this is an outcome of interest +
We first define the outcome of interest (GI-bleed, cohort ID 77),
+explicitly stating this is an outcome of interest
(outcomeOfInterest = TRUE
), meaning we want the full set of
artifacts generated for this outcome. We then create a set of negative
control outcomes. Because we specify
@@ -308,8 +358,8 @@
A convenient way to save targetComparatorOutcomesList
to
file is by using the saveTargetComparatorOutcomesList
function, and we can load it again using the
@@ -325,11 +375,9 @@
createTrimByPsArgs()
function. These companion functions
can be used to create the arguments to be used during execution:
-
-nsaids <- 21603933
-
-covarSettings <- createDefaultCovariateSettings(
- excludedCovariateConceptIds = nsaids,
+
+covarSettings <- createDefaultCovariateSettings(
+ excludedCovariateConceptIds = c(1118084, 1124300),
addDescendantsToExclude = TRUE
)
@@ -356,7 +404,7 @@ Specifying analyses
+
cmAnalysis1 <- createCmAnalysis(
analysisId = 1,
description = "No matching, simple outcome model",
@@ -371,7 +419,7 @@ Specifying analyses
+
createPsArgs <- createCreatePsArgs() # Use default settings only
matchOnPsArgs <- createMatchOnPsArgs(maxRatio = 100)
@@ -427,7 +475,7 @@ Specifying analyses= fitOutcomeModelArgs3
)
## Note: Using propensity scores but not computing covariate balance
-
+
fitOutcomeModelArgs4 <- createFitOutcomeModelArgs(
useCovariates = TRUE,
modelType = "cox",
@@ -444,8 +492,12 @@ Specifying analyses= fitOutcomeModelArgs4
)
## Note: Using propensity scores but not computing covariate balance
-
-interactionCovariateIds <- c(8532001, 201826210, 21600960413) # Female, T2DM, concurent use of antithrombotic agents
+
+interactionCovariateIds <- c(
+ 8532001, # Female
+ 201826210, # T2DM
+ 21600960413 # concurrent use of antithrombotic agents
+)
fitOutcomeModelArgs5 <- createFitOutcomeModelArgs(
modelType = "cox",
@@ -464,7 +516,7 @@ Specifying analyses)
## Note: Using propensity scores but not computing covariate balance
These analyses can be combined in a list:
-
+
cmAnalysisList <- list(cmAnalysis1,
cmAnalysis2,
cmAnalysis3,
@@ -512,18 +564,17 @@ Executing multiple analysesrunCmAnalyses()
.
-
+
multiThreadingSettings <- createDefaultMultiThreadingSettings(parallel::detectCores())
result <- runCmAnalyses(
connectionDetails = connectionDetails,
cdmDatabaseSchema = cdmDatabaseSchema,
- exposureDatabaseSchema = cdmDatabaseSchema,
- exposureTable = "drug_era",
- outcomeDatabaseSchema = resultsDatabaseSchema,
- outcomeTable = "outcomes",
+ exposureDatabaseSchema = cohortDatabaseSchema,
+ exposureTable = cohortTable,
+ outcomeDatabaseSchema = cohortDatabaseSchema,
+ outcomeTable = cohortTable,
outputFolder = folder,
- cdmVersion = cdmVersion,
cmAnalysisList = cmAnalysisList,
targetComparatorOutcomesList = targetComparatorOutcomesList,
multiThreadingSettings = multiThreadingSettings
@@ -536,7 +587,7 @@ Executing multiple analysesparallel::detectCores()
function).
We call runCmAnalyses()
, providing the arguments for
connecting to the database, which schemas and tables to use, as well as
-the analyses and hypotheses of interest. The outputFolder
+the analyses and hypotheses of interest. The folder
specifies where the outcome models and intermediate files will be
written.
@@ -557,42 +608,28 @@ Retrieving the results
-
-psFile <- result$psFile[result$targetId == 1118084 &
- result$comparatorId == 1124300 &
- result$outcomeId == 192671 &
- result$analysisId == 5]
-ps <- readRDS(file.path(outputFolder, psFile))
+
+psFile <- result %>%
+ filter(targetId == 1,
+ comparatorId == 2,
+ outcomeId == 77,
+ analysisId == 5) %>%
+ pull(psFile)
+ps <- readRDS(file.path(folder, psFile))
plotPs(ps)
-
Note that some of the file names will appear several times in the
table. For example, analysis 3 and 5 only differ in terms of the outcome
model, and will share the same propensity score and stratification
files.
We can always retrieve the file reference table again using the
getFileReference()
function:
-
+
result <- getFileReference(folder)
We can get a summary of the results using
getResultsSummary()
:
-
-resultsSum <- getResultsSummary(outputFolder)
-head(resultsSum)
-## # A tibble: 6 x 27
-## analysisId targetId comparatorId outcomeId trueEffectSize targetSubjects
-## <int> <int> <int> <int> <dbl> <int>
-## 1 1 1118084 1124300 29735 1 86294
-## 2 1 1118084 1124300 140673 1 86718
-## 3 1 1118084 1124300 192671 NA 84447
-## 4 1 1118084 1124300 197494 1 86718
-## 5 1 1118084 1124300 198185 1 86718
-## 6 1 1118084 1124300 198199 1 86718
-## # i 21 more variables: comparatorSubjects <int>, targetDays <dbl>,
-## # comparatorDays <dbl>, targetOutcomes <dbl>, comparatorOutcomes <dbl>,
-## # rr <dbl>, ci95Lb <dbl>, ci95Ub <dbl>, p <dbl>, logRr <dbl>, seLogRr <dbl>,
-## # llr <dbl>, mdrr <dbl>, attritionFraction <dbl>, calibratedRr <dbl>,
-## # calibratedCi95Lb <dbl>, calibratedCi95Ub <dbl>, calibratedP <dbl>,
-## # calibratedLogRr <dbl>, calibratedSeLogRr <dbl>, ease <dbl>
+
+resultsSum <- getResultsSummary(folder)
+resultsSum
This tells us, per target-comparator-outcome-analysis combination,
the estimated relative risk and 95% confidence interval, as well as the
number of people in the treated and comparator group (after trimming and
@@ -608,63 +645,96 @@
Empirical calib
the yellow diamond represents our health outcome of interest: GI bleed.
An unbiased, well-calibrated analysis should have 95% of the negative
controls between the dashed lines (ie. 95% should have p > .05).
-
+
install.packages("EmpiricalCalibration")
library(EmpiricalCalibration)
# Analysis 1: No matching, simple outcome model
-negCons <- resultsSum[resultsSum$analysisId == 1 & resultsSum$outcomeId != 192671, ]
-hoi <- resultsSum[resultsSum$analysisId == 1 & resultsSum$outcomeId == 192671, ]
-null <- fitNull(negCons$logRr, negCons$seLogRr)
-plotCalibrationEffect(negCons$logRr, negCons$seLogRr, hoi$logRr, hoi$seLogRr, null)
-
-
+ncs <- resultsSum %>%
+ filter(analysisId == 1,
+ outcomeId != 77)
+hoi <- resultsSum %>%
+ filter(analysisId == 1,
+ outcomeId == 77)
+null <- fitNull(ncs$logRr, ncs$seLogRr)
+plotCalibrationEffect(logRrNegatives = ncs$logRr,
+ seLogRrNegatives = ncs$seLogRr,
+ logRrPositives = hoi$logRr,
+ seLogRrPositives = hoi$seLogRr, null)
+
# Analysis 2: Matching
-negCons <- resultsSum[resultsSum$analysisId == 2 & resultsSum$outcomeId != 192671, ]
-hoi <- resultsSum[resultsSum$analysisId == 2 & resultsSum$outcomeId == 192671, ]
-null <- fitNull(negCons$logRr, negCons$seLogRr)
-plotCalibrationEffect(negCons$logRr, negCons$seLogRr, hoi$logRr, hoi$seLogRr, null)
-
-
+ncs <- resultsSum %>%
+ filter(analysisId == 2,
+ outcomeId != 77)
+hoi <- resultsSum %>%
+ filter(analysisId == 2,
+ outcomeId == 77)
+null <- fitNull(ncs$logRr, ncs$seLogRr)
+plotCalibrationEffect(logRrNegatives = ncs$logRr,
+ seLogRrNegatives = ncs$seLogRr,
+ logRrPositives = hoi$logRr,
+ seLogRrPositives = hoi$seLogRr, null)
+
# Analysis 3: Stratification
-negCons <- resultsSum[resultsSum$analysisId == 3 & resultsSum$outcomeId != 192671, ]
-hoi <- resultsSum[resultsSum$analysisId == 3 & resultsSum$outcomeId == 192671, ]
-null <- fitNull(negCons$logRr, negCons$seLogRr)
-plotCalibrationEffect(negCons$logRr, negCons$seLogRr, hoi$logRr, hoi$seLogRr, null)
-
-
+ncs <- resultsSum %>%
+ filter(analysisId == 3,
+ outcomeId != 77)
+hoi <- resultsSum %>%
+ filter(analysisId == 3,
+ outcomeId == 77)
+null <- fitNull(ncs$logRr, ncs$seLogRr)
+plotCalibrationEffect(logRrNegatives = ncs$logRr,
+ seLogRrNegatives = ncs$seLogRr,
+ logRrPositives = hoi$logRr,
+ seLogRrPositives = hoi$seLogRr, null)
+
# Analysis 4: Inverse probability of treatment weighting
-negCons <- resultsSum[resultsSum$analysisId == 4 & resultsSum$outcomeId != 192671, ]
-hoi <- resultsSum[resultsSum$analysisId == 4 & resultsSum$outcomeId == 192671, ]
-null <- fitNull(negCons$logRr, negCons$seLogRr)
-plotCalibrationEffect(negCons$logRr, negCons$seLogRr, hoi$logRr, hoi$seLogRr, null)
-
-
+ncs <- resultsSum %>%
+ filter(analysisId == 4,
+ outcomeId != 77)
+hoi <- resultsSum %>%
+ filter(analysisId == 4,
+ outcomeId == 77)
+null <- fitNull(ncs$logRr, ncs$seLogRr)
+plotCalibrationEffect(logRrNegatives = ncs$logRr,
+ seLogRrNegatives = ncs$seLogRr,
+ logRrPositives = hoi$logRr,
+ seLogRrPositives = hoi$seLogRr, null)
+
# Analysis 5: Stratification plus full outcome model
-negCons <- resultsSum[resultsSum$analysisId == 5 & resultsSum$outcomeId != 192671, ]
-hoi <- resultsSum[resultsSum$analysisId == 5 & resultsSum$outcomeId == 192671, ]
-null <- fitNull(negCons$logRr, negCons$seLogRr)
-plotCalibrationEffect(negCons$logRr, negCons$seLogRr, hoi$logRr, hoi$seLogRr, null)
-
+ncs <- resultsSum %>%
+ filter(analysisId == 5,
+ outcomeId != 77)
+hoi <- resultsSum %>%
+ filter(analysisId == 5,
+ outcomeId == 77)
+null <- fitNull(ncs$logRr, ncs$seLogRr)
+plotCalibrationEffect(logRrNegatives = ncs$logRr,
+ seLogRrNegatives = ncs$seLogRr,
+ logRrPositives = hoi$logRr,
+ seLogRrPositives = hoi$seLogRr, null)
Analysis 6 explored interactions with certain variables. The
estimates for these interaction terms are stored in a separate results
summary. We can examine whether these estimates are also consistent with
-the null. In this example we consider the interaction with ‘gender =
-female’ (covariate ID 8532001):
-
-interactionResultsSum <- getInteractionResultsSummary(outputFolder)
-
+the null. In this example we consider the interaction with ‘concurrent
+use of antithrombotic agents’ (covariate ID 21600960413):
+
+interactionResultsSum <- getInteractionResultsSummary(folder)
# Analysis 6: Stratification plus interaction terms
-negCons <- interactionResultsSum[interactionResultsSum$analysisId == 6 & interactionResultsSum$outcomeId != 192671, ]
-hoi <- interactionResultsSum[interactionResultsSum$analysisId == 6 & interactionResultsSum$outcomeId == 192671, ]
-null <- fitNull(negCons$logRr, negCons$seLogRr)
-plotCalibrationEffect(logRrNegatives = negCons$logRr,
- seLogRrNegatives = negCons$seLogRr,
+ncs <- interactionResultsSum %>%
+ filter(analysisId == 6,
+ interactionCovariateId == 21600960413,
+ outcomeId != 77)
+hoi <- interactionResultsSum %>%
+ filter(analysisId == 6,
+ interactionCovariateId == 21600960413,
+ outcomeId == 77)
+null <- fitNull(ncs$logRr, ncs$seLogRr)
+plotCalibrationEffect(logRrNegatives = ncs$logRr,
+ seLogRrNegatives = ncs$seLogRr,
logRrPositives = hoi$logRr,
seLogRrPositives = hoi$seLogRr, null)
-## Warning: Removed 1 rows containing missing values (`geom_vline()`).
-
@@ -679,9 +749,9 @@ Exporting to CSV
-
+
exportToCsv(
- outputFolder,
+ folder,
exportFolder = file.path(folder, "export"),
databaseId = "My CDM",
minCellCount = 5,
@@ -694,24 +764,24 @@ Exporting to CSVminCellCount = 5
, the count will be reported to be -5,
which in the Shiny app will be displayed as ‘<5’.
Information on the data model used to generate the CSV files can be
-retrieved using getResultsDataModel()
:
-
-## # A tibble: 171 x 7
-## table_name column_name data_type is_required primary_key min_cell_count
-## <chr> <chr> <chr> <chr> <chr> <chr>
-## 1 cm_attrition sequence_n~ int Yes Yes No
-## 2 cm_attrition description varchar Yes No No
-## 3 cm_attrition subjects int Yes No Yes
-## 4 cm_attrition exposure_id int Yes Yes No
-## 5 cm_attrition target_id int Yes Yes No
-## 6 cm_attrition comparator~ int Yes Yes No
-## 7 cm_attrition analysis_id int Yes Yes No
-## 8 cm_attrition outcome_id int Yes Yes No
-## 9 cm_attrition database_id varchar Yes Yes No
-## 10 cm_follow_up_di~ target_id int Yes Yes No
-## # i 161 more rows
-## # i 1 more variable: description <chr>
+retrieved using getResultsDataModelSpecifications()
:
+
+## # A tibble: 188 × 8
+## tableName columnName dataType isRequired primaryKey minCellCount deprecated
+## <chr> <chr> <chr> <chr> <chr> <chr> <chr>
+## 1 cm_attriti… sequence_… int Yes Yes No No
+## 2 cm_attriti… descripti… varchar Yes No No No
+## 3 cm_attriti… subjects int Yes No Yes No
+## 4 cm_attriti… exposure_… int Yes Yes No No
+## 5 cm_attriti… target_id int Yes Yes No No
+## 6 cm_attriti… comparato… int Yes Yes No No
+## 7 cm_attriti… analysis_… int Yes Yes No No
+## 8 cm_attriti… outcome_id int Yes Yes No No
+## 9 cm_attriti… database_… varchar Yes Yes No No
+## 10 cm_follow_… target_id int Yes Yes No No
+## # ℹ 178 more rows
+## # ℹ 1 more variable: description <chr>
View results in a Shiny app
@@ -724,12 +794,12 @@ View results in a Shiny app
+
cohorts <- data.frame(
cohortId = c(
- 1118084,
- 1124300,
- 192671),
+ 1,
+ 2,
+ 77),
cohortName = c(
"Celecoxib",
"Diclofenac",
@@ -743,7 +813,7 @@ View results in a Shiny app= cohorts
)
Next we launch the Shiny app using:
-
+
launchResultsViewerUsingSqlite(
sqliteFileName = file.path(folder, "myResults.sqlite")
)
@@ -758,7 +828,7 @@ Acknowledgments
Considerable work has been dedicated to provide the
CohortMethod
package.
-
+
citation("CohortMethod")
##
## To cite package 'CohortMethod' in publications use:
@@ -771,16 +841,14 @@ Acknowledgments## A BibTeX entry for LaTeX users is
##
## @Manual{,
-## title = {CohortMethod: New-User Cohort Method with Large Scale Propensity and Outcome
-## Models},
+## title = {CohortMethod: New-User Cohort Method with Large Scale Propensity and Outcome Models},
## author = {Martijn Schuemie and Marc Suchard and Patrick Ryan},
## year = {2023},
-## note = {https://ohdsi.github.io/CohortMethod,
-## https://github.com/OHDSI/CohortMethod},
+## note = {https://ohdsi.github.io/CohortMethod, https://github.com/OHDSI/CohortMethod},
## }
Further, CohortMethod
makes extensive use of the
Cyclops
package.
-
+
citation("Cyclops")
##
## To cite Cyclops in publications use:
diff --git a/docs/articles/SingleStudies.html b/docs/articles/SingleStudies.html
index 7c2e6d54..c8a5061a 100644
--- a/docs/articles/SingleStudies.html
+++ b/docs/articles/SingleStudies.html
@@ -33,7 +33,7 @@
@@ -89,7 +89,7 @@ Single studies using the CohortMethod
Martijn J.
Schuemie, Marc A. Suchard and Patrick Ryan
- 2023-04-17
+ 2023-12-21
Source: vignettes/SingleStudies.Rmd
SingleStudies.Rmd
@@ -113,8 +113,8 @@ IntroductionInstallation instructions
Before installing the CohortMethod
package make sure you
-have Java available. Java can be downloaded from www.java.com. For Windows users, RTools
-is also necessary. RTools can be downloaded from CRAN.
+have Java available. For Windows users, RTools is also necessary. See these instructions
+for properly configuring your R environment.
The CohortMethod
package is currently maintained in a Github repository, and
has dependencies on other packages in Github. All of these packages can
be downloaded and installed from within R using the drat
@@ -138,154 +138,145 @@
Configuring the connection to
We need to tell R how to connect to the server where the data are.
CohortMethod
uses the DatabaseConnector
package, which provides the createConnectionDetails
-function. Type ?createConnectionDetails
for the specific
+function. Type ?createConnectionDetails
for the specific
settings required for the various database management systems (DBMS).
For example, one might connect to a PostgreSQL database using this
code:
-connectionDetails <- createConnectionDetails(dbms = "postgresql",
+connectionDetails <- createConnectionDetails(dbms = "postgresql",
server = "localhost/ohdsi",
user = "joe",
password = "supersecret")
cdmDatabaseSchema <- "my_cdm_data"
-resultsDatabaseSchema <- "my_results"
+cohortDatabaseSchema <- "my_results"
+cohortTable <- "my_cohorts"
options(sqlRenderTempEmulationSchema = NULL)
-The last two lines define the cdmDatabaseSchema
and
-resultSchema
variables. We’ll use these later to tell R
-where the data in CDM format live, and where we want to write
-intermediate tables. Note that for Microsoft SQL Server, databaseschemas
-need to specify both the database and the schema, so for example
-cdmDatabaseSchema <- "my_cdm_data.dbo"
.
+The last few lines define the cdmDatabaseSchema
,
+cohortDatabaseSchema
, and cohortTable
+variables. We’ll use these later to tell R where the data in CDM format
+live, and where we want to write intermediate tables. Note that for
+Microsoft SQL Server, databaseschemas need to specify both the database
+and the schema, so for example
+cdmDatabaseSchema <- "my_cdm_data.dbo"
. For database
+platforms that do not support temp tables, such as Oracle, it is also
+necessary to provide a schema where the user has write access that can
+be used to emulate temp tables. PostgreSQL supports temp tables, so we
+can set options(sqlRenderTempEmulationSchema = NULL)
(or
+not set the sqlRenderTempEmulationSchema
at all.)
Preparing the exposures and outcome(s)
-We need to define the exposures and outcomes for our study. One could
-use an external cohort definition tools, but in this example we do this
-by writing SQL statements against the OMOP CDM that populate a table of
-events in which we are interested. The resulting table should have the
-same structure as the cohort
table in the CDM. This means
-it should have the fields cohort_definition_id
,
-cohort_start_date
, cohort_end_date
,and
-subject_id
.
-For our example study, we have created a file called
-coxibVsNonselVsGiBleed.sql with the following contents:
-/***********************************
-File coxibVsNonselVsGiBleed.sql
-***********************************/
-
-DROP TABLE IF EXISTS @resultsDatabaseSchema.coxibVsNonselVsGiBleed;
-
-CREATE TABLE @resultsDatabaseSchema.coxibVsNonselVsGiBleed (
-INT,
- cohort_definition_id DATE,
- cohort_start_date DATE,
- cohort_end_date
- subject_id BIGINT
- );
-INSERT INTO @resultsDatabaseSchema.coxibVsNonselVsGiBleed (
-
- cohort_definition_id,
- cohort_start_date,
- cohort_end_date,
- subject_id
- )SELECT 1, -- Exposure
-
- drug_era_start_date,
- drug_era_end_date,
- person_idFROM @cdmDatabaseSchema.drug_era
-WHERE drug_concept_id = 1118084;-- celecoxib
-
-INSERT INTO @resultsDatabaseSchema.coxibVsNonselVsGiBleed (
-
- cohort_definition_id,
- cohort_start_date,
- cohort_end_date,
- subject_id
- )SELECT 2, -- Comparator
-
- drug_era_start_date,
- drug_era_end_date,
- person_idFROM @cdmDatabaseSchema.drug_era
-WHERE drug_concept_id = 1124300; --diclofenac
-
-INSERT INTO @resultsDatabaseSchema.coxibVsNonselVsGiBleed (
-
- cohort_definition_id,
- cohort_start_date,
- cohort_end_date,
- subject_id
- )SELECT 3, -- Outcome
-
- condition_start_date,
- condition_end_date,
- condition_occurrence.person_idFROM @cdmDatabaseSchema.condition_occurrence
-INNER JOIN @cdmDatabaseSchema.visit_occurrence
-ON condition_occurrence.visit_occurrence_id = visit_occurrence.visit_occurrence_id
- WHERE condition_concept_id IN (
-SELECT descendant_concept_id
- FROM @cdmDatabaseSchema.concept_ancestor
- WHERE ancestor_concept_id = 192671 -- GI - Gastrointestinal haemorrhage
-
- )AND visit_occurrence.visit_concept_id IN (9201, 9203);
-This is parameterized SQL which can be used by the
-SqlRender
package. We use parameterized SQL so we do not
-have to pre-specify the names of the CDM and result schemas. That way,
-if we want to run the SQL on a different schema, we only need to change
-the parameter values; we do not have to change the SQL code. By also
-making use of translation functionality in SqlRender
, we
-can make sure the SQL code can be run in many different
-environments.
-
-library(SqlRender)
-sql <- readSql("coxibVsNonselVsGiBleed.sql")
-sql <- render(sql,
- cdmDatabaseSchema = cdmDatabaseSchema,
- resultsDatabaseSchema = resultsDatabaseSchema)
-sql <- translate(sql, targetDialect = connectionDetails$dbms)
+We need to define the exposures and outcomes for our study. Here, we
+will define our exposures using the OHDSI Capr
package. We
+define two cohorts, one for celecoxib and one for diclofenac. For each
+cohort we require a prior diagnosis of ‘osteoarthritis of knee’, and 365
+days of continuous prior observation. we restrict to the first exposure
+per person:
+
-In this code, we first read the SQL from the file into memory. In the
-next line, we replace the two parameter names with the actual values. We
-then translate the SQL into the dialect appropriate for the DBMS we
-already specified in the connectionDetails
. Next, we
-connect to the server, and submit the rendered and translated SQL.
-If all went well, we now have a table with the events of interest. We
-can see how many events per type:
+osteoArthritisOfKneeConceptId <- 4079750
+celecoxibConceptId <- 1118084
+diclofenacConceptId <- 1124300
+osteoArthritisOfKnee <- cs(
+ descendants(osteoArthritisOfKneeConceptId),
+ name = "Osteoarthritis of knee"
+)
+attrition = attrition(
+ "prior osteoarthritis of knee" = withAll(
+ atLeast(1, condition(osteoArthritisOfKnee), duringInterval(eventStarts(-Inf, 0)))
+ )
+)
+celecoxib <- cs(
+ descendants(celecoxibConceptId),
+ name = "Celecoxib"
+)
+diclofenac <- cs(
+ descendants(diclofenacConceptId),
+ name = "Diclofenac"
+)
+celecoxibCohort <- cohort(
+ entry = entry(
+ drug(celecoxib, firstOccurrence()),
+ observationWindow = continuousObservation(priorDays = 365)
+ ),
+ attrition = attrition,
+ exit = exit(endStrategy = drugExit(celecoxib,
+ persistenceWindow = 30,
+ surveillanceWindow = 0))
+)
+diclofenacCohort <- cohort(
+ entry = entry(
+ drug(diclofenac, firstOccurrence()),
+ observationWindow = continuousObservation(priorDays = 365)
+ ),
+ attrition = attrition,
+ exit = exit(endStrategy = drugExit(diclofenac,
+ persistenceWindow = 30,
+ surveillanceWindow = 0))
+)
+We’ll pull the outcome definition from the OHDSI
+PhenotypeLibrary
:
+
+library(PhenotypeLibrary)
+outcomeCohorts <- getPlCohortDefinitionSet(77) # GI bleed
+We combine the exposure and outcome cohort definitions, and use
+CohortGenerator
to generate the cohorts:
-sql <- paste("SELECT cohort_definition_id, COUNT(*) AS count",
- "FROM @resultsDatabaseSchema.coxibVsNonselVsGiBleed",
- "GROUP BY cohort_definition_id")
-sql <- render(sql, resultsDatabaseSchema = resultsDatabaseSchema)
-sql <- translate(sql, targetDialect = connectionDetails$dbms)
+library(CirceR)
+# For exposures, create a cohort definition set table as required by CohortGenerator:
+exposureCohorts <- tibble(cohortId = c(1,2),
+ cohortName = c("Celecoxib", "Diclofenac"),
+ json = c(as.json(celecoxibCohort),
+ as.json(diclofenacCohort)))
+exposureCohorts$sql <- sapply(exposureCohorts$json,
+ buildCohortQuery,
+ options = createGenerateOptions())
+allCohorts <- bind_rows(outcomeCohorts,
+ exposureCohorts)
-querySql(connection, sql)
-## cohort_concept_id count
-## 1 1 50000
-## 2 2 50000
-## 3 3 15000
+library(CohortGenerator)
+cohortTableNames <- getCohortTableNames(cohortTable = cohortTable)
+createCohortTables(connectionDetails = connectionDetails,
+ cohortDatabaseSchema = cohortDatabaseSchema,
+ cohortTableNames = cohortTableNames)
+generateCohortSet(connectionDetails = connectionDetails,
+ cdmDatabaseSchema = cdmDatabaseSchema,
+ cohortDatabaseSchema = cohortDatabaseSchema,
+ cohortTableNames = cohortTableNames,
+ cohortDefinitionSet = allCohorts)
+If all went well, we now have a table with the cohorts of interest.
+We can see how many entries per cohort:
+
+connection <- DatabaseConnector::connect(connectionDetails)
+sql <- "SELECT cohort_definition_id, COUNT(*) AS count FROM @cohortDatabaseSchema.@cohortTable GROUP BY cohort_definition_id"
+DatabaseConnector::renderTranslateQuerySql(connection, sql, cohortDatabaseSchema = cohortDatabaseSchema, cohortTable = cohortTable)
+DatabaseConnector::disconnect(connection)
+## cohort_concept_id count
+## 1 1 109307
+## 2 2 176675
+## 3 77 733601
Extracting the data from the server
-Now we can tell CohortMethod
to define the cohorts based
-on our events, construct covariates, and extract all necessary data for
-our analysis.
+Now we can tell CohortMethod
to extract the cohorts,
+construct covariates, and extract all necessary data for our
+analysis.
Important: The target and comparator drug must not
be included in the covariates, including any descendant concepts. You
will need to manually add the drugs and descendants to the
excludedCovariateConceptIds
of the covariate settings. In
-this example code we exclude all NSAIDs from the covariates by pointing
-to the concept ID of the NSAID class and specifying
-addDescendantsToExclude = TRUE
.
-
-nsaids <- 21603933
-
-# Define which types of covariates must be constructed:
-covSettings <- createDefaultCovariateSettings(excludedCovariateConceptIds = nsaids,
- addDescendantsToExclude = TRUE)
+this example code we exclude the concepts for celecoxib and diclofenac
+and specify addDescendantsToExclude = TRUE
:
+
+# Define which types of covariates must be constructed:
+covSettings <- createDefaultCovariateSettings(
+ excludedCovariateConceptIds = c(diclofenacConceptId, celecoxibConceptId),
+ addDescendantsToExclude = TRUE
+)
#Load data:
cohortMethodData <- getDbCohortMethodData(
@@ -293,42 +284,14 @@ Extracting the data from the server
cdmDatabaseSchema = cdmDatabaseSchema,
targetId = 1,
comparatorId = 2,
- outcomeIds = 3,
- studyStartDate = "",
- studyEndDate = "",
- exposureDatabaseSchema = resultsDatabaseSchema,
- exposureTable = "coxibVsNonselVsGiBleed",
- outcomeDatabaseSchema = resultsDatabaseSchema,
- outcomeTable = "coxibVsNonselVsGiBleed",
- cdmVersion = cdmVersion,
- firstExposureOnly = TRUE,
- removeDuplicateSubjects = "remove all",
- restrictToCommonPeriod = FALSE,
- washoutPeriod = 180,
+ outcomeIds = 77,
+ exposureDatabaseSchema = cohortDatabaseSchema,
+ exposureTable = cohortTable,
+ outcomeDatabaseSchema = cohortDatabaseSchema,
+ outcomeTable = cohortTable,
covariateSettings = covSettings
)
cohortMethodData
-## # CohortMethodData object
-##
-## Target cohort ID: 1
-## Comparator cohort ID: 2
-## Outcome cohort ID(s): 3
-##
-## Inherits from CovariateData:
-## # CovariateData object
-##
-## All cohorts
-##
-## Inherits from Andromeda:
-## # Andromeda object
-## # Physical location: C:\Users\admin_mschuemi\AppData\Local\Temp\2\RtmpQZKV1W\file312c37fb2b73.sqlite
-##
-## Tables:
-## $analysisRef (analysisId, analysisName, domainId, startDay, endDay, isBinary, missingMeansZero)
-## $cohorts (rowId, personSeqId, personId, treatment, cohortStartDate, daysFromObsStart, daysToCohortEnd, daysToObsEnd)
-## $covariateRef (covariateId, covariateName, analysisId, conceptId)
-## $covariates (rowId, covariateId, covariateValue)
-## $outcomes (rowId, outcomeId, daysToEvent)
There are many parameters, but they are all documented in the
CohortMethod
manual. The
createDefaultCovariateSettings
function is described in the
@@ -349,22 +312,6 @@
Extracting the data from the server
view some more information of the data we extracted:
summary(cohortMethodData)
-## CohortMethodData object summary
-##
-## Target cohort ID: 1
-## Comparator cohort ID: 2
-## Outcome cohort ID(s): 3
-##
-## Target persons: 50000
-## Comparator persons: 50000
-##
-## Outcome counts:
-## Event count Person count
-## 3 36380 7447
-##
-## Covariates:
-## Number of covariates: 62004
-## Number of non-zero covariate values: 41097434
Saving the data to file
@@ -374,7 +321,7 @@ Saving the data to fileAndromeda, we cannot use R’s regular save function.
Instead, we’ll have to use the saveCohortMethodData()
function:
-
+
saveCohortMethodData(cohortMethodData, "coxibVsNonselVsGiBleed.zip")
We can use the loadCohortMethodData()
function to load
the data in a future session.
@@ -388,8 +335,8 @@ Defining new users
-When creating the cohorts in the database, for example when using a
-cohort definition tool.
+When creating the cohorts in the database, for example using
+Capr
.
When loading the cohorts using the
getDbCohortMethodData
function, you can use the
firstExposureOnly
, removeDuplicateSubjects
,
@@ -423,7 +370,7 @@ Defining the study population
+
studyPop <- createStudyPopulation(
cohortMethodData = cohortMethodData,
outcomeId = 3,
@@ -456,16 +403,8 @@ Defining the study population
+
getAttritionTable(studyPop)
-## # A tibble: 5 x 5
-## description targetPersons comparatorPersons targetExposures comparatorExposures
-## <chr> <dbl> <dbl> <dbl> <dbl>
-## 1 Original cohorts 856973 915830 1946114 1786318
-## 2 First exp. only & removed s ... 373874 541386 373874 541386
-## 3 Random sample 50000 50000 50000 50000
-## 4 No prior outcome 48700 48715 48700 48715
-## 5 Have at least 1 days at ris ... 48667 48688 48667 48688
One additional filtering step that is often used is matching or
trimming on propensity scores, as will be discussed next.
@@ -483,7 +422,7 @@ Fitting a propensity model
We can fit a propensity model using the covariates constructed by the
getDbcohortMethodData()
function:
-
+
ps <- createPs(cohortMethodData = cohortMethodData, population = studyPop)
The createPs()
function uses the Cyclops
package to fit a large-scale regularized logistic regression.
@@ -501,36 +440,31 @@ Propensity score diagnostics
+
computePsAuc(ps)
-## [1] 0.81
We can also plot the propensity score distribution, although we
prefer the preference score distribution:
-
+
plotPs(ps,
scale = "preference",
showCountsLabel = TRUE,
showAucLabel = TRUE,
showEquiposeLabel = TRUE)
-
It is also possible to inspect the propensity model itself by showing
the covariates that have non-zero coefficients:
-
+
getPsModel(ps, cohortMethodData)
-## # A tibble: 6 x 3
-## coefficient covariateId covariateName
-## <dbl> <dbl> <chr>
-## 1 -3.47 1150871413 ...gh 0 days relative to index: misoprostol
-## 2 -2.45 2016006 index year: 2016
-## 3 -2.43 2017006 index year: 2017
-## 4 -2.39 2018006 index year: 2018
-## 5 -2.37 2015006 index year: 2015
-## 6 -2.30 2014006 index year: 2014
One advantage of using the regularization when fitting the propensity
model is that most coefficients will shrink to zero and fall out of the
model. It is a good idea to inspect the remaining variables for anything
that should not be there, for example variations of the drugs of
interest that we forgot to exclude.
+Finally, we can inspect the percent of the population in equipoise,
+meaning they have a prefence score between 0.3 and 0.7:
+
+CohortMethod::computeEquipoise(ps)
+A low equipoise indicates there is little overlap between the target
+and comparator populations.
Using the propensity score
@@ -538,43 +472,30 @@ Using the propensity scoreWe can use the propensity scores to trim, stratify, match, or weigh
our population. For example, one could trim to equipoise, meaning only
subjects with a preference score between 0.25 and 0.75 are kept:
-
+
trimmedPop <- trimByPsToEquipoise(ps)
plotPs(trimmedPop, ps, scale = "preference")
-
Instead (or additionally), we could stratify the population based on
the propensity score:
-
+
stratifiedPop <- stratifyByPs(ps, numberOfStrata = 5)
plotPs(stratifiedPop, ps, scale = "preference")
-
We can also match subjects based on propensity scores. In this
example, we’re using one-to-one matching:
-
+
matchedPop <- matchOnPs(ps, caliper = 0.2, caliperScale = "standardized logit", maxRatio = 1)
plotPs(matchedPop, ps)
-
Note that for both stratification and matching it is possible to
specify additional matching criteria such as age and sex using the
stratifyByPsAndCovariates()
and
matchOnPsAndCovariates()
functions, respectively.
We can see the effect of trimming and/or matching on the population
using the getAttritionTable
function:
-
+
getAttritionTable(matchedPop)
-## # A tibble: 6 x 5
-## description targetPersons comparatorPersons targetExposures comparatorExposures
-## <chr> <dbl> <dbl> <dbl> <dbl>
-## 1 Original cohorts 856973 915830 1946114 1786318
-## 2 First exp. only & removed s ... 373874 541386 373874 541386
-## 3 Random sample 50000 50000 50000 50000
-## 4 No prior outcome 48700 48715 48700 48715
-## 5 Have at least 1 days at ris ... 48667 48688 48667 48688
-## 6 Matched on propensity score 22339 22339 22339 22339
Or, if we like, we can plot an attrition diagram:
-
+
drawAttritionDiagram(matchedPop)
-
Evaluating covariate balance
@@ -582,15 +503,12 @@ Evaluating covariate balance
+
balance <- computeCovariateBalance(matchedPop, cohortMethodData)
-
+
plotCovariateBalanceScatterPlot(balance, showCovariateCountLabel = TRUE, showMaxLabel = TRUE)
-## Warning: Removed 23590 rows containing missing values (`geom_point()`).
-
-
+
plotCovariateBalanceOfTopVariables(balance)
-
The ‘before matching’ population is the population as extracted by
the getDbCohortMethodData
function, so before any further
filtering steps.
@@ -603,113 +521,34 @@ Inspecting select populati
matching/stratification/trimming. This is usually the first table, and
so will be referred to as ‘table 1’. To generate this table, you can use
the createCmTable1
function:
-
+
createCmTable1(balance)
- Before matching After matching
- Target Comparator Target Comparator
- Characteristic % % Std. diff % % Std. diff
- Age group
- 25 - 29 0.0 0.0
- 30 - 34 0.0 0.0
- 40 - 44 0.0 0.0 0.00 0.0 0.0 -0.01
- 45 - 49 0.1 0.1 0.00 0.0 0.1 -0.01
- 50 - 54 0.2 0.2 0.00 0.2 0.2 0.00
- 55 - 59 0.5 0.5 -0.01 0.5 0.6 -0.01
- 60 - 64 0.8 1.2 -0.04 1.1 1.1 -0.01
- 65 - 69 25.8 29.0 -0.07 29.6 29.2 0.01
- 70 - 74 25.0 25.0 0.00 25.1 24.9 0.01
- 75 - 79 20.6 18.9 0.04 18.8 19.0 -0.01
- 80 - 84 15.2 13.2 0.06 13.4 13.7 -0.01
- 85 - 89 8.3 8.0 0.01 7.8 7.8 0.00
- 90 - 94 3.0 3.2 -0.01 2.9 2.8 0.01
- 95 - 99 0.6 0.7 -0.02 0.6 0.6 0.00
- 100 - 104 0.1 0.1 0.00 0.1 0.0 0.02
- Gender: female 59.5 61.8 -0.05 60.0 60.3 0.00
- Medical history: General
- Acute respiratory disease 18.4 22.2 -0.10 20.2 20.4 0.00
- Attention deficit hyperactivity disorder 0.1 0.2 -0.03 0.2 0.2 0.00
- Chronic liver disease 0.8 1.4 -0.05 1.0 1.1 -0.01
- Chronic obstructive lung disease 10.2 10.4 -0.01 9.4 9.6 -0.01
- Crohn's disease 0.3 0.4 -0.02 0.3 0.3 0.00
- Dementia 2.9 3.3 -0.03 2.8 2.9 -0.01
- Depressive disorder 6.2 9.3 -0.12 8.0 7.8 0.01
- Diabetes mellitus 18.6 25.3 -0.16 21.7 21.5 0.00
- Gastroesophageal reflux disease 10.0 15.1 -0.16 13.3 13.0 0.01
- Gastrointestinal hemorrhage 3.6 3.3 0.02 2.3 2.2 0.01
- Human immunodeficiency virus infection 0.0 0.1 -0.02 0.0 0.1 -0.01
- Hyperlipidemia 31.5 49.0 -0.36 42.7 41.8 0.02
- Hypertensive disorder 50.2 61.3 -0.22 56.8 56.5 0.01
- Lesion of liver 0.6 0.8 -0.02 0.6 0.6 0.00
- Obesity 3.8 7.3 -0.15 5.6 5.3 0.01
- Osteoarthritis 47.0 49.3 -0.05 50.6 50.0 0.01
- Pneumonia 4.7 4.7 0.00 4.0 4.2 -0.01
- Psoriasis 1.0 1.4 -0.03 1.3 1.2 0.01
- Renal impairment 4.7 10.5 -0.22 6.3 6.1 0.01
- Rheumatoid arthritis 2.4 3.0 -0.04 2.9 2.9 0.00
- Schizophrenia 0.1 0.1 -0.01 0.1 0.1 -0.01
- Ulcerative colitis 0.4 0.5 -0.02 0.4 0.4 0.00
- Urinary tract infectious disease 8.9 10.8 -0.06 9.7 9.6 0.00
- Viral hepatitis C 0.1 0.3 -0.04 0.2 0.2 0.00
- Medical history: Cardiovascular disease
- Atrial fibrillation 8.7 9.6 -0.03 8.2 8.4 -0.01
- Cerebrovascular disease 10.3 10.5 -0.01 9.5 9.6 0.00
- Coronary arteriosclerosis 17.2 18.4 -0.03 16.4 16.7 -0.01
- Heart disease 38.5 38.7 0.00 35.8 36.1 -0.01
- Heart failure 7.8 8.2 -0.01 6.3 6.4 0.00
- Ischemic heart disease 8.7 8.6 0.00 7.1 7.3 -0.01
- Peripheral vascular disease 7.3 10.1 -0.10 7.9 8.0 0.00
- Pulmonary embolism 0.7 0.8 -0.02 0.6 0.7 -0.01
- Venous thrombosis 1.9 2.5 -0.04 2.2 2.2 0.00
- Medical history: Neoplasms
- Malignant lymphoma 0.8 0.9 -0.01 0.8 0.8 0.00
- Malignant neoplasm of anorectum 0.5 0.3 0.03 0.4 0.3 0.00
- Malignant neoplastic disease 17.3 17.8 -0.01 17.4 17.5 0.00
- Malignant tumor of breast 3.0 3.2 -0.01 3.1 3.1 0.00
- Malignant tumor of colon 0.9 0.7 0.02 0.8 0.7 0.00
- Malignant tumor of lung 0.7 0.5 0.02 0.6 0.6 0.01
- Malignant tumor of urinary bladder 0.8 0.8 0.00 0.8 0.8 0.00
- Primary malignant neoplasm of prostate 3.9 3.4 0.02 3.6 3.6 0.00
- Medication use
- Agents acting on the renin-angiotensin system 44.2 49.6 -0.11 47.9 48.2 -0.01
- Antibacterials for systemic use 60.9 65.6 -0.10 61.8 62.4 -0.01
- Antidepressants 25.7 26.7 -0.02 26.2 26.5 -0.01
- Antiepileptics 14.7 17.8 -0.08 16.8 16.8 0.00
- Antiinflammatory and antirheumatic products 25.0 28.6 -0.08 28.9 28.8 0.00
- Antineoplastic agents 4.1 5.1 -0.05 4.8 4.7 0.00
- Antipsoriatics 0.7 1.1 -0.04 0.8 0.9 -0.01
- Antithrombotic agents 22.1 19.7 0.06 19.0 19.4 -0.01
- Beta blocking agents 34.4 38.1 -0.08 35.3 35.6 -0.01
- Calcium channel blockers 26.6 28.7 -0.05 26.9 27.2 -0.01
- Diuretics 42.4 41.2 0.02 40.5 41.3 -0.02
- Drugs for acid related disorders 37.1 38.9 -0.04 35.6 36.0 -0.01
- Drugs for obstructive airway diseases 38.2 44.7 -0.13 42.5 42.4 0.00
- Drugs used in diabetes 17.3 21.1 -0.10 18.5 18.6 0.00
- Immunosuppressants 3.0 4.9 -0.10 4.2 4.2 0.00
- Lipid modifying agents 49.1 56.0 -0.14 54.5 54.3 0.00
- Opioids 36.4 34.2 0.05 34.9 35.4 -0.01
- Psycholeptics 30.2 29.8 0.01 30.0 30.5 -0.01
- Psychostimulants, agents used for adhd and nootropics 1.5 1.8 -0.02 1.9 1.9 -0.01
-Inserting the population cohort in the database
+Generalizability
-
For various reasons it might be necessary to insert the study
-population back into the database, for example because we want to use an
-external cohort characterization tool. We can use the
-insertDbPopulation
function for this purpose:
-
-insertDbPopulation(
- population = matchedPop,
- cohortIds = c(101,100),
- connectionDetails = connectionDetails,
- cohortDatabaseSchema = resultsDatabaseSchema,
- cohortTable = "coxibVsNonselVsGiBleed",
- createTable = FALSE,
- cdmVersion = cdmVersion
-)
-This function will store the population in a table with the same
-structure as the cohort
table in the CDM, in this case in
-the same table where we had created our original cohorts.
+The goal of any propensity score adjustments is typically to make the
+target and comparator cohorts comparably, to allow proper causal
+inference. However, in doing so, we often need to modify our population,
+for example dropping subjects that have no counterpart in the other
+exposure cohort. The population we end up estimating an effect for may
+end up being very different from the population we started with. An
+important question is: how different? And it what ways? If the
+populations before and after adjustment are very different, our
+estimated effect may not generalize to the original population (if
+effect modification is present). The
+getGeneralizabilityTable()
function informs on these
+differences:
+
+getGeneralizabilityTable(balance)
+In this case, because we used PS matching, we are likely aiming to
+estimate the average treatment effect in the treated (ATT). For this
+reason, the getGeneralizabilityTable()
function
+automatically selected the target cohort as the basis for evaluating
+generalizability: it shows, for each covariate, the mean value before
+and PS adjustment in the target cohort. Also shown is the standardized
+difference of mean, and the table is reverse sorted by the absolute
+standard difference of mean (ASDM).
@@ -725,7 +564,7 @@ Follow-up and power
+
computeMdrr(
population = studyPop,
modelType = "cox",
@@ -733,13 +572,11 @@ Follow-up and power= 0.8,
twoSided = TRUE
)
-## targetPersons comparatorPersons targetExposures comparatorExposures targetDays comparatorDays totalOutcomes mdrr se
-## 1 48667 48688 48667 48688 7421404 3693928 554 1.26878 0.08497186
In this example we used the studyPop
object, so the
population before any matching or trimming. If we want to know the MDRR
after matching, we use the matchedPop
object we created
earlier instead:
-
+
computeMdrr(
population = matchedPop,
modelType = "cox",
@@ -747,8 +584,6 @@ Follow-up and power= 0.8,
twoSided = TRUE
)
-## targetPersons comparatorPersons targetExposures comparatorExposures targetDays comparatorDays totalOutcomes mdrr se
-## 1 22339 22339 22339 22339 3118703 1801997 226 1.451674 0.133038
Even thought the MDRR in the matched population is higher, meaning we
have less power, we should of course not be fooled: matching most likely
eliminates confounding, and is therefore preferred to not matching.
@@ -757,16 +592,12 @@ Follow-up and power
+
getFollowUpDistribution(population = matchedPop)
-## 100% 75% 50% 25% 0% Treatment
-## 1 2 60 60 126 4184 1
-## 2 2 45 60 67 2996 0
The output is telling us number of days of follow-up each quantile of
the study population has. We can also plot the distribution:
-
+
plotFollowUpDistribution(population = matchedPop)
-
Outcome models
@@ -779,52 +610,28 @@ Fitting a simple outcome model
+
outcomeModel <- fitOutcomeModel(population = studyPop,
modelType = "cox")
outcomeModel
-## Model type: cox
-## Stratified: FALSE
-## Use covariates: FALSE
-## Use inverse probability of treatment weighting: FALSE
-## Status: OK
-##
-## Estimate lower .95 upper .95 logRr seLogRr
-## treatment 1.25115 1.03524 1.51802 0.22406 0.0976
But of course we want to make use of the matching done on the
propensity score:
-
+
outcomeModel <- fitOutcomeModel(population = matchedPop,
modelType = "cox",
stratified = TRUE)
outcomeModel
-## Model type: cox
-## Stratified: TRUE
-## Use covariates: FALSE
-## Use inverse probability of treatment weighting: FALSE
-## Status: OK
-##
-## Estimate lower .95 upper .95 logRr seLogRr
-## treatment 0.9942982 0.7212731 1.3608869 -0.0057181 0.162
Note that we define the sub-population to be only those in the
matchedPop
object, which we created earlier by matching on
the propensity score. We also now use a stratified Cox model,
conditioning on the propensity score match sets.
Instead of matching or stratifying we can also perform Inverse
Probability of Treatment Weighting (IPTW):
-
+
outcomeModel <- fitOutcomeModel(population = ps,
modelType = "cox",
inversePtWeighting = TRUE)
outcomeModel
-## Model type: cox
-## Stratified: FALSE
-## Use covariates: FALSE
-## Use inverse probability of treatment weighting: TRUE
-## Status: OK
-##
-## Estimate lower .95 upper .95 logRr seLogRr
-## treatment 1.15095 0.83724 1.61142 0.14059 0.167
Adding interaction terms
@@ -833,7 +640,7 @@ Adding interaction terms
-
+
interactionCovariateIds <- c(8532001, 201826210, 21600960413)
# 8532001 = Female
# 201826210 = Type 2 Diabetes
@@ -843,28 +650,16 @@ Adding interaction terms stratified = TRUE,
interactionCovariateIds = interactionCovariateIds)
outcomeModel
-## Model type: cox
-## Stratified: TRUE
-## Use covariates: FALSE
-## Use inverse probability of treatment weighting: FALSE
-## Status: OK
-##
-## Estimate lower .95 upper .95 logRr seLogRr
-## treatment 1.24991 0.87272 1.80961 0.22307 0.1860
-## treatment * condition_era group during day -365 through 0 days relative to index: Type 2 diabetes mellitus 1.05089 0.68593 1.62105 0.04964 0.2194
-## treatment * drug_era group during day 0 through 0 days relative to index: ANTITHROMBOTIC AGENTS 0.63846 0.42639 0.96305 -0.44870 0.2078
-## treatment * gender = FEMALE 0.78988 0.54227 1.14572 -0.23587 0.1908
Note that you can use the grepCovariateNames
to find
covariate IDs.
It is prudent to verify that covariate balance has also been achieved
in the subgroups of interest. For example, we can check the covariate
balance in the subpopulation of females:
-
+
balanceFemale <- computeCovariateBalance(population = matchedPop,
cohortMethodData = cohortMethodData,
subgroupCovariateId = 8532001)
plotCovariateBalanceScatterPlot(balanceFemale)
-
Adding covariates to the outcome model
@@ -875,48 +670,33 @@ Adding covariates to the outcome
remove bias. For this we use the regularized Cox regression in the
Cyclops
package. (Note that the treatment variable is
automatically excluded from regularization.)
-
+
outcomeModel <- fitOutcomeModel(population = matchedPop,
cohortMethodData = cohortMethodData,
modelType = "cox",
stratified = TRUE,
useCovariates = TRUE)
outcomeModel
-## Model type: cox
-## Stratified: TRUE
-## Use covariates: TRUE
-## Use inverse probability of treatment weighting: FALSE
-## Status: OK
-## Prior variance: 0.0374185437128226
-##
-## Estimate lower .95 upper .95 logRr seLogRr
-## treatment 0.9985318 0.6870719 1.4438456 -0.0014693 0.1894
Inspecting the outcome model
We can inspect more details of the outcome model:
-
+
-## 900000010805
-## 0.9985318
-
+
-## [1] 0.6870719 1.4438456
We can also see the covariates that ended up in the outcome
model:
-
+
getOutcomeModel(outcomeModel, cohortMethodData)
-## coefficient id name
-## 1 -0.001469294 9e+11 Treatment
Kaplan-Meier plot
We can create the Kaplan-Meier plot:
-
+
plotKaplanMeier(matchedPop, includeZero = FALSE)
-
Note that the Kaplan-Meier plot will automatically adjust for any
stratification, matching, or trimming that may have been applied.
@@ -927,9 +707,9 @@ Time-to-event plot
+
plotTimeToEvent(cohortMethodData = cohortMethodData,
- outcomeId = 3,
+ outcomeId = 77,
firstExposureOnly = FALSE,
washoutPeriod = 0,
removeDuplicateSubjects = "keep all",
@@ -938,7 +718,6 @@ Time-to-event plot= "cohort start",
riskWindowEnd = 30,
endAnchor = "cohort end")
-
Note that this plot does not show any adjustment for the propensity
score.
@@ -948,7 +727,7 @@ Acknowledgments
Considerable work has been dedicated to provide the
CohortMethod
package.
-
+
citation("CohortMethod")
##
## To cite package 'CohortMethod' in publications use:
@@ -959,16 +738,14 @@ Acknowledgments## A BibTeX entry for LaTeX users is
##
## @Manual{,
-## title = {CohortMethod: New-User Cohort Method with Large Scale Propensity and Outcome
-## Models},
+## title = {CohortMethod: New-User Cohort Method with Large Scale Propensity and Outcome Models},
## author = {Martijn Schuemie and Marc Suchard and Patrick Ryan},
## year = {2023},
-## note = {https://ohdsi.github.io/CohortMethod,
-## https://github.com/OHDSI/CohortMethod},
+## note = {https://ohdsi.github.io/CohortMethod, https://github.com/OHDSI/CohortMethod},
## }
Further, CohortMethod
makes extensive use of the
Cyclops
package.
-
+
citation("Cyclops")
##
## To cite Cyclops in publications use:
diff --git a/docs/articles/index.html b/docs/articles/index.html
index f2de5a33..f7083f19 100644
--- a/docs/articles/index.html
+++ b/docs/articles/index.html
@@ -17,7 +17,7 @@
diff --git a/docs/authors.html b/docs/authors.html
index cbad701f..c6f6ab79 100644
--- a/docs/authors.html
+++ b/docs/authors.html
@@ -17,7 +17,7 @@
diff --git a/docs/index.html b/docs/index.html
index 39433ada..2567e06a 100644
--- a/docs/index.html
+++ b/docs/index.html
@@ -44,7 +44,7 @@
diff --git a/docs/news/index.html b/docs/news/index.html
index af639beb..b4dca012 100644
--- a/docs/news/index.html
+++ b/docs/news/index.html
@@ -17,7 +17,7 @@
@@ -63,6 +63,29 @@ Changelog
Source: NEWS.md
+
+CohortMethod 5.2.0
+Changes:
+The computeCovariateBalance()
function now also computes standardized difference of mean comparing cohorts before and after PS adjustment, which can inform on generalizability.
+Added the getGeneralizabilityTable()
function.
+Improved computation of overall standard deviation when computing covariate balance (actually computing the SD instead of taking the mean of the target and comparator). Should produce more accurate balance estimations.
+Generated population objects now keep track of likely target estimator (e.g. ‘ATT’, or ‘ATE’). This informs selection of base population when calling getGeneralizabilityTable()
.
+Deprecated the attritionFractionThreshold
argument of the createCmDiagnosticThresholds
function, and instead added the generalizabilitySdmThreshold
argument.
+-
+
The results schema specifications of the exportToCsv()
function has changed:
+- Removed the
attrition_fraction
and attrition_diagnostic
fields from the cm_diagnostics_summary
table.
+- Added the
target_estimator
field to the cm_result
add cm_interaction_result
tables.
+- Added the
generalizability_max_sdm
and generalizabiltiy_diagnostic
fields to the cm_diagnostics_summary
table.
+- Added the
mean_before
, mean_after
, target_std_diff
, comparator_std_diff
, and target_comparator_std_diff
fields to both the cm_covariate_balance
and cm_shared_covariate_balance
tables.
+
+Improve speed of covariate balance computation.
+Adding one-sided (calibrated) p-values to results summary and results model.
+Adding unblind_for_evidence_synthesis
field to cm_diagnostics_summary
table.
+The cm_diagnostics_summary
table now also contains negative controls.
+
Bugfixes:
+Fixing runCmAnalyses()
when using refitPsForEveryOutcome = TRUE
.
+Handling edge case when exporting preference distribution and the target or comparator only has 1 subject.
+
CohortMethod 5.1.0
Changes:
@@ -103,7 +126,7 @@ CohortMethod
Added empirical calibration to the getResultsSummary()
function. Controls can be identified by the trueEffectSize
argument in the createOutcome()
function.
Dropping arguments like createPs
and fitOutcomeModel
from the createCmAnalysis()
function. Instead, not providing createPsArgs
or fitOutcomeModelArgs
is assumed to mean skipping propensity score creation or outcome model fitting, respectively.
-Added the exportToCsv()
function for exporting study results to CSV files that do not contain patient-level information and can therefore be shared between sites. The getResultsDataModel()
function returns the data model for these CSV files.
+Added the exportToCsv()
function for exporting study results to CSV files that do not contain patient-level information and can therefore be shared between sites. The getResultsDataModel()
function returns the data model for these CSV files.
Added the uploadExportedResults()
and insertExportedResultsInSqlite()
functions for uploading the results from the CSV files in a database. The launchResultsViewer()
and launchResultsViewerUsingSqlite()
functions were added for launching a Shiny app to view the results in the (SQLite) database.
Bug fixes:
- Fixed error when using integer
maxWeight
when performng IPTW.
diff --git a/docs/pkgdown.yml b/docs/pkgdown.yml
index eac4527f..26d02b32 100644
--- a/docs/pkgdown.yml
+++ b/docs/pkgdown.yml
@@ -4,5 +4,5 @@ pkgdown_sha: ~
articles:
MultipleAnalyses: MultipleAnalyses.html
SingleStudies: SingleStudies.html
-last_built: 2023-09-04T12:25Z
+last_built: 2023-12-21T10:18Z
diff --git a/docs/pull_request_template.html b/docs/pull_request_template.html
index 4483fdd5..72060d2f 100644
--- a/docs/pull_request_template.html
+++ b/docs/pull_request_template.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/CohortMethod-package.html b/docs/reference/CohortMethod-package.html
index c5e40f0c..0f5c06dd 100644
--- a/docs/reference/CohortMethod-package.html
+++ b/docs/reference/CohortMethod-package.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/CohortMethodData-class.html b/docs/reference/CohortMethodData-class.html
index 1ff6e539..321c8d0a 100644
--- a/docs/reference/CohortMethodData-class.html
+++ b/docs/reference/CohortMethodData-class.html
@@ -21,7 +21,7 @@
diff --git a/docs/reference/adjustedKm.html b/docs/reference/adjustedKm.html
index 99d107ee..13a28688 100644
--- a/docs/reference/adjustedKm.html
+++ b/docs/reference/adjustedKm.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/checkCmInstallation.html b/docs/reference/checkCmInstallation.html
index b3abbdde..bb354370 100644
--- a/docs/reference/checkCmInstallation.html
+++ b/docs/reference/checkCmInstallation.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/cohortMethodDataSimulationProfile.html b/docs/reference/cohortMethodDataSimulationProfile.html
index 53cbae00..044a8043 100644
--- a/docs/reference/cohortMethodDataSimulationProfile.html
+++ b/docs/reference/cohortMethodDataSimulationProfile.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/computeCovariateBalance.html b/docs/reference/computeCovariateBalance.html
index c33c618e..b0d9f47e 100644
--- a/docs/reference/computeCovariateBalance.html
+++ b/docs/reference/computeCovariateBalance.html
@@ -1,7 +1,8 @@
-Compute covariate balance before and after matching and trimming — computeCovariateBalance • CohortMethod Compute covariate balance before and after PS adjustment — computeCovariateBalance • CohortMethod
@@ -19,7 +20,7 @@
@@ -61,15 +62,16 @@
- Compute covariate balance before and after matching and trimming
+ Compute covariate balance before and after PS adjustment
Source: R/Balance.R
computeCovariateBalance.Rd
For every covariate, prevalence in treatment and comparator groups before and after
-matching/trimming are computed. When variable ratio matching was used the balance score will be
-corrected according the method described in Austin et al (2008).
+matching/trimming/weighting are computed. When variable ratio matching was used
+the balance score will be corrected according the method described in Austin et
+al (2008).
@@ -85,8 +87,7 @@ Compute covariate balance before and after matching and trimming
Arguments
- population
-A data frame containing the people that are remaining after matching
-and/or trimming.
+A data frame containing the people that are remaining after PS adjustment.
- cohortMethodData
@@ -117,7 +118,43 @@ Arguments
Value
-Returns a tibble describing the covariate balance before and after matching/trimming.
+Returns a tibble describing the covariate balance before and after PS adjustment,
+with one row per covariate, with the same data as the covariateRef
table in the CohortMethodData
object,
+and the following additional columns:
beforeMatchingMeanTarget: The (weighted) mean value in the target before PS adjustment.
+beforeMatchingMeanComparator: The (weighted) mean value in the comparator before PS adjustment.
+beforeMatchingSumTarget: The (weighted) sum value in the target before PS adjustment.
+beforeMatchingSumComparator: The (weighted) sum value in the comparator before PS adjustment.
+beforeMatchingSdTarget: The standard deviation of the value in the target before PS adjustment.
+beforeMatchingSdComparator: The standard deviation of the value in the comparator before PS adjustment.
+beforeMatchingMean: The mean of the value across target and comparator before PS adjustment.
+beforeMatchingSd: The standard deviation of the value across target and comparator before PS adjustment.
+afterMatchingMeanTarget: The (weighted) mean value in the target after PS adjustment.
+afterMatchingMeanComparator: The (weighted) mean value in the comparator after PS adjustment.
+afterMatchingSumTarget: The (weighted) sum value in the target after PS adjustment.
+afterMatchingSumComparator: The (weighted) sum value in the comparator after PS adjustment.
+afterMatchingSdTarget: The standard deviation of the value in the target after PS adjustment.
+afterMatchingSdComparator: The standard deviation of the value in the comparator after PS adjustment.
+afterMatchingMean: The mean of the value across target and comparator after PS adjustment.
+afterMatchingSd: The standard deviation of the value across target and comparator after PS adjustment.
+beforeMatchingStdDiff: The standardized difference of means when comparing the target to
+the comparator before PS adjustment.
+afterMatchingStdDiff: The standardized difference of means when comparing the target to
+the comparator after PS adjustment.
+targetStdDiff: The standardized difference of means when comparing the target
+before PS adjustment to the target after PS adjustment.
+comparatorStdDiff: The standardized difference of means when comparing the comparator
+before PS adjustment to the comparator after PS adjustment.
+-targetComparatorStdDiff: The standardized difference of means when comparing the entire
+population before PS adjustment to the entire population after
+PS adjustment.
+
The 'beforeMatchingStdDiff' and 'afterMatchingStdDiff' columns inform on the balance:
+are the target and comparator sufficiently similar in terms of baseline covariates to
+allow for valid causal estimation?
+
+
+The 'targetStdDiff', 'comparatorStdDiff', and 'targetComparatorStdDiff' columns inform on
+the generalizability: are the cohorts after PS adjustment sufficiently similar to the cohorts
+before adjustment to allow generalizing the findings to the original cohorts?
Details
diff --git a/docs/reference/computeEquipoise.html b/docs/reference/computeEquipoise.html
index a2722142..5ee05a17 100644
--- a/docs/reference/computeEquipoise.html
+++ b/docs/reference/computeEquipoise.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/computeMdrr.html b/docs/reference/computeMdrr.html
index 9b8d26bd..b3a86948 100644
--- a/docs/reference/computeMdrr.html
+++ b/docs/reference/computeMdrr.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/computePsAuc.html b/docs/reference/computePsAuc.html
index 0c05a816..3e34f0e9 100644
--- a/docs/reference/computePsAuc.html
+++ b/docs/reference/computePsAuc.html
@@ -17,7 +17,7 @@
@@ -108,7 +108,7 @@ Examples
data <- data.frame(treatment = treatment, propensityScore = propensityScore)
data <- data[data$propensityScore > 0 & data$propensityScore < 1, ]
computePsAuc(data)
-#> [1] 0.6879579
+#> [1] 0.7343716
diff --git a/docs/reference/createCmAnalysis.html b/docs/reference/createCmAnalysis.html
index c8252a84..503cc5e7 100644
--- a/docs/reference/createCmAnalysis.html
+++ b/docs/reference/createCmAnalysis.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createCmDiagnosticThresholds.html b/docs/reference/createCmDiagnosticThresholds.html
index 74464395..9a0a699b 100644
--- a/docs/reference/createCmDiagnosticThresholds.html
+++ b/docs/reference/createCmDiagnosticThresholds.html
@@ -17,7 +17,7 @@
@@ -74,7 +74,8 @@ Create CohortMethod diagnostics thresholds
easeThreshold = 0.25,
sdmThreshold = 0.1,
equipoiseThreshold = 0.2,
- attritionFractionThreshold = 1
+ attritionFractionThreshold = NULL,
+ generalizabilitySdmThreshold = 1
)
@@ -101,10 +102,14 @@ Arguments
attritionFractionThreshold
-What is the maximum allowed attrition fraction? If the attrition
-between the input target cohort and the target cohort entering the
-outcome model is greater than this fraction, the diagnostic will
-fail.
+DEPRECATED. See generalizabilitySdmThreshold
instead.
+
+
+generalizabilitySdmThreshold
+What is the maximum allowed standardized difference of mean
+(SDM)when comparing the population before and after PS
+adjustments? If the SDM is greater than this value, the diagnostic
+will fail.
diff --git a/docs/reference/createCmTable1.html b/docs/reference/createCmTable1.html
index 48ed5d1c..10175a7e 100644
--- a/docs/reference/createCmTable1.html
+++ b/docs/reference/createCmTable1.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createCohortMethodDataSimulationProfile.html b/docs/reference/createCohortMethodDataSimulationProfile.html
index e4d26e8f..13fe5e01 100644
--- a/docs/reference/createCohortMethodDataSimulationProfile.html
+++ b/docs/reference/createCohortMethodDataSimulationProfile.html
@@ -19,7 +19,7 @@
diff --git a/docs/reference/createComputeCovariateBalanceArgs.html b/docs/reference/createComputeCovariateBalanceArgs.html
index 9501a214..5dcd574d 100644
--- a/docs/reference/createComputeCovariateBalanceArgs.html
+++ b/docs/reference/createComputeCovariateBalanceArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createCreatePsArgs.html b/docs/reference/createCreatePsArgs.html
index 13907d3d..e0109fe9 100644
--- a/docs/reference/createCreatePsArgs.html
+++ b/docs/reference/createCreatePsArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createCreateStudyPopulationArgs.html b/docs/reference/createCreateStudyPopulationArgs.html
index 141e6159..a7f0f7af 100644
--- a/docs/reference/createCreateStudyPopulationArgs.html
+++ b/docs/reference/createCreateStudyPopulationArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createDefaultMultiThreadingSettings.html b/docs/reference/createDefaultMultiThreadingSettings.html
index 2498eb4e..d700c232 100644
--- a/docs/reference/createDefaultMultiThreadingSettings.html
+++ b/docs/reference/createDefaultMultiThreadingSettings.html
@@ -18,7 +18,7 @@
diff --git a/docs/reference/createFitOutcomeModelArgs.html b/docs/reference/createFitOutcomeModelArgs.html
index 7b9aeaba..95a0e609 100644
--- a/docs/reference/createFitOutcomeModelArgs.html
+++ b/docs/reference/createFitOutcomeModelArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createGetDbCohortMethodDataArgs.html b/docs/reference/createGetDbCohortMethodDataArgs.html
index 0c4173e0..f7e33227 100644
--- a/docs/reference/createGetDbCohortMethodDataArgs.html
+++ b/docs/reference/createGetDbCohortMethodDataArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createMatchOnPsAndCovariatesArgs.html b/docs/reference/createMatchOnPsAndCovariatesArgs.html
index 18035581..47c14f02 100644
--- a/docs/reference/createMatchOnPsAndCovariatesArgs.html
+++ b/docs/reference/createMatchOnPsAndCovariatesArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createMatchOnPsArgs.html b/docs/reference/createMatchOnPsArgs.html
index 0c27fd97..e6ae22ad 100644
--- a/docs/reference/createMatchOnPsArgs.html
+++ b/docs/reference/createMatchOnPsArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createMultiThreadingSettings.html b/docs/reference/createMultiThreadingSettings.html
index 1a6faf50..96f1e04b 100644
--- a/docs/reference/createMultiThreadingSettings.html
+++ b/docs/reference/createMultiThreadingSettings.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createOutcome.html b/docs/reference/createOutcome.html
index 2cc5d955..1cc48d4a 100644
--- a/docs/reference/createOutcome.html
+++ b/docs/reference/createOutcome.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createPs.html b/docs/reference/createPs.html
index 1d781739..d80b3c6d 100644
--- a/docs/reference/createPs.html
+++ b/docs/reference/createPs.html
@@ -17,7 +17,7 @@
@@ -171,10 +171,10 @@ Examples
#> Removing 0 redundant covariates
#> Removing 0 infrequent covariates
#> Normalizing covariates
-#> Tidying covariates took 1.25 secs
+#> Tidying covariates took 0.494 secs
#> Warning: All coefficients (except maybe the intercept) are zero. Either the covariates are completely uninformative or completely predictive of the treatment. Did you remember to exclude the treatment variables from the covariates?
#> Propensity model fitting finished with status OK
-#> Creating propensity scores took 4.71 secs
+#> Creating propensity scores took 1.69 secs
diff --git a/docs/reference/createResultsDataModel.html b/docs/reference/createResultsDataModel.html
new file mode 100644
index 00000000..8a1920bf
--- /dev/null
+++ b/docs/reference/createResultsDataModel.html
@@ -0,0 +1,121 @@
+
+Create the results data model tables on a database server. — createResultsDataModel • CohortMethod
+
+
+
+
+
+
+
+
+
+
+ Create the results data model tables on a database server.
+ Source: R/ResultsDataModel.R
+ createResultsDataModel.Rd
+
+
+
+ Create the results data model tables on a database server.
+
+
+
+ createResultsDataModel(
+ connectionDetails = NULL,
+ databaseSchema,
+ tablePrefix = ""
+)
+
+
+
+ Arguments
+ - connectionDetails
+DatabaseConnector connectionDetails instance @seealsoDatabaseConnector::createConnectionDetails
+
+
+- databaseSchema
+The schema on the server where the tables will be created.
+
+
+- tablePrefix
+(Optional) string to insert before table names for database table names
+
+
+
+ Details
+ Only PostgreSQL and SQLite servers are supported.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/reference/createStratifyByPsAndCovariatesArgs.html b/docs/reference/createStratifyByPsAndCovariatesArgs.html
index 366ad2ac..ab838557 100644
--- a/docs/reference/createStratifyByPsAndCovariatesArgs.html
+++ b/docs/reference/createStratifyByPsAndCovariatesArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createStratifyByPsArgs.html b/docs/reference/createStratifyByPsArgs.html
index c002b77d..163a41e8 100644
--- a/docs/reference/createStratifyByPsArgs.html
+++ b/docs/reference/createStratifyByPsArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createStudyPopulation.html b/docs/reference/createStudyPopulation.html
index 9e712693..066f3bfc 100644
--- a/docs/reference/createStudyPopulation.html
+++ b/docs/reference/createStudyPopulation.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createTargetComparatorOutcomes.html b/docs/reference/createTargetComparatorOutcomes.html
index 48d9d724..5bee64d9 100644
--- a/docs/reference/createTargetComparatorOutcomes.html
+++ b/docs/reference/createTargetComparatorOutcomes.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createTrimByIptwArgs.html b/docs/reference/createTrimByIptwArgs.html
index f734c078..0059a4a6 100644
--- a/docs/reference/createTrimByIptwArgs.html
+++ b/docs/reference/createTrimByIptwArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createTrimByPsArgs.html b/docs/reference/createTrimByPsArgs.html
index e354248e..9f5a8659 100644
--- a/docs/reference/createTrimByPsArgs.html
+++ b/docs/reference/createTrimByPsArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createTrimByPsToEquipoiseArgs.html b/docs/reference/createTrimByPsToEquipoiseArgs.html
index a9683246..ddae52e6 100644
--- a/docs/reference/createTrimByPsToEquipoiseArgs.html
+++ b/docs/reference/createTrimByPsToEquipoiseArgs.html
@@ -17,7 +17,7 @@
diff --git a/docs/reference/createTruncateIptwArgs.html b/docs/reference/createTruncateIptwArgs.html
index ddc388f9..10b30d97 100644
--- a/docs/reference/createTruncateIptwArgs.html
+++ b/docs/reference/createTruncateIptwArgs.html
@@ -17,7 +17,7 @@
Returns ResultModelManager DataMigrationsManager instance.
+getDataMigrator(connectionDetails, databaseSchema, tablePrefix = "")
DatabaseConnector connection details object
String schema where database schema lives
(Optional) Use if a table prefix is used before table names (e.g. "cd_")
Instance of ResultModelManager::DataMigrationManager that has interface for converting existing data models
+to assess generalizability we compare the distribution of covariates before and after +any (propensity score) adjustments. We compute the standardized difference of mean as +our metric of generalizability. (Lipton et al., 2017)
+Depending on our target estimand, we need to consider a different base population for +generalizability. For example, if we aim to estimate the average treatment effect in +thetreated (ATT), our base population should be the target population, meaning we +should consider the covariate distribution before and after PS adjustment in the target +population only. By default this function will attempt to select the right base +population based on what operations have been performed on the population. For example, +if PS matching has been performed we assume the target estimand is the ATT, and the +target population is selected as base.
+Requires running computeCovariateBalance()
` first.
getGeneralizabilityTable(balance, baseSelection = "auto")
A data frame created by the computeCovariateBalance
function.
The selection of the population to consider for generalizability. +Options are "auto", "target", "comparator", and "both". The "auto" +option will attempt to use the balance meta-data to pick the most +appropriate population based on the target estimator.
A tibble with the following columns:
covariateId: The ID of the covariate. Can be linked to the covariates
and covariateRef
+tables in the CohortMethodData
object.
covariateName: The name of the covariate.
beforeMatchingMean: The mean covariate value before any (propensity score) adjustment.
afterMatchingMean: The mean covariate value after any (propensity score) adjustment.
stdDiff: The standardized difference of means between before and after adjustment.
The tibble also has a 'baseSelection' attribute, documenting the base population used +to assess generalizability.
+Tipton E, Hallberg K, Hedges LV, Chan W (2017) Implications of Small Samples +for Generalization: Adjustments and Rules of Thumb, Eval Rev. Oct;41(5):472-505.
+R/ResultsDataModel.R
+ getResultsDataModelSpecifications.Rd
Get specifications for CohortMethod results data model
+getResultsDataModelSpecifications()
A tibble data frame object with specifications
+Compute covariate balance before and after matching and trimming
Compute covariate balance before and after PS adjustment
plotTimeToEvent()
Plot time-to-event
Get information on generalizability
Functions for running multiple analyses in an efficient way.
@@ -339,9 +343,9 @@Get results data model
Get specifications for CohortMethod results data model
Launch Shiny app using a SQLite database
Create the results data model tables on a database server.
Migrate Data model
Get database migrations instance
Upload exported results to a database
Upload results to the database server.
Migrate data from current state to next state
+It is strongly advised that you have a backup of all data (either sqlite files, a backup database (in the case you +are using a postgres backend) or have kept the csv/zip files from your data generation.
+migrateDataModel(connectionDetails, databaseSchema, tablePrefix = "")
DatabaseConnector connection details object
String schema where database schema lives
(Optional) Use if a table prefix is used before table names (e.g. "cd_")
&pA5oAVzR<>U-qxWgbGX$uNu3S>yrvQ538iZ51bKK==;5H&BCfH+O;^r+;IYu5 B)Gk*NVVO0br@6AC>3$^uLC!A z>=0QtbtQ#VH~;)2-hB(czP>h}TRYo_M`U$&c8*adZl3>2Sx?@-85vQQ8(sbrH>T-3cXzSxiOd9&rrw4##W^is;fBBW7qm69&wboaJ3d3CZn9i znBw-a*DGI5zGRyvaiE~0LN$nx9XjX3u@;+XS1*@(F`j$WL3Q)q@N#<6z?IM*bTj^z z^S1zgtxM{pBPt!1-pGxPDTx&`1r8GbeK+XgeVDi)d}!FpoRE-E3E+ARkqW2+0E_-P ze&DtE*>yb+eZWDF?wS^-Kicb`mvKk&Z(yKh?z!$&6cs5}RaKD^6A!0|c)att0#Gq9 zu$g2+>?eA)1cHfZI-Tc1oPsyUY}c>PmIVaO9IW=aULQ}gVo6;8F}(Vt-366Y&Jse@ zN&w)jzt}2t&P46)>#Op uJj|uXJ$M)-BW67uK}s4kylVS z{%ahD?2Y4?9F0H8u<;&z#vvC-!}b36^&-~ioG4p`8M&fr_3cDWGrb%Xw8mod6n6$9 zyi;H+pkEq*e$4NOLM#Rncq?wIt}Z59kjyP1kxN8CATP}OD03QgYK `_TUo!nQC3&a7V$W-INpK*%F)1QKVOuQ zK||t4JyTs}Hb5%!uN76N#k&b4q?)QKkLjh@RfpJkuGqCDl0*Uki?*GqgWt}3?7_BR zg+qn3C!U_03WlSQ=0gh5jQTjS)Ya5BQaljK-0|a}v(fjJI}Ig=0H^HyKb?ZSc&q_b zZqm@FiM&f$LgK3^U{r;fnG*KyeY!`bw-=>W78dVWqE#Fn_h0Rfm9vSQuSC|L&U*kT z((EZ`D>i7-L1vcA?e*z8VW{A%qUbC#yV}>bwze0)8Aa3U>jhthX1Z=qY^L69rn>C> zE#vugb-iOQ*5^u 15h-^53 20T7NM z`2vXKTBp2X0@4xx$&Q1&?BK~zMVA94t3Pdi;YOnaSFGr0qPU!hBgVORS=?FriNy|o zOL&7>NJOk*iB#&@@2Jqmb2AV+LAQTto(rjyr5$z9mGF1hTbh O`_*ID67RBxN%M@2)cL!qJCrf{6(o%snq%kb;6y;Najp z3SqbR_MULE#}HtZ@2k+e&Or&CnVDzk;6#EOXU&<)lLyEySJHf8B{vFMW%v*tQkqCY zMd}N_?i;!b7cq)s<1(TLPqS6>XEQWGjPsF2r)yUywdZqtd8p3--tRjpen*pVZD(th zcXxL$4q{u1Kw}R9$xc8(P!Fix(b1UFHekt=71~I1?i=0B^(ZScVE7WfJs(1HAJ1lm zduv4vh NxHPuj=}2ahLA8lY2I6J_;&bl=EwtBg)jKOsXJKn z|DEgqe^u!Jn5+R}0CdOyG3Bjy5OVKO-N(nLSs~8#&!0b_NFrVe3m0)>9qK*B0H_Rr zpOl0|`N;!N(`cZ#Mn^@-MwCw#jc&T6u)d*I$W{r=kOgxT5J}M3qUC$4;bV@2(VP^K z>+<9>aX8>*q=06-Vx0ZDJXl_y4sX4R{hobFzyD7#XT+P$(oy;?HJ&AK2`P?6Stek7 z@r6HH-(9(l#OheoYOf^rOJ@oc2<2o?V*QY#n8NBv{y~r-EK`nL;cn????g2*zAR?o zx@9s6?%9%z+ e=tbf+42>w}No<#aQl4epUW0vqh?ymV5%Cfwz=5>-_(gzV*Nn&y@xG`i8z zITlv{Dp4G$?{vv8&mM#e3f`}(JhQUgcio9A hynT6X=Bp-icWLlJtnhlx@smHXZzwZp~UiZ%WC}LoN9*l*5^(NtQrCZ z1>7XZ$a^u;_|K(&_DD{*EQ~dKrA$wIx*L!;;^6qz_5Ib#3&_!@kMvVvmIxx!m1hW< z)zhC0L7kuMJEb+{9zAo*;&h36=TAx+*x)xa<0TgXCuU*iRag0#*B~^!)RB|o4b@v+ zjUgqo<>i~!sqT;2#ydeqDvksh2&2JXGYJa873)0%C7IAOj4!es9qAXBaB+y60A `#g0Gqls^=d zbHCS&EsZtgGnw(zD0Zs!Zp!SGzHDqaL*8xNozT1%#T4jrlvhe2BupW&*mFn|5-Pog zz!{$*v+@* 8iac`*ZiWQ*ZsbTh~j@hAuEp{TwA)s#EDzW%-| z1YJItRy!sg4HK(aYbh_|quo#?FIIk0znXS@zMr4l>zbZlHJKfnx{0NuDyqD^j(;1x zJb?ZfQ&QUw%Xw>O7jw5w(MX8A#7|9BRL9oyURG^WZpSCk9j&yRoA-Xy>gw^p<7%ea z^ke;ys`V~cY}kZOL2(uSgV2i$V1mD()9 de8Q_|#7ap09-$U0nR+&sD|D<$Ozf(sjuE8h(&OoWwZC6Frc z^2U^F%ZP}426xKK>n=~{$*yB!ZNoO+$6*c|Hrjq`FK@9b@9_@8h7|w1sgU>J@cQ`V z;dDK3W|N1AX+`?<465s=hZqw1<)doG# ;KR@b++ za(Ss@U06D;qo7+&R|T!2bporQpfKJIR78?WO5{6yyzadwZ(Ln|x0T>qF1=9*BAk+E zSht2+<`wn72mZF1&@eYogLk32x|*sff3ULX*3i^U+p +6 zlCL%|AK_-*`e+?VLnK9m-_!h(K@G1a`bqktB!C>Gq&Do@9}9CChet9$e?s&ZG|s9? zd@W8%;X)=$U(C0OVrSwxq>25`cQv5#;B{84LGRXN%RAQOm{7`N*vQjvPKGW#EhqB! zo(itF`PF %kD%=A`lJOl{KM6@)g^Td`Sc0`*+?(0v3iq&Z#*o4G@zii zH*>pSlSCZPC#JXhPP3(g*w_see_cX}r5l-aU90lv=il4fHh f9~WN#lC&`XP_@Ac7=5X`}-0I`r&Q&6XNn8qZDK8G2^z H8F7 zlzC+%6Y#Jw%k{ycl9pyI80_2j-D1H=#-2np6m)cT2%oNIFB} en&a{^p&OxZ%Pz~K}$f<3%=U}+nyL% z0@LQ`)-Lw=JZ0Ep2Xf16sQ>rogRI)W<9^to^$ilh;^}6Q!fXZh^XBpKmCgF?z)C0j zd?B!iLmBnrz%%0%F9R57u=^>m@tPu~5a1l4s$mqN*@w^kMkk8l7X$X4yu##Lt@o6S zjl;LL*?C_Il&@)53!R42g07NNP{3AIePLybJw8Wa^#?;2SC|usVR}uwwo>wRYdUj3 z*OvFcy=kk%Gb8Q7%h_JJjrEOhy;Ue31Uss|kf#8f5!>5`$E~QSR@Qw5#q5P^Yek?b z#_FkLjjD*=h!J7qqEEMo;lPPFK2?2`dIO+o{&F6pSTDXC;tp;Cg~2U_ab0+Q_UXU1 zo`3%wkTeWz7MAbU%6YoDsC^(_tE0hZ=_^;$)9ZKqW*ILWnsr9Des<=T!pcIfj1Q!_ zav;tshXj9{uHh(bYZE^{t|leZk3Zy`aV3{g#_u`3_KZD90DZv2BQ1tdsf4vy8xCr1 zbTyXosJe!LH9XG$*4O#Lx_lfn;8+X}CEwz~wThSmqu%8n>SJJNIXadV1%y2c4G!M; z7{;kTsZQhQ{gyb`qB~NNA#xATY_130lrK^J$piPIhu>`u__dI HK=tn%9j0gfB0 zl*eFSkugQqdZD=p3Vj2ruID9fvF>^vfGpyRgfQ(dwliqY{>iK8C#z|gZMNPs6#5iw zJwl@}RaA `q>T z~^TLcEZ)raP1X3C&e zRh@6`;Q~}F_SnO&baz$(h_-IZ&j;f3e&cQ>O8`+o1Rn(n4;znC=OeR(vB+Q&B}6=e zU`Br+)~O%YP>;?>MfcpD-ONvK(;yBltb$EaT`Pz8_N`CJBSS6T^zYpCf1L&Re22TQ zQiwKvZWxP`g(|iXp)%aHC6$(ug=&E(gF#X(rdy*5`bhk?+<(dWr2CKt=;CA>6njjg z^{#f!3dk*8hft((zI6F5V#5-+A)9wB=4@-{2nqNw7EQvp$wU-mQKV!P>T18|8aY~$ zU q0|U)B|8Rqd{~&u8J_-_d(g1qliU}9AwY|SQaVMp~hW*{hYfdSd3(_~u z&hF#jcU|<&1#PXMw1&WMw9NQ3NaT;|>fs0^t~!$-f$&fZU+Us|uL8OjU-|Ii8_?9s zi`?iWESZ-Oo=f#e2y+E|VnPFnlA!g8U5xy?*$7k7X}Y-5JVG)$&W5udY%uKip3;ik zxalZ~C2PpYq|G&m5D<~o@034SLI_ux1Qde`1&~J}R4(W(JpL6obSvI`6r;ma6Q~X- zi;Bp7Oe0EClE=rx62^ki@yVr*lu#;MBzpuE{?J%+y$X jE zK_S$p%eg;n)yPHqxsW}lpC>u_Go_o68i4nN1YvAk8J)UO4gF;L02d;_;B;E%l|f6b z<}(Cz?29 CQ2caWs9>(4;`{}e@KIVHAVaej06O1B8@db(MqmM; znN6YoNJVCK^+AI`LqUO?$yZyl={lam!FPsXB7A&YvHZPv;L_UIwGQCBwp%LvsGrC3 z8Ub-cUjxqQ?RyIfu($j`p914Gx244xgC;tT+jPy)*k3L3os$zkAXln7%?T4dLb}2j z=?cBy!W^SRJ}DE;UfR6;msa+tRSxOd(X85v*rj2na@!NHtZaG5C)NOcQF$T@wrC}w zx0k8~MJqMPu!@UvW`H~mQWK=uWE3NT%{OW9&a8u}V6xn_TBEq!^e19jnD^yhorZ~# zR&n92%uL#0pPTg`)6-X;vB=KYuW)C?w}v!2=4d_D5yS<;B#HG;D#&>J(9w zAv;~!qwdJ1-w@|sVr>DJ*1SCJG8X410ko%+m-ETi#deU1HaP?=&}xG$_(dAWvtOnD zRFvw5hG`-m!mmOrKYY!U{PMH-z0_1uhNARA 6BMt){U!-&*eK$bcdS2RuT;`~ztKy&m^iNL)Tc7Ngl+F%o?SSp>b> zuLLX>R3ea-CQ2cz0^h9Br~~msLYT=ZLP}+%@k2sc4wSTO*|4&o|LB-w5`4P#Zbu@* zgxIlL$gw!GRLM-68}D&EIKL7hM*{v13^XAmkPAJm?*&_5At9j=a;h?+ZoMWn-_wua zRJhwYiSlYXy=t7m45Mn~%=CV-+O}^awnY(ez44oC?daG A1jPP51`)=VMtZ+!jy`% LSIp dCe zVRcPiou=t H7FH;HNmtBvVwB_r4iSqs~L7 zN9~w!D|_7W*&RTbM*|5?EK`5Wq?|WjmAv?#oQyHHDd0ds6(%vazb_EX8|)@(Cl{dD z6(BJ-xe?=4 9`wVGE;{(ld<$+b7h!6{HMA~1~x3)N>C$1zx10 *X zQ&=%Jd9Yed$gwtauJ K_1k2{^O{d))*;!++3$XJy44ccJrzypt~Vl)-(&TJ{v!;;9YPd zh=uG@_YbVGD0OXS+UU_n{e8QSBLehzb^ACP#8yw7FuLC+RF`*N&5iZVjc@1wO2W zwM+&@#kE@A*4E^~N-qf4>&vaJOkQX`cW(PKYnz$%Z4r68>5vi$-_CJ+GDs>%A-rL^ zAgc5q=vUK!Tbrl^0HiZ9VI?e-g`6~yh$PY^Cbo}_J-MX~QONa{q=$x=EpR33n-zst z#;;m=GsloGdGG9%8jCC(-V1u`DdYc1#3$%sdJ&_nSCp11VFs`z 3~l=JlDcTab&}yQG9MAul@5q zsn^ftxD{2Uxk=Dl0N4fvuFyw y-@T<(GX5BO+=$rd=O@27!A2 z4_O{@Up>V}zv8Amf?Qsr7VB?YwB2u?^L&hszDgmS$CMny#hqbfAzN;%4TD91VXg5? z#V hNCn!m1->Pu$2)?ITeOXVBON+xC|G;W)aTk3#ZPP zFKu+Cm6gd+81eR3vu1K1{_}%(%Vs+zw?z^GV3ucq<1CqQ<9uN}U9XR@?fd1%0aPp_ zJ0u7jh3+>B%!+va)dVsSDk}Aoov2j!RSe 0KFxKAS^4cHs(V& z=UXYirw}Zqof#(Pa7%44AZKP~Vstq4y5V+k;=k3~&I u#&Pe`q6g z1GUu+H|_al8y}k4?YO)s82*!?#jZXML<58LE?s4XV?MAEA#iY{LAO&^C!o3>!d%d- z&X-$SH1Z@NjaU+JB4m$Ps#mR-ar9;h;?_|WNxK$Vwc09qPq*1N*XIJiFr(l3$6GUm zR+ODBxT}xn=eUvB^P8d-5Ct2JgGN2{U&2Z^zxx-MFL?(9epT&0TiRXORYipb%$Q?2 zF?Njg_ M{0pE7)@<>T-zZe&SpWF(btAKWKhQsWgkOzjRH{i^ zT3(q8BDaRt$nQAkk4SUGJHH0&Nx^Ro?s$MOG(bu-%DB*)nt+~Zms9w`8G|>nH&VN} zm}LA0;l%5zE*pNdh>)JEP1lwaa@(ay3bE{j69xtiT>v4GZXwoF3~`oR8|rC@abK#L zYUnn(F}bJvk%Uxa?DL=P?*%1fuXNfYLGxM=6Io0S4fv%U`TEf0x$O*NiPDz9Alho_ zXparCwD4#~qn??mDT2GMJ9)F7M^g-7#2|@SGU!<7KqHc&B k zBOA$)fa`Zx*WiT8x1qX4_k&NwDUXUOv_XU9e6;-jl5K9o53U$5W^81c0*-S_w0MWSOIWZs;TO+w^fgN$GWo{&8F~4 zKUs1TQN;SzO-EBGEvEzn+=*-i<=-m3y_w;nkC2l$^YKPWF9|<51-JE-t;N&4 ae03CCRtVKQh&V8-o8EL_&DAJy}FY2U=Fgta11R-S<%OND!S z+9XG%ZgEnU&ID{>8 !#^|HUX%1>^`l`LIJdF)C*e&_(pP;2={`JPBCv8jRU++WWLy!b zZ$>#p+VMk%J887XbL;mU*lW^ePS$WZSpEc dN)fQ`%I6< zqQEe+L$6@M`?4Eo1+4(2jS}u$7|9vseREiukWdb+Q?<8G tWRsjzJ(sN|tI*p{+6uu)qA{ZcZh3i==%}Pxj=<$%WcwZdd6hz$pGoTt1vI%_ z<%3Vl)tXs$s<)1rc6<`7aJ5LE)nk8o_AgR`OQN_Nion1QJ?N?Yv^Tf~##9lJbg&no zt}3yPKqGo^aEeeD;7kMI>oqaa&{VQ619AJvh~s{mf`1@d+n~HRBi^mc$sh 8tPEJ@VY!mxORqD1UdJGygz*}9AyBkSpcP6|C zCR}U=f7 vf3u*fmN2t&s>@ zA|gUgyC d6~U%{m#t2ZWfzt#b%mrdwk>x zl1`A{U3J)bC}jJzWhk$1Y7$8FK%8gS>(=taWIqsgZ-B6plPC|vvbzsWo+1Xd|Adyh z@}h!VRU4D+z8prZLKiyC=xyK3e=#H8Z`>YLnZm|K_An6<3@CJkLYQf9m^vQNDQheI zlT+3T*TS9EzEwSFev*2jiPqKaV=xedov~K5a#w3 OMr^N8Md!V4`$icJ@42N=b>z*76Zf!h1hAG z@^AJXM0)tlGviU)v+7T}i;AO{>d#BC)}OCJb#%u{b3l#q;v?IzE8Nelj*xoqPp2ZC z$qNHEkNuBQ#>N<-ca-P$FIPN0eL3L^wU@$0{d?YwkU7CRduZg2RM3GwP}5q`x|CwT zG+z#+8xZqKK|xQjYp2urS3?4oiggfuul4in7pw5SKAKr}0eZW}e>$p~-%pO!x;e9C zPG4>`k?EKwg4nZyM=m}YXYQExAqTgOohS(4plfE@uC9TdUD0z5nuDX+PNj`1gT2vs zLFGVraQ_12YhUe3NNFx(s^a#4WHOz~?umod7={?qcC?k6N~R5j#@|WCkgzUe5+LfS z1Z-Q|4*X20@2e^my0N&)^dLJ*QP44oS57JHDB^w#GOWCykRT+iPwbN-n1a59;>5#X z`)7XDeA?>j@;38q>k$ejacmz!v|rfWsYwxE(M3c~{>IfecmdK_SUoWwNZ`3W;Qya$ z!9YhR2o3EmDERog>b_<|x7FL@w3E{)YagXLEIF}-$uzQws$T9#?(3+u@lj)c^DH(; z1y)4BwIMYi;inX0m^QW}kJ~Zj8&}z#CQGt20Sf9>o-scIH$FVNyd5sn33l!*faImF zo}*Emq5G1my;;=V9~U{a3cRGNC{mPWiQX&cL`C*=1I|3uCIZ1>T?Ud-juQ@;&}3x~ z-AXb))=W&3NkGw`Ac9X2){Tw#pHA%n#od?npEoeIvW~tIyOlZVjplv?1;-1e{E%KJ zL5los=w0KJL Qa z*kPVbpXZi~uw|QYwVpQp@e?A-g(ab3Tig$hYd19==;%hz)v20EecxI7fTpisPROou zkYK&Srk}9q1X`#W7P88D^1ue1l8<0PbMt#3^nv|Wi#b7=VBf}nfI=ClMz(_%E+PVy zjd*qS^B0&EX$4G~(U^WNc{PfD$}AEQ#^3ABOwq?m)~O{T%8WkG7O)^|e0n;{Yn1>* zIk2 B}ddcU?^R&Z96kLxRcgf-f9`IP|mp0FvgUr_=n636NrR%@zR4iEau ziC#nrE6ulkmDh96#siy-m@mLK+ZSmhmS7Q6;v7e&{!VUafS9k=`vIxPWJXfqe374@ zm7G*%c#Kvm9IQvEm4I%g0>^lx%8p83-tQ*-)!M(_Ne+_2Dq-+8*zOm|Xcw1mF{qk1 zq$bJlAyA@r29xCtysDD0$>N4U-@w2_`gJVoolN~+^HZR!Pd5p%jzs}Qb4B0*+WC`{ zqwc+p_X3>HO}V+Pt?2rCA=7_lA|MSc-qo=#q3p1#qVR;w+0}V-Ta_)n*@M} zp0l$bgpw-{PceI(JB3_dtvh|%RXMoWHs$ICL<64~Nl0_36QjK^#*|%M|0FkjV6vTF z>2-Y!Iiw~fi-i1319`Oc?-!^9k{%O~M`+cxHfpae3T!!n)&g4+e5H|hTy=f^0^8Yl zsoqJWG^WWSypdKWYOaAlWhk)+Y4pcKGy0Nx`G+mUpf{o2A;#f~dpOX46?b==Pj-4( z5~KO-T%r{55NlFTx~;8LqI#GB8Piq+wZpccPR)VEtA@q;2%GmbwwIgc`oO4(oM(7z zE8UaG|5ZDfEh;P{zaZ?s%Y-sDcr_IDlwo_E7MvyTMd=%+?s0>hHMk=O>(t00e -_n(Ppb2U3;)S?ty 9{XlR^LgFh$o2PrC^1&3Yk>=0jE>L(<)Xoo1&*2=~W z;F(s$asEbZpy>Kg`X{3$*)HT%m9(!&W7d6)Aj{0vn#vCnO4#9v*
{BU8NlDQd4N-8>(>CIcK8{!t%9Hq{qaHJf7kaY86FJ%; zj$GgG-^`Ma4B{x4{TQ1MU9ETQituVLU%U!uCyf+^Z?oG(N9a)@Of&PQ`VSI!CNv^a zuh{~gF+L(9%D*~@eSnFH-+b%UTiO!(xTBUf@W+oIR}Tpqkyl&_;b*zJS~oQ_N1d2> zGnlL}Jd*Cj&!_xNOcW=@e`>nj&Z4!TG)yRZho>6Ni$XVR&UZu+vrZ+D(H}a+vMufF z`Cx`&WG3ixBhv_LzF`lKjqUx-epnAU@%U}ZX7=&j-ASUA{`k=?0$ryFn>1jps3k71 zB_5th1RMgs1IO RY;7H7BD{LbXfK4SW&iq(D6&qe1m^k;y+1mFXo)7w z{gvISNd1FAbfzZ%AtR-UOQyfwwTCwy;&jIS=dMs#+`t79JsWvlK^!q!`q=Tw$FCe| zUMq!tLUb}uPt$1W&`eLayHTEm)56yGr9}O gz8E8(Q(kZ zH0sxTYF=Ee&fw3k2-WY#+8rj3?EFmVk}Kv83#hbRUd?l z683df%uLIApH%#{wLN?G> ~Tu>s}UMJ9L++4KlmC*Scp>g#0PDAdILV?x3jQa8II|L%>w zDo^1)SGl9pNCBS|3*$^E%FHs&IpA8oGxpXdUqMFvmcENp*WbU7>Y%(zd}Q3*%VU2n zMi%z0FvR=Yk6qgp{g`6Ea{jio4-<+Y7sr-MZ4PBWFo*~UG0Q0*xUx79DA-tu1m)Gx z07W5F!!^=1$T?aLin5zU84BW01D7C|7gT>EDY-&xV=tpur}8{8gqZR~HkbY3Lp_^0 z(vawV;V=2jiusaiOhz@;U$#3b6mas^D#7V`TL%7w_yN9;AMzlm%?>(Yvwe&eWTdj* zYdjAkBZ1ps+KV4#YwP7ZYMQ?nlPLjr9WblhMaWv7z9X6yZDLZw&rcZdagai+9^PZy zxe#4W-tYU3TNZvUqz(DZz-_Hnld-QTwEI^QWCdW?4X$>r>+fuSCn=r?^*PEOof_9v z<0&e^R}UTbX=Xs_V*8w!*RBu#p-=|pi}QF_0`KY5b8w$-SNr!13#la}L_3LJ=sU?V zs3SVrr~f|hHw$oj70g9C@Z2M(uuM`(Atieks_|*-e1At-dO`;tN{{Y~Ca@P~^Mj@l zvOuq+o9VVoy2?L4tY-kS?OZKyk>Kf5cO_!d6llTkWTAVsw3N~Ezq1wOMlGWo$}l?& zDT*}MG-P-L-p%G``6XGr7@pJmXjc(_{4s&*ZHf^?F2%Bw{f^(mh%nJW*sw3Zu;Q;; z*>jjC^IYxc{p-6-lMNaB@eo|BmAKZu;`dSI&_ c;^=UiNBZS!PO-3jwI5Ee zkeOD-g^*QoW_wq&`$Up(jO;Dl&W}W6#`D8A?ND9v4({>Ka?aRbE~m?%EtB! zJY4hS1xFW!5(tb{ltu{v@r~=FTmmp^b5~T{v$m!Nl~Y->jaSD(EW!CoTSUv&gRC(B zF&G}+(ew6~FSduXZ_WDdD`F EAH&5(6 zg%K7Y*Hpe%LJ6LROYF`;L;xG?6s}>Vebh4DF$BdSN5miv=S<)-`eU>MuK}Jwg_@!iG;@{(AtxcEN z@Rgr(jBhT4`12p|!MBTNr>qw1_9 2CW;l^FAKqyPu72;Y_g|RHQVu&i+@Y z%`&xhBoJXxfUjDu%+O1@{63a-@9eNGfSvIv@C7?fjE`?~Brnl#$dPLKAsfO(NPDw) zJ5b(C$tk?ERL{xzQw>r(^gux&&M6+OE6-3yMg|Y1YY)HR_O{@vnUWlR>~7o~ 794sar?6XaXg%fc*e`;~n*YbOFJOmY(QRl;}`tK(9_9*oXJak-b zzF==<706>gm^VxCH2-M|k>hIbLsnMN8oOw&hY!*9^)&YmF9Y_4+l3&U=fa(*j`j5* z*PSLua_|f`e*J>Ujt027ZwVat`105`%xFFL51Sj8JP4Qh=jXpa?-jeU`22j&C-uVK z3-ffP=Rd9Qo@yj5=_l4e8a_Tg0SH2G7J#h@?zi-%9!AXmA&=R5_4d=HQTkXl`PjB= zhPc$O2lqxM2sOM!G*}2Abm6$KU ZNL%-|hDjT=n$SW+J=oJyjkdw_C*C4Fb_ZW_$*o{@!n0D16U^;;=QIufLA>G~A5@ zmEWE)J^>>>>E+7d(^!sN+1#(SnLVWe2kseFycxyp!);dOVac`&onbZrupD;ZrciFK za!|(SR1E??NY1IjqCVFn#QjC=<%8>MPzPTwO1-OeZjiKW_G<#Oo0vt7F+uqh8wz+R zZ?@L)D|nP7KnQ{uhQa ~oJn6 P^Xt#dR+HP66{BHwmOD%v77OGY(gOd&Qyduey;%)%X1nzK8&FrLdIMQc)opiGz82 zO0R!Bg>V6=2!%$8!a+r9wlMb_0Y1O6N&nkX1v=ZrdY=oWy-nSXiin->xXtMFc#pTX zj$FP73(QoPu_=_&PDSIMm~Fb$ua!DBZ1S+HW{M@3`dn|L2%r2l&KldiylAp-K4-XH z61(zm2&yqO*CxE(7rT+6Jpb0PRoZeRczd#NGZS`wq2zO+>h*)@se8BgL8LH8 6AyKWINpDWZg}!atg^0Z_CBGolS@#_1fI|N z6k0^6omJm8Pv?6Qk~TwK+PO)8k1K%z@Lc*!? ygfD?8yWd*s%^qeQleMa4jzsjTDYED*x$AHNz@P#WuYnOXu7luATs#7 zHb{{Qtm83{V1b+#GG%b{eg_0DPzsNNf(5cF?vVULVQywt$MS v>rn?MLMye~kvy0Ss)VN}l(i=tx>ThhtwH)D Ld(@s(6hpd52!68dAeQqf-j@CgJ;PJaKcK2g#xs&$GQrf&`Ms=3xxMw}D!GT_X zE 4xq9@zn z3zv4fIYN z
iptm@S+rn z33(MyE=l-vV;8N9Ixg|+DPbkUqrBTKP2Nsv b|SWoac=1m9} zMkg`I==-gESA%=Mp8^CHeUolf3$2oULBu7z6ZuY8uQ4vIGk1yHB1erzPoI8z{-N=& zmy-rKwI!W+1{M{ybK;``Xa4c L{9DdZfIjwm$zGm7>O zm@uEIgw~Rip8nZ+eXy*m>W|Mk$L)6;J`YG|*We(|E!^h{7yQ-$3vg%w;&Tcyeg|eS zloWK$l9e8l`+(n*cP)c%Ao)fp>7>F4c*HL*nWCsuy6BX$1@BciROi5jFDh<;H=Lcp z-Mi60d-tN&xaVGsfOpdGI_GbNg=dkGZZ-C*P{b?65V0J ;;oYtiXSV;7vXeCuJ7a02B( zI5>&Zhf< lzl&4iBIB z#d%T!@;fm4Q|F_$L$h(vNY$rKZg#qQY(Rb2IH;0ysPj=NSKH`-F?}>~Nks-I3zxYF zr5n0C>Wf#A!0uLR*m4tKSi`pc>^9May9fEpOqvg(OLtsfK7Bfc+DUwKv)30K3I!sH zjV%e7^}u7c_z1|z2_KjVDyz_9zR!*0gLW;lt~ESa*7Hw?EHj>voOL;WY3!;cHdl`R z$f8L&P7=aT2up6fLH8Cp53&DEAP3Ur UmGFam%IUta*`m1yl?pw^j27oerg9QpdsE)jwfGU$Z_-L6-Cv#S8|0G2l%jgn z%nxv|!&@9Bc-(=CfWuXs2WHjQI 3o{@Xl;!%MMjNfsj^DlKRR%{wlnMhqUx=qs_dTk zVNoyuK^mm=kdh+ZDhH8PK)O@9yA-7nL|RHgK)R$Gqy?nAOG3Isesg%fpS9k1Etdbd z*17k+@7XigTyxFr9$y-bko>YoiI$Cxzhn1~Q-d{ZB_gi _rLuueQMbF6En)W4UxvYh#8_N!kdMN=m#(s8>0v|*LumtZ>pW{VKK~5SjCYG%a zSoKcP=*L!S_K%(7;61r`$2H4mAaCD@;y~Zsjg`VsWHX@V$9G1A%WIj4-C{4OCHQu|1ks f1Y7O&tcNC lZKVEWBWMl{YXkbjBAJrj<)XWeU1_Z+F)66^}=I z=0fSfBjCni!si%`K@pYGz(>JJi !J2LwBVL~j$M7y=%n(ipDz9_16Bg{fN+r~K>=;G zdmYSl&z=c^9hH`1pPx$t5jraHy79vY+^(s?4Xtn;Tgs-#hs5D;fgm*ZbxR@>7AIYK z7A?ewW#i^<8%oU(8+Tt*D-9?i@UpFxvT%T@@N*+kU)XiG3)8w*P0xwvUc@baZ;>Y& z@oZ{0di|X?d3akXB@Cw}84zqvn=f$iWpa3n3kKV;2!u(Mb~@e_Upf3mmat(3yitmb z-&|L`3uyQ*p_>0=|C~Tr^&_Emm?J_L9sQJ; )~YnqW1l?XC!_sGH!tI)O$phm{0!v-fpL 0j17;jHN{rbYg~U49TXpO}o(-nd?HmwemL|5lM2rb^x&kZCZWFtYF$A2?k4 zuUM2jCoOmkT}Ei4ibzRq3LSj~3Q{yf8Ivzy9?bP9MZ{ruN|~j9X(guOC^3(;B+6dW zKZXj%QaF@;5PbONm5RS87c#Q*rOP@;qXo}~A(w7LicGz%g2x$Rc0oLpb*+pemR zx>nNSk5#yT#K6LuIYp-7M?1mFd9nDM4dea3^{A%Db=iH&@N*x{qn!_>zmX92d`7zx zJujG-3t3+HWVWysP9XdT{C?GcYsd5RTN@5$VN-VAI1$mTT=*oDH6h#v?lL8j68>dr zZ5RT3zoD<;R~st(DKMZ=hWS56{J}BGp}i^Aj9FOhR4vtFbsZkgrnCNf#~DH)p6g?j zKU+{V37PdJk&= EbKi0SwCE&Bd`B@3m)>>L8(Rpz(RDnwOe!EF-3vMWs^>R4;O zT*DG>+hUV9BPI(!Nlr98ks+L&td%XOl$z}zl1an=@5isGO9ltDWo*EllZnbj4q;08 zuzULzT)gHgV^2&lF&Z^PU%QicmV!kZ;?gF53{KQI3U4}c!|EL0pP&$wd=2MmLv@8K zv*F2>=h>_t&{mi#`S`Bd6TM*S3nH1oK!(c?3T`SLwaa&~u%PGUpeOK0%(UTYg_=Dn zD8H!}@PN4R^XGa6G`b=ZV9 3L1K ?Q*&4?bAPA!eYkKzaF?uo4{ zr9zw)^qLv!wcNDVMC>hSmJIEFkFv6spPvu ;VUaacqt(ani?R(0jICLQg z_~H35k~U+pmKG0n4a@Fi`77#NL+|h1)6{BS*wvMB6nYmRM5y(FuKKsJZ57&Zpo`5;sy&Aa}?q{jXiS#xZlxb)S+z?i|PUD{1XH4PGsAe*~Nc0 zow<`r{Lni)WACBfIV-=_?gyvFg5IiW`@BDoJuJR;YYvI)82Bb4*B59Cv}FgUr*~%L z CcaQS`%d86N>~UB@^O>`?u`QtPWmXTf5)XBob=C@o*;PU%np_Z+7QV zy6VsO_mzRc4MgIgK{#|(Ut^N7 3)kpCoUIhjj0yUNsZ7znYj&y;2OgW#vO zuey4C-iGl2eO%IgbM1oGnleUN=-{Ao*RQPHx1B@(x6U%#ftePg-AGLpDi{qP*aI8I zkrpk7a0nf02FAs`DdY%Kc%ZLapOz=LXeG QxX7zgqd#jSv{5(7 zx_*3QrgJqr;=+(u95Hlespt2+n^6_7>-PDzXX0FO&%f0xJaB+VRW83llKbBhOffMk zMMaVLcsx%}R3vQA ${ qk03L|nQ_6I{5jth}>7LCO!&HuKgfWoOazy?wos(z&n?c#C#~ z$u2(s^ZPq!>bBq_lHy{wTvyT_L74&z3yndZ{?Eoxpjincl*c-jIRph&1JRYWv?jul zI3*5 CqGX=Z3;A z(Fj3sFq*ummZ1TSp;<>p)%fhdgQ%(!CQWspd-s^43gSP{h0-7%{}*yy+1nSMaVnC_ zo sC(i02VrQ02G-W1VLB4=MMD=*tEftjqu<~?G$f*HreoaFKUaIGW4f|^t(a~3* zJ@fFGwpmg5WcX;qaOWXLc6RJT+K%PI)wP%L>_y&*ui0#V6t+zm(?Ea`{Vdcy;Ahp* zwxRZ41UO%nn~EZW0H80#ehC` ?(5wnrhj&H9m8`LDVV*fX>HQ%|_NVpm=K z6mAiqtA6-k%YYv)Z`eB?{eR@ECP%aVe1_`Cz=Xx9Fp7JfxtFflLQX}ms`|y|@7 -nadG5A%WL-RW60+vZv!Zi)U1RW;tbR`eswEUdr=bjU1CChr5|@m;0cd_w;eO5 z+nBhes& O!b8Zp2Ggl^>oCdj-4)sem ze^HT8UID4?Q47kYUwCO9?$`_CU6K2HHWXO`ng;Qb+)SCA}r z7RO<1<=JyyvO(3CK3F_xR%|k1ba7*knJf;YxNkdqwYTN^zXjT DN z_tG-0ONt3>_s!mRbp`akOm2A=k45rhZSD7#XUe`~iU-vd4UGJ@w~GRP!HN9!4=abH z=#x=ddgL|X5BIAeu6%;pvW?|!Z!^rzAyz5&n)ff;QC0F6D^4}wn^UBkZ5E}HK79B% ztA{)63$Ulu3Xke;LkA{3ih2gx{-lY7e?%cLAUH7|0y-4Rn~P>nY}ltM66jX`gu9JW zW@KX#h@&eis&4AA YfSY(^vvl~ z;V2aCg8LR_{m0K+NR!<2ey&^8Lv6Hz!uHrM$Uz;exQTXbr2fcj-=;(fvvcq@PPlsS zM{IEYhT!<&8N@dB>$Ki>wp6WsH2Xz@-P`kb#E @y!#sI(@sExRnp|^)mu^-Y#BbSll(R`b 9c@a7B%-(?y$Dr-)Xz`Vj;fK-UC%~v( zR)Yr6FYBKw^aMK#5g6c@+uPrS;()YIiwP&~-n|aPfF0K>IV~-**8e7!z75#!6*YBtbGb*EK@i-1isiPTGi18!as6vR`> zCmxvyA*I|mwfIqP^W=sZr83le>4UQ7$yQf4l6=?@CAn3FNK!Qx3vT^7=ilm$p-0|s zk2n}2-ff(J8YWd_b!83i6Fs}8N}TC^07*Inuy$Wi=7}eSOXXc%-}0<&tH6J)u{k(T zphk>ERl6lpD)!;PtQo@p*Y*-3yij$*s=gHaf^S&7k>@O8ehtO*bYI0c>Ao`4^XtkB zw?(3xVqKS*5CgR({+fghit>wU+Bjrn+kuGS&KRi0DU#r3LlH2t6A1ymomLr&I<$Dl zk^4Uu0)cdOoh^y8q7a ZzTa-G@cjp?m^`K6re$ z%wKyB^+)CMnS>mCv9yUpbE^-rqGK8kIifJ-+}$T^k+j6b8S2gfA 8GaI$~3VD=6!ztvQ-g8+_#N^~etI+(}91LiR zD`xf~=i(z{VC#}wi!5_aP5@3lw?N4abg@C9QNY?1MlV03iU&FuTmOh_8ygMgrLqr2 zbv(b_%%a-_woVGnJ^U1fHy1wMCEs;jjKY_eM(gZEgZ%lmwMvEih%YnqVACKb2Jpkb zuWBs(7M!dVGqgxA?ou7%VVB#R8L{~nAf-J@iS;}|03-SptE=q3t|$TefcGME<#ph8 zCTK4A{x|SzoIQl2BRAs`Z}U6-nmw{**igkUV+1O^nGjPK+X*J01R@Qo3*Hn M1cP0rB&Yh8g>hzU~~zW18)Zh?i>GkgOX$P2V*6e zs |7igLe0tOgBt1bcTw}==i2h3Vm!qQhX=&urXrUr}7KkL=jr>j_ zKS7QWhy8Da^xqMmZ?;Oq?q!H6%1506e%i+0TX23K#RY`pQIGq*MI(`jAVY_<3-KkQ z$AAw1Qc3_DR>TUGdvW^?1N|Cb3y${U)Qs3p?w^#hX2_JPc;sb-1QiQN`2P8e(lQZW zVu&K^%V(ei{=e`Dga90l7beyG@ZroS8MC!KCHUrd#F5P@Ko7JrxYaFKO1XyAsqBbw zj&(Ms!^b9 ^IuCb?>cF* zcWh`oNe^k&b@Df+N?6vfM{yx`+ATVP$`&{1%a5*3Ba{#Oz7SP2|Dz OuL5ugL)L>x939@(^O$k-YzM8alpI*DD& zl%Q&O?8snEF9s~hR%^?DAJQ5H)G}ZGXO<{mNquN)?ab$?&XCpaFD@vE^li+?QISrA z@;~B6h`lpIe$PlH{Gz+EfGiaJ??pA%dB*EHM$PT7MNWHg{ BJ!hp=~JHOvUy9Q7Tm^JndXG5s>|&6FQW?(7(W;KHYn z1E|Xm_Bxs4%?m05@k?z 9po_ZCUl z)kZj{#?Hm{YA@MUy3)Eg*-=aclN44}!MV9*Z1UjZY#DO3)+4V=p^SbiIG1mJqy{c4 zY51Tk0v$?Jke+TmUEDyo*0C%qWMF5{lR1=s4p@Jy)3vZq=AFM#!!gz=lLsuE0y9f6 z6rE_@+(BEo-P*DxA@Kl4YJ#)2%>WWD&{x?GKH9>vX#Q5Gj@z$a=oVJbP-G6*pImK` zvB{>v#uxKsNW&gJ;=MZFyxH?lhoilUkctDXYW5z3*3Hf4^dz824U&ZI<>hwoWkK}d z;LH6u9xxqbM<{@j$5Y!it#QY$ZO5*Dr61?-U$^*}Fk(w<0;1r-$*R2Ura`2k{Al#o zI{V=C$ey2WKtmDow#q%KodUio^S+!3!^X_L$WD|Ts$rGM7vxZz2Y)Y#9FMFYJ0y6S zD5VvzRV3tU#>O6@RI)Z_<$fC-tyn55*_ hS}NWvA|j*WiLnnPfEPCJKl57@eh9toj;E`U}) z7IGv<6GQ2XG7I-%kOvSg>N*UN>abu*10f?sTZ%yvYVoL|7p2X`8?WXY6?@Hcy~^IU zHR7&bnnHP)kdG*TAg_pA%8M%o0Yk%?7saAYQW^81j41SKdLtinAxo!z3`vZiy8M7F zrH%MKy@}#V7w3J=%`fvo)%|?|O<5!rQ>V|aMCt2%cFvmKU+N-VVf10U8+Ps!z2E#) zL*I>0%g+rw@M#vW4#zudm)x2D$1}TQcWF;lOu2uc=)`D%I;N04)6j_My)JinAc^>D zry` {(Am;+vA`7z0`p@5?KqJZjbKW^33v6;0wWk~*|mRK zSSz|NQ^JARnZ6K-(mA_oAag0aqr&{~IzV#u4u!FSx2}Zhqd3|=Y!=tYp^Axg*~7j5 zH791_fq@@EBcjkgx3dVN7(Q2>+KQBCj{;X6*yMT(yR7W1rk_5ZVBF8l>|nvZw}$_x z6hS&QmHribDWdg i%DSz27 z@XM+MH(bCwQ_r1> x#Dzuq}0>-#uHLMmn0JV6w>$i7n9FJ7-MM0rds&D zC^~!dHsO`o7-|^Wt@uLioim=@LFdRg{z0~3!Cg>3SP%NCjIkYTr^TbDUFngr4)^J) zx|XQU#n01|4t3af@^bFL>*E?!x!_f023E3r1`aHXs3cM&uF&G&Qdd{M0ykMK>>i`^ z6GMG{PeGJG*`5chon{rXuh8rg?H&L9`|$}H?segWo{Wua{(s-aa4h{k_l~aF{bw$b zPct$VTJFT!vt-=%^S!)#mAJL_-0us`jpqD|5S;lQmS2Vs^`1i|VsNnGzANpSUSE&= zL7$$xFt7`{8h*!J&z>q$y3hF&dry=F7*OUVH){)6 t dM@)TaD35BMP*P}Evut0%MSz2 z9iV{!qY3S8TH2T6v o1r6Z|4~@2*fTi3Jd*dCB79vK$Y_4P=X1ZvQe>`gCOY2)QM|}8tuI3o z@XmpkyC yi(H9-o;vlk`%*7e3mtp@mxEaa@nCCduZ3<=M7v^{k%Lzu}7%P>sgk6)n^P zUGF jZcg6jX4=A7SU06up$OxBP?6 zS& _ZU?*769)r{q%#CuyvIKaq|Sn|oZxn;!mi_C0!-zU2l`?e?HWB^0M29Dd8m z?1euFL4J=X4$yUu4*<}HPI)C}i*ArODi@lE?4wu-F{hvss)WdiuJ$GD2}#g*$z=sx z(FE%3!#VL!*RBDsgRit@V!Sow7613*STwVG`#OqidjLW0{@n#eL{ZP#UIrCZ)q%+N z_0hBWO&6t5!Hio^3f3o67{GBo17zMvP{A}iBCZRRp_n$7{>IS*O6|{la=p@9g2H4D zj* m!eeU^hm-h7J@#PaA zUz*GdCbB%O&f*dx4xXjnK7IR>A2_cfw17#|4E_G%N@w4h7GCD9np{EJX=c~+-z}t! zZaCfo1&;;rS}5hs5+)~Pc6udlagsG2yFx!eE^*EH_@4{qoDevTAEZ3VpMq) z-5wq*j;C3tt?MbQZ}^iT?#Ifea|s&KiB40aiS*qA!UFQUKT@9~&E7W`rX%twIY2>z zYl*m~Vx73S7GrEGx3b0TdX&&Dw;qZEHFQutRBM!hP#yAK5XXj|kC}fHIpqHoCZs?a zP79{<5tU*EjzXVOY~pQz{@X%LsbTbzv`Q`Q8|c6>0|_k7cl)+&E7VIADQ_ETG8jSz zH(SDU(nTCr;~s6T`8lKbE1|OxKRQci f7;f{LupSQsqBzQA-p2e+I@ z_m}f`Njs>%#j-8;oUQEkhfM!{^L{&3>M^msP*9Mg^Y_o>$URNXpX8Lyav8RvRLz9J zN^ofu_yS32DxvL8YPNG|@d432E9y{ks6oAO))!PdKT*Q|d};TprIPjyadY$kIIR+H z;It69vSN>P3j|wVr~A$h@~}BrZ0|j1_qMSLYyjo*_QD-NqZz5!s#32N*T|Mv%RJ 3Ul9JKvh`nVGZ2fn>M@I8~9 zde39q!I@&~nnEdYp!^$wT_$rYpmc*c;_lMTt5;3zbi2yNc$j&>GcoUms1`o)ZE^ zy25=;HPoM~T7#9vPuTtS^%t&dtO=nwpJLeO5B695a^S6lP*x?cc84^{CFL<23AiFi z#%yPO$e{v(_mU_0mKbE~*?Bk)L1h6HS8t8~o~~$@DTN`doKe5uVvI0VoNfDxQ)tlT zdz|Db1tG230hHe2<+(XLbko@Qrm;!k_}EQBK`?I+n)tvLxX)hQ@WmR@(i$5D!uU`e zb%HY5NSm 0QQC zoEFt}7s-G7&918x_cj9TR7ZK_OiUb=-Yw(h<>DDVI!aAN4@5`xM|f>^U@j(~p;uLq z9X4X%XUIqJx#CL~?*ac>h7twsJb+szFw3mnW#pmj4+)An$+Uruj;V)>k9voc_ieaa zGdECVhM`<3;d6$k#`axG|GW=2g!sBoYd4QhU6X1G%jJDYp8KFnzEntQvyFUawm9<1 z^291fMk6C{+%TOCZv5#FWrz^45|$lbp`BkTpI;f_U;kNYDKS9SF7Ud&$VemFCF0)m zk5_KpQN7}>TUa&X8LD%k8Wf$=R2_{Q+GGBV#xBf6l$6aWdHM7#Y16r-ftMJUV&s>< zb4YJ^mLH|OGCwmhO_3F)@ua}7#;XnMEKhi(Esd{tmcs5A2nvA4>^oczq&tp0IbCCZ z%WL&ht@xSP7?j6lmEhefp0%_JA8h14;$Gww{#)#7A33B< rXu-8)@y9d2`PRmz^jZlr4_9R~3j0le)nAG1L?31G;t6)2 zLub=XQ{c0-M7OthF)*0nQ|)C8=Qb}G&bi+=9o9b`mg{wjKx8Lj!nqrMEV5Z_A|L26 zW#TZ9RyM5Hbr+Vv^ys>1M1v9&sewZQ10q{ny1J!1iK5kHoKI-7r8Lj=fP!o$*(zz_ zq*ile8_8C1Md3k$5~j(dfK2YQk3GW|@o^3D3Fw`TR#q)mmd!ne1ridO&sP0+q43iE znVU~octluu|H54KG1BnW%eRo^H<*>Y@uy;n=(#~5xb#VG++r%{tC^i6hVRvM^;}$) z;O2r9x3Itf+EkI@{Utt;@s7ecp6g@5Le53W7IY1{*Az#h{qo!U44U=%ma$5%Jjr|y z)u5eYx5D8=w1No9oF!wJ50F!AQmB}T)$}SiogM*en4IL((&9@@97#-EU1mlUJ kkx!Bvs-n&vCge*|Vblc{Q%DUhWuMcW% zC}I<$@9TYAXQg$1vBIT|cu9RSG~qF83vQF^a-;4U7*M$%D)TvaGem5YQH1=S$VscR z+dnQYj=fhODYgGjgz|l0Ks842zvxHp-+S6Ch;d8=NLM@TFW$1=t)=h$OcbbF^5d~I zR`>CE2*-Ykb%o4}%*Say4~T7ha1AjEEUL8sK%pQ1mQOmplu3_1L&Q<2SgvwafoE0j zZE*8zj@e#`p*YX!3x7=CN8k%wZq-c5S2N$I^_hxPo}PGa?)%})A7(*_K}^C?oBtK` z#&wU~i#ocez?RkdxhB2zj>dB1)64a`D41TgOX;c+51HpuYCBhPq-(EhY0`dIT5VEe zCi2k_pPAsln(^ zXTP>Uex{KjrABfA2I%(FcNpGkJ4+E#abgaNt`bI9Bd& zX72*pUM*Gnfl(_m4)|wr^J-X(eGAht#a3h^U}4dfN;|qILD}L+H7Pnd)3oss*cfI~qJ|YjmK=VrTDJyuKtNLCFqfYhb zfY(sVPrHOJ$KRjLjG9d(`8bw4I+4{zVC;JsQ_p_f6A`rK~2B8+EM4hGV z+VrC2>;2oWB3ib4bj(bdqD~g$fb@Bp5~}AeAKG6*EVuaiK@}c0@u*$G B8=8vgy$I_#LBjv!KmBfr3O0(bII@b46VH5=B-_-Y0R4{9$1?(1alQh_W) z#DJ>9UgE9NfRT_6`;J>}v*&5q)>lfCKe91Rp^Qabd^k=5w^#P!Hy#a|od>1ZRZMMs zRYJI&e=JkY(cY5H057`U`!Xxs?Lp(lIxRTq@DJpHIN0B3m=Shym+n`pJ7IgYs;px` z$@BYIH0w)s^&bu@cNyB2IZEz=2cZlTL+Y%-Lb_A(E1o`Dwb#${73-(8SnJsR`|jFA zt n*z~Bp7LZzRC*!78ZO5Yn_;UC;NP&+GpftUX0gK zcNW>4oCIKF-ytLYp1 uckhs8mNZ4C{l3b$06mSF01;{4 z@8Ki~0x>Y#%8#z4t}(Qz>$^fe7SUgnDsAq7UR8yjYI}4v-oPQxm%yGi(rmxBniK1m z$U}!`u!6idMrx|EvcCu3_YbSo6)HC+C&QL*7pqm 6bPEh6PAoZKirn=dW@&bj3hcNmGU wXk3>a(1|_h$T5yyhwp3cvCQ?tz(go4?^7JB+?`KUNS>$Vza0_@g`__OL=^z zv_T8b)FUcyv8JrgJ1t}QcdQ^HI6%|>a{K+Hnzw$7(P8?6x %as4h^XxAf5}AiK{s xxqtfch&q-X2{y=8Y`JPh2aIoM?pDAL6EBr0m*@|g(yJXhs;rx$M{5`nv#Ezx* z2ob1Zb>C_JgX)+yemId-vK6=(UBjP?5~5jlOCb_$Bn1W%A%+SD$_r!Q!~VaZ;}rv? z^XOpwykV*DY@G5?Y2?HDy{AEk)`DV8OTLsZ4fky+68?FubDOVvx-rmyrxXr~lEI%W zcq>&Qv%eT4dDdcmXR=T%mmF!>j6z)t2%}<=Fkxjx3>Ke#i@~ITe<;phiVp4tR{)Oa zY50Itn24MZc;T-etfFH(F2^x`i;K0^|4Q79Yq2p}=L-++tQ=Ap!4p@8``BzMtDukr zV>G7XI0yXv3wx62ifm~XpQ0^Gh$3v)a`&lfqKVA{+B&|V&YAf4!K(-B4>(1w`Mf65 za=VP1u6w@kC|T_3*7vyr`oRo^FYh4v{;gVHIfT(yc$WbjUc#zwL2 $;AC1_P=3XFvOw<|6vN?QKZpBG0dDHG0@~Kkk^}kQRe*PH zS&h(pzf}-CUtenZGI2Y`yt~Kn;UI`3*T4qxZfR*0wd76lKU*5(`)q3BIet}?VofjQ z<3hKkiR80$lyRFmX^{8AIjZ2VF620Scf(dd;9+pE3jqPG*P=?mMTEBxQJ6pFE962E z3M%bjt&gkBx8cR~zWfisMZQ*Q)~YB4!R-qceJMInJ4?|vLMaci+)RTe=w1}E98cDh z@j%(Y2=qO-(>K$QDHAj4VBOK_r6VTp9u{ks6Jup9xGuv5gAOCkeu>zuMy2B zwhd@|^ld2|f!Nfw>zf}nfx*`Y@HhIS{cD99b)wzhV@^(8Y> ZZHQ2rP^piK=kHDB7C%qTF&uJZ^ef!6y=hEXLc5U?}r5&thADQ$hc AR5Hw 8^A(U`#L|JffV7_W@_qWZ1WntE!q zSm|IR&0(|VXw(9X8_5=gTDbM>4bkt9c$RF>Ty76@W8W6 vy>9nN4*}{hx1zZ3?<*@J5C4RscqRQc|Wl9X;Im!-zF~ zbbU}o8_l2*D8rplrMx?Jyj+X_(*ocf^QWe=i5@i|OnTfF?9epIo=MU>-l4z0?FokG zqi~A_OE3ykFdXs(pHQW!wNtyF32##4oVtk>&W>bArun%Z7v8-sm=us;B91_oS<#gm z+(=hSfsTJLRHY=Z&%QUP?!_kK0 m*zS9bQ-K78Acv^#?JbyxjDU`kE+`;j+11aVnvGLbXR~U2!lKwVHK+=ojXk1a zzzkZ5`&PB^!i_qMbj8N6icMlLS;5v0JPm3M&0VgDICpuA(*+-n`7WIR7t-N%fUZ~4 zYIgpdynHdV5x#W&=X%oz4rxI3+w>m+0SRT!n+OoR1Dd%aCiap=I+u)AC|w nthGTISCxig?LvCE zrI)b3hPK7PV0yCU&rSXW8I7GtPmG5aO63SLm=oO+eGc8%j8ZJ=b9wKXW8)go(b6@C z(B22MVdC3gEj?UNo35;P;Aoy*aiU+2cbTat#*uOSVk?J`_9qI)n=HY~sH3*_pg3&9 ztL}ZdgxWYxlSa@uUHuNeq!GlA5%S8h@w=cg9x1plWu7fUMd#r~{|>}!Pu}FBoNJIi z+BvHcnEZA~b3>+6H<9AHaD`uMsMVLbEZNAmPkL&|>(>MRjy!q6qA*#8>una)cYdgN z-Y);}S%wjkRx?XCpe;)|8$)RCMc~s06T_dWsZDy50+cT#k02HLqL7(C_2Jq5EJ*;z z%r6C&Y(1@`4-BJvEjSsL)JfwkXE!|Gg)H|2I}tuyC@2sx=|ZE-CG~ga2}EC}9$|^a zb_R}GY9}E)wdR3$DoABk18kVUfZ*iZudV9+-aM3hq8gcYKQ`{|y?P}kRZB}f;5oyu z+gUW}n!FuCekj!BypjE~)S-L#_w0v>A5NhfJTmhm>h~Fo2tjjPxR&24=!og;5cx&( zx&e$y$PDvR;v_)Q5@s2~TRCoh>6~ri28hi>)U-b=5r`acR*8OduRb~MK4a9wHxW|4 zP@_TK?M|sO4PPqnY_CubH6B~BexO&t7hyejE@omXZ+0cV?UU{}pGGOrEae$;dc-6R zhiXSlypo=C1i&ZgF%X8odd7Me$c+v|k0h_k#$Ph^5OC@rzjsil4lLq2g7F!<57Y3S z5NM%eegG$l0|R=xlUU5vJ7l_fZ4E0wz`5F(tn+<$x1PSpQ1wl*ahGpxb;-3GfMZY( z+LlYkfpeYnD9ae$Nin-2N!Vy<`I=mk!^*0iYVk8_{~`zUIP7w^7RN#^sF@oM7P6>< z(H$gFa8EDIDBV!pl}xI7_(qmQKZu102#0NQt9nwgzx8NmGJbt!wc?J`wFRkxZA>Xa zegQd5&|kVu-TA9~=yan+pZMLom#fP&;*BPnB%(1ij8O#^8lDqKq;g?&dFQ^T+Sn&< zIb#hzj^U2&QRNLUN30C{E|e+k0!=w`n^!kOq1+#LK2cC&F*D;k+^tNS>UfCga{=Jl zmbJ5*%^gaaZM*ZWb1p^iHDgNlw2>qTL)P92bQt9*Wz~jA!-H}IEJ@dY6;~J}dC;lr zbX9$vo24Ni5gZF6`SVfE@IgV!OGK5Eo}=@865m_r1C}TfZC#M~J`vZ(GkvHvPJ_s9 zsj#Q&5iV rXRwZxI0njO;C1w zX>?GS%n6ByhPZa^&WE`~kq@?75B_Dzt71=hr&GM)8m{ d)pf0g~E@=6F_ z4HIELj6YdP a*$R5Ji#IJ5nF!Th~0D%}d^V z{mK}cI+w@7flAzUJB&g&%s}IW*kq*K9T|*CV}7;hz}6MMVk~DAU%ouZO40cvh2~NC z7Qe9Y&r)m12{S20LU={G8$A#8pkV>b4YU8_)y?|)E8fR3-t9ksB58Qo5q7=l=^h6g zT83s>Q_jx2ON^P3K~+V2=qP58jxlOAmkhC*0?29eS0Q%1z!KSNQif8n&6cL;m2pLy zwWHSAeHORX!w2|0RjKd(bSDC=Eot)KW?*==Kop*xcOPR4`rx;d|Mh$8PgXjf?XX`w zI-Ib_ZPqWuW2Zyxlxj~q9}YYSOQMQTqH;_RPiyZntlxb&m^H=;rEEG=dFA7@e?lwe zkcMF~6thyqo6c{j+yLy>6v|MvI+~mN)A?^B3!8$V!_A8ce&UU)urHc8&`OYV=XR#U zgwwbs$-iprX_0?}btfu6e*D-95A%4Rk?eJM*pYl2Mv{>XoNh$N#qa72JM#v?7yYzw z3LNi~iV!2Uk#@t8K$z7M7bt)xHt7mvs9~oq3V6Jd*#W=ku{m(J`Y}U=c%xxj|IfTC z9K~F1#+_)?0+cXO3%GLoINJMUC8K?LR|qa=wq`sK*=eJx-t^o|&CCq!fpKXRd^V5I z-Ne7$U-fj>+MkRvYDkW6Iuc1&YLudH7 urf@yjn>tr|m zuj^N&!_}9xG9X5?7|z2I*WUC9AO7CPj8OTmC-E^iH`ik)B6$II0q?O#(&=X1#nZxJ zM=exCyUqLu-oEMS9;l^W_`^h=XX223ES1z&_4lt Xbqu@<_z0S*4$?97#?;lsiXH?7Tr=4uYJP%BvY%z|fT zX}OqJwc#dKcsQhCjiC!ecTo3Hf2*pBM_sU7z1JtLKFExS$(?d8Ll%-kPk(R!{P)c~ z7E{0tzj}sMHph6B5qOY> ;ofohTO1$np#Y*v-^r zWszV6Y#g6*>2{-F>-VH-yptYOTYTi7Bni%3C(gQ (z zx4`A=TSA-aaLSE-kv3fheJnnwoM%FzC{rN$^-C90B{uW@JlG3Iu)(}33aw1>mL1d3 zt{-*+!@p`yuk(L46_YkcHtO+m 8u^4@ri2%i!C z*Lu;08b8t=!Ek)kDk{eoGmpADT3&d6fUa(eCQPWY6)yPUB?@B~hyTj|x0-)cc5yIm zH>~Bnjhf!GK3NrW? Oc{`td*8iBLt=T(0LmudkDKtgb@_*vM+ zw&>JH6=b7gHpcqsZgFjumKKhur+OelIyt%h@tr$LOwjjy37%fvTO}=ZTx?^ZCF|r> z%y~B2r)Jm_j%Kyqvm?(VM^ISuMSMZMC_X;Wayk_7`bMFFu9^~+XCz# XG^>fh=XGiryaFWU)LaH7J3f6yS1Dr#S!86;=6W8ythdxL_A{iJ^z}Y z!OX3HMs Wmae;jSg;zu7~t2j!dQk`%R*hF9l#VW!&JTJp}75 zK<~K}T|WxbD?%FPWNHHi%M#P7R(A5=6NXEhl$GF7-fa|i!ah9OKKURo^knrNb&k9& zA)f1v+Yh#;9QtL;MTa?;VDo2S7}7<(`_wL#0*ULL(=p5m!$-$h)ThgEAr;B1o3*6u z)mvAa^}}_}2Xz+vlmG4cjtdG39ygNI)>~g3SnvKMkl$K?fk`Re#CWD5P5$N-0!s4( z4`QG-=Y0Pq0`KpY_CAsYJJKE$bRxJl@DMt3^Gc9EO-$sY1CQzHmlX+#4J=kzw(8%q zo{jmLTB!^Ar*T*yhC?QMpBSzwgRXIcfT6yqK=A>7*))|C7oOVO-7$li$xScb%SlN$ zM7GdCmBwaHetNiB_x lhqv zOgcC^YV|rH!f*0U*ffa?AJm=2T-eR_w2+FN^oD5 Bn zy|!&@s{m5*`=Ezt!i|YbKIFG;OiOM0TBp;jrA99lb`RiaOhZvY-gcl2qzuV1p;hav zT^!N!_5#B)wfL?3mK&qIZGScBM~7aa7}6}fmA}n)S`&pFIjSKYpcBgIM>? %{n^(S;|fLV{qX0oI&J1NIeGcIzfL&R zBEMiXOH^6S1^dMgyRGukVtGFG)8UDUGrzZvh?u#O=@kb7&)HINv#vYGHzB(ZM22Wp zeL|+1b=@OrE&W?|9#Tg<5qOJYoc_B~01xjxGWDxbc<1ie@~7*09Z@dlB5P99?2+d4 zQqvN6x*GDn-9 rlMJ7HwaSVc9!T}zmZAy4n{bJ zLlM*@>1$ek*hu~3kzO{qi@SkT%-r03 QCK7K+Gx`AeA|S2yJB?B(^(k~*Si`M5XCYp?WBBDMEb2~GYr-HN*@ z%fY%$O~D|1dIj}u7!JyQt`Ownv41Ag9#zb;79@9F`vHK=tXEr}2@V1SEOthF)(k4a zQ_^ qxHRkC(n(Qy`~m z32cqK7%Ka7^2;RPVBeSWZwgFguv}x3t`JZb(^^?xj?)UF+m7cE7M8uk@`1A #Vv*40o}8(W9l+++u<>qvRDBkqA_4y}hWs_0X8{8~RpB!Qy@N<$&jQ zy)MvUNQ_SrYhhIMaiLa)cdMyT?rQ2>^7B-+XmPt`x%DMuCV!k+lVZcoT5@(ERltZi zRMMVSc%8cm^;B&g4l{Mv)YQ~%S;*c&hY@C1uHU$k|15zSKdcpvk#MTERjESr`j=fw zzqtvI-7X=(V%F~7BFQ@2CNTano;|1h*Yd7(dH{Q;JlEZ{ii$W*ww_66lGvmJJ{#R| zrH+)0)>MUU034lfEXPSi<~^q~l(T(5HIY-d<_>NJi-Sm)*TcJGXL~GAcY1g^OjD%Q zmTet(wfJ$~dm}Zz(07}sD(69M8%p8n&rwC4_2k#aTIPXS;~i}dk%DEizwvRGPnGg& z(~9zGt@Ui~Y}TVKd7mii^G-Zo%+^UeR!mm!!@>Z!bJJ($DY;B|XW`hYRHo8uV~=}g zp43}sH@6ek7`!^=@)}j0qehZEc=H|gQ(v }$L1_$#XsA9C1;154haBUPo`xT= zvzCEOsh|LnEh)}SZ4<@GHxNo*O12V^P`WmPyxS7m8n`hb2@H!_Yv@}`Mc35_pEkWO zg6afRkzR$_c813&^kaQ=ii+Hn!;Z)K7yEVEAG&nxtDHBN%ymxh;))#ILt(!&ztf;* z1UnQdwp|ir<9U1|Ul=-ODq&Cma|+e(eaN%aH@*40w~$snAXpP$Gy-kf-;5}`MtB{x zS%{b%Ry6|^o# &cuhh3{j_*8G?rk`S;JuiQbn5ETD** z+w^iC=|$2S-p>4ZHcS0)HXbvzv)%gNk~MxHa-_ MNtF+P|(*Q9x-F zkklg~(v74Ng3_RbARQvz4I qQ?XY#u4%tn&nQ*euv;G z2W8BO;tBk#^UT!Edz0UL(|>jZ#7#rCP1L Ohve z|ADF?mvXxWq_|yiYN?#;&~&}_m4MVi 5H^LhRv44EX(*6iM~yj89B6`5EkO z8riYGOHF3TfQ)E|$9GtqU~Fm5v=oS9R&mgK`84GfcsSM@ysO%z{vf)|1g@~qSIA(q z*kAn2ndCYXd!?ak7g!i;*TWRIV<)wI=<0HB@o3A+UdhwIeW}g1>wn(^RUNoU0Ul-{ z;CpL7_EHv%ecR*k?te6^vU9Z qR2?`ToTiQw(a^Y1{-bTe~n5PObetHv`Yed*el#_8{wSQwwAHT z-6tqrJp~qw!ZcV5Ne>I3vQ9~EgpiLmiMS9>mepHT5T#Wo%_w?)6aC3=1Pv?Z{d;h? zLcuJC@E?+&w@JhLEuwE|^w$ahp5O8_{447y#q1)s=0h-6# $W}UoIg{VQqbCoWgn>=AF)B-N%(_DU zyH4_ELxKd!7sD9m(Ay~Mb~~z1LQYB|HW+{kvos|i^%?SbE*XBz2$xd3#*1+kO%EvT zbdP!l;P&qx`6nlnqXV6kjP^Gm9;Up)IiLBDw+`fyLm|+;w6(n-w;xZF$ %wD-gjMVq` z_V)S>glPJyCb8^dWAK6DxKw0y+`o1mmcIShel+SWVtxhemW#8=R9Z2-YKF`7iaf9f zwt_Lp6BUT;-Tl APRum7}T`3-RqC}iUM^=v$Qyqdc4Q4y`Jx&ip 0PJQk>k|C(}kRq;7CTrji20| z&l?h(@2F;6r}6wQm-n2Ua&vyses1Y8G|neSa+ib&R5QjMocv)xM3V0Da?71^HxcAW z^ZafE4%f@b?fo5wQo}{fx60R&C9~gM3nCoM^BwTHPy#t?QVE)mha;|OFIZ8`8WMMI z!|Mfo@P(IITE!dF;IjxI;i$Lwv7cp}X_9AjLxa@L@}RW4yHKadE>)-4smCo;P&4rk zC?{rSJi=+6gPQd9KI+dVN@i1c-fEs+$Z%%8!W@f>aAh)c6GTY~B6r>5nB$qg+!Y
!*eY{+wW7{g uHK-Sdk`Q>u$uy7?@yZO zw*M>wG-6b&)m{k;q(jLnC68_XQ3dy7WT65cB^7u@Vn#;Q{KtTw@dY4I6;yeWz1Xg= zY2#9sFmLc`R$*sz7+vV%GvFAc%~W1Z3q9WI)Wak3 E zR!%QtAZ;Dxin*ZonvkI2K%fjNzaG;EMqKYe^k7sd`Cizc`7GlvZaz@MejhdJA$Ulk zG}c}E&=5GHzaYro2n(Dc09E5#3vC?A60is?KIM`Fo!Ojka;~f9J~D;T$N5S-{H=d> zP^Jv@Yl5?BudBVb*?=Dt#re?=A`cYWrOxv{Ax6c#6Om>t%-FZ~7tq|g#fG6(`~piy z4Ie5PLkM<=%mK-G^zC`Lz6lrkimt_K2HFe!Zmjnp* mZz{D@$CiINi`ak;M?*E$vj!=;3Hw zLja4P?d|8*)+4-+ pg=_S#Xrjp SqQ_9ieZ_-v{-HF8h$(u)Iy8J z6=`lu D#a|d3pb)3 zt*_M)RFCJ;9?wa;F6A|0?CG8^dUFa0^dXN16PKWzN?2oZ%Pisrn4K8(Jq^6c9JEUE zv%N=s^=b6MDohe-GYr0xr $|^aNxZ z7rdkOyc+u9V<3w&o|YiLfIxG7J*J}AF;7m##$``WPXr17l9jcL7hz8o0L_Ev83w2h zDiVro%5OjW{7(xYEh( ;mp`~e6#kJ&;e=!;lN7`t^} zQ&EmUm#QJ%LSe?4U?#~j) `;v~f0_jaf>A%r3k8DQDm-j_cJm#L>ua!lM9yZc zI6JV6t NA5d%!tBW9uV=ULb^|i_DCJdYVB;`!`@J{N zf!bHH&0VI@#9#!Lrr^4@CE(y_=I56wSMw7WoRBcjIjO4l*}{vjcKiQ?(kA%w&K!Rd zW@~)4_Wo1)V_ntc17~v3J6Mj2!t={~{;pv*@z#FTt{ $b zW7^1$D?*sghcs`j9c_kyX(^oQdIQop{tRAU`0yj)`zb2corEF!QrkLY0Nj5827l2_ zB)qf84kEtgv(2Tu`DpK=X{vHJdmtJ$`7%LPJ0;>lzPe?W%W}pQmU06(Xoo|gGM4rD zjy5 g-pYPaCTD$c7wypJi$mTpCns(?$1uR5@cp_xE4IxToVHiidx3S6A)pjT<0;jIVww zbqw{kWOl+=!@iM($;U;0T^5`BKIWvQ4Fb+qA`*X;(dttM0*in3gQY4+7{NsC_G-Vx zfwLDDbLz=oRNH;{?A;M}b`0H zy0cy1Bk3bTm)&RNMn0Bl@D&^?VxpviO?ehKd> jF_}iKtR{{^*pO{V3UI65-+mG{V?Fkl&UMdBm?__qg3rN!#4lklE`+R$(s! z?y2#JMfyC&n=Ct{dx Q(l!DD4^|Mm6>1x}F3Pr{xP!uXeF7u7RW-I4~zq z7$OP`e4Wbt#Ms?A{M)Pz-ERexn>EGwRwf2ItnVb*{G${mEe2l00;Qyu+`XUNz<=pC z5u !4z$m -A!2K z5e*9^(uvq_qGS<~K?pk=*WGy-Zo$QW5ZC~+?kLvVq&P58jWkQ*JQN6#NaULEl_a;4 zQQG`dW3vVG?<-6?0zFNeX@B*;6W*Vq)(n>bq0>jTIuX~m1M8EIb-nIF$l2gSh4szN z|4!SF6x%0B^xV-zxI<_s*hF{oBD%n=iszu?LX1`A_TLAR#OO=5&tQ-+T2q4~G|Y8t z`>uln&(sjQ`7N;#7a=A{^vbhX&ybuAQmQLOPVL;$vi5>8sSEutrAlYksw_zaqSfev zE2t5{+)O*PhkdN9EMBYbl7F-5yP- <7F#;)nY&^Bh)h4wl?cNnWm18 zOLFT{COB}jt(P?Xc&WzOdWR4biU$~H#9&UTVTG_O8|A0}(KGS=q@6bJ!6qHhCF%G+1h6cTD5r7?*eakQV#jOx)7>q~>qsEbR8sMG=`QrN32n)gb; zIV{)F98=o9$LQKxfoA9|Xn4V+$!#(OH0F+^ppN2m{>iPbU<08Yz>O;V=P!dZR&CMF z#KZX)nbAL}X&9QPZ;PFjkj@6NaN=p!#I;RR JJPc6+;^ zDP4xIm APu|ZD1Gu=*2YXkorcL-*Pp_~R5ZgN0~Y{c zkc5HCvHehEhfiqSsrhCY94327f-Ho z>n&PjhQT07qkb6e>FY``btwwF_1n^7m1%9vN8ZC>)~xKqHzaDmU3$ELckIFpRJ($= z7`o8B4y@?DO8EwA`+t@;r@E_xFjG(QimW_m$AeJJL1Ns#IOhRDO(FRJKS!I2;7Wc0 zV}P{ {?ymY{T~BoJ 1Y*N}$^R4%f=;1up^4?wp;>KnSL-(LN zW^XG^tQ`u@mYm~(oM$DcRfj8v`ufRToKQ70kzS}e9KPId@t-j2@gqRrMPA8W0t2L1 z_Yf)>ZOn3g8+i4GB3x-M%b!t`!VWGkxm`W0Ln;#B_-V9EQ68*+@~S{3o6Cgw_b!cA zCq;Nn-GUqhh0Doy^+_+rkoW1a2T+mLMpQ(l;t-@=K0)6q0HhHJj@v1qhUvNWK8X>g ze3q|&QqFwNgN);#>@LE%U4n?FKa9IYn;B|3`$&gdO 3y<^;W5LVl&tYbQw5!JJ8rwb+!-y`Xlm7Ax@i*NqMdc!Xq%%- zkwZvJx6)iwgK2vX{Bkszs0MQ@g%kmz+>^V)or0+tmO1;&K^khe=r1zP{OBstBalMx zyS5UDWx}s=^9A%pTB8S~NLjMxOFNaGmT2(&IrgB}VodOZ&u3*W0;DPuVtO1e>RLma zTzdXK1^`R?`Zx7JVUW?5TCN4R;)nxI{Z$drCBJ>>61U0=p;9x^qAj #WFdH(7Qf^fL=>7@89A>$&Z*&D9glh<&RCBhX}D|dA;s*a}Ft1b0+XOH`nj;~_? z Q#Q*e~56 zcr|HdvdOdz`Mj<8)h2UoGG`;q-KmG)#P%OTQ>d2WkE`i^vOjySuf%pzN-QlWhv22> z;SMkAZJ>6lT;ps5Sq4BkS1+Fva()o>;K@)_PXhqEWS8IjU9_Rflti7K!|ofT{_P;8 ziq>aIV&EWU58EB14AC8Y)5V2_?sd(g&2c^w*eSJV5j_>|7(kYPHugD=?jp-ScVmT| zowDmZ{p09*ZAsD`PtD>>?pu~vY704Pf6;t%@p%`<{jN*%zGY^7UjltGFP^{kJcOJH zrM1Bj^odRdRj3sGshQYVnZj--s$jI#es`Ir`qY=mOpt;8Z9#l&X%g8-Myibt=`oQQ zd$C>js_Nrk->Cnh;>;v2k7%$wNLKnW)b!k|RD0y5B+5=qyVqvNBFLc2)UF$NU&7_( zr|mbwnzXbr4LPRs!737JPr`sfb=i@5ZSG5Y6dt1!h5=--oL~qwG>o?|yi$(|h7Qw= zNneiF^nWS6Zcgoga}AFcrq|Dv*iE-SJB FF{Zbw^fKR6+v3>o(Qh z{F_NQK^Cqj-NaDrkbmArnn=952|X>J<8-cG`*yk6w-tAPH?&Q f z-ci-CLX3_kV$-~O$D9(7g_Zl2R$#Wx<@c&{>w8m`=S`J#Jl9gf{*4MaPr%-%&Od{A zdZ|$C$;>@-bg4K=WIt*4&%({4W}T=CRB=gOM6Ya`t*I#!9Ho_?a=-E zT){?Nz;WTKs;cVIAI0OVMV6Z&QNwJq#n>B~6>XxluBiF*=TE!X?o^~A?A76e13m=H zw7$J<*~29eq9e9h`>4h( J zcIca#`C{jmMf6VD{dpj5y)tRJKdsyNi0&1 v8S069OO lQ_eCcEPXOHr!3yy{y67Zoh~#kTfYaI znDcA+99VKdB6fw0?H5SY5V;Yug9^wEkr#y&0r=DthO~#|2)nr#o3wxZQ`g*s2Sn&{ zrHt!|uBmdOtK341wqy6+8>#q3^RQX#!(frVHd*=STXP6`mVa?cK@H=*hFR&fI-Z!L zTx10bqmM67HSMc*RVcSGZZcsZY!dC07Nc bZ>7b6%k4Pcq<4K@b0|^6xfPi4#+degTH}7nsmL6esuwNwdu)5FoqsT z!{!%1AFY)4+1o#6e}?+*EjE69=nGH%mKl$tBnC`ZfCc*d_cmX9@jN~CS!9bpJe=Gh zCH(=`l)+~&Fq$@BNON3iFw7B;wSs%JL)Os!$!c%~W4<}rxQM$ABg5K;YpZF8-kqqA zhCxd*lw387hoZlK|CUo! ;{xj>IKYORT-QS<$H+o}A9?K{4^2<=a#nN&X=MLubupeM z&FLumBD_gO@>BIVFQ5-N_N)MJ*rg_K;W;CeP-|J}oNHneP=u$k7qIZx5Iozn?b*wk zn)@fm)u+4g?bp45nQfTkkwX6f=R91zyP*+6C2SB>zJIt+kyu@V9qwyOm1T y*_s zXIs{Q+0+{>qt`a#z7`_dDWc90xvr_bCE&y&C!B4kP7#;hvsTdGGyiPx>}^FN-r$eK zXs$$`#ZtQHLc#j0d3a(21y^mj&|`jO ~-t1?S#a@fWB{t*dbq@!61zb1iPLW4g z*a&Z$QRw*5{^f>=g%|!sd~qV~4wPkq#U+7&j+IsjR%Qe_AsmtHp(-f3V_WzQwfxl? zor&^2y^KcfesxKn-<3FYf$^sOSucFqIxm!YQEJBR$d)zt>Roq?BSew9|F^hmyJ~s` zFWT7lV9N|?x4cewy=h+}HJ!1ott~4A9-6IPq0sQ`li5T6krC|YvhSN;)M$aJr|u$x z?oK6ce<$r5R51{YF{*kqHoY#Q9lMGx7%j7A5!c)jIaVf6S64zp@<+#=b?z}rsP@C< zksPn>3=H6%lg1EKb!ER(E4qQKdQZx0$JCqjWdN<03-YmS)0?8O9{ypPNVi$UXX#t@ z26vpLj GPMf(#Je^`c2R6)qzHq*AG18_DL{kypPs-MZyC;I1F)T zJEEAPa19N?L7%VOX+D~q2m_* #l6-%zfHH`0^A8x zZt-X~H|IDkXEVTsQo_!6ae|#aM*)yod=KRP;1%jt{#cOSs=*+YN;P?;L=ee*K&VOQ zvByCdDd6i1vlmeXrZv0Bce8}=);_z5!INX#%Cs@Un#)h4D9QJLZ*hGhgf}|cBRWPT zoCc35?Ay?_b8Fr6U}pP>9$d#9`apdgn2EkfwcPwW`YU(-(4$nU>GT?jpEKL{^LhB% z@&gkE&Q(Ql^StK1`*2!l6h|<+xu}qBRVT|)KwtQiNC1kmm0Z9%TRd+Ltc!m2oBbZ9 zz6}3Q^f)Vv-nS>Av-*sY_Yp=DO #6DO;YUTp(?5HBS)z#VOOxQdgHa(s zJvvQnb9B>o2FwLMf4<*D+#k)8NY{M>Hgz@rdDqq)=xbcl_4({6F-5^j90;j4 mc`OQQ@ ^lcyyB#3!Mo1HhBlM4lK`HiW*E%O zH0+g+jcvOOjT>g?x^mUH1BYr8lCVabU{(k5XOxuWyVuMYKi5HwSBlG0l&2E3p>e9e z!8_c3Cydo&?FPK~9lr>@L=~rhUJqToL@jHi1$(se>pZ=AeLMT!OQ@>{X0v6P5SG@s z&>(f&<63gxEX4L&l24{@=WbWREu%xYE}E&yrQ($18Y30Njxh~VG@^;L@%Oixo%vd| zCjKj3vM9=NU!5+dO+ROygF-irKG-6#Eyi3E7B3dDAUxQ?ZWXfyr+7a zuXk1%-3}ZeeiZpRnz;=O4E&wcmo!Q+=AwyVd!nharlGkm;L0UV_o)XLRlG5Scnbhb zFgWOoAm%8%8>BmT`zvdE8k{agKh0yQ>o>>R>;E13+JeME7<~^6uwE0;)F;H(zc5R- zI}siqF3%GZLQ?aznUrrczT@Hx|8?7#>} gb#98OQV=#)~{RyW3g+O^&2Md!*mIh zeds`!__ {O(VyT3d9@bl%}?eU6ZzHOg#pSo^G+0V)xwRr~_&wnE 8J`{dEL&-F2EqE3@Q-ZrRs6 zOE*2n?jgT2-SXJAe*F=Tf{vd*A<-ydhf{UU3ZRN7*;(`!978Ad{`cx$Do#tOJ$W1( z>%k4Fnq6_pP$27?N_UBPxf9Vy0hXx;6?@EQj~QIAJaWj=tPk52?}2>oRe0`r?f9x# z<%hU9&If51ExyY&0-~Q?yOA%S_k8om1>3hR%VJS3hP?=@CJ ^ z6#tR?>2dED4VI$uQB^0lw)|Wc^Y+fqb1jjNGoJix#vXrs`Si$DrFt(!!>!UByZEF$ zT8_clBj T^ivu8q!#Omi5 z7aPuE1bYllii(@nSrMvl`=#LZWPj)vt-#l~rrA4degukLJ=%jqbO-w~e}Ck&gmRjD z(P$ +*vph|VI20CIcxgV+O$w?P=+n$ieRn3Y>JBH zXJw4be&-WsFUCJNb)O8QsH?jM_Dkf3bMrXLS!%xFg=|)Ggnt$tJw^!x!!1;cEJ-4o zYRDJt?FIIQt7i3^CQGs@9fbQ$tmIY}N^hAH;Rje(*{%y)H&vUV!wd)cl9gmSKT}W1 z9A3T ;F&03V_!JQe4zTFDP7rmRp3)=_&p6vbm6YBzv cr)>b_6@&4Xoy^Q*`L01 z WDHRW1iF