PK .Aoa,mimetypeapplication/epub+zipPK.AiTunesMetadata.plist] artistName Oracle Corporation book-info cover-image-hash 959961828 cover-image-path OEBPS/dcommon/oracle-logo.jpg package-file-hash 618298512 publisher-unique-id E25609-03 unique-id 575294814 genre Oracle Documentation itemName Oracle® Database Semantic Technologies Developer's Guide, 11g Release 2 (11.2) releaseDate 2012-05-01T06:29:17Z year 2012 PK^)b]PK.AMETA-INF/container.xml PKYuPK.AOEBPS/sem_apis_ref.htm SEM_APIS Package Subprograms

9 SEM_APIS Package Subprograms

The SEM_APIS package contains subprograms (functions and procedures) for working with the Resource Description Framework (RDF) and Web Ontology Language (OWL) in an Oracle database. To use the subprograms in this chapter, you must understand the conceptual and usage information in Chapter 1, "Oracle Database Semantic Technologies Overview" and Chapter 2, "OWL Concepts".

This chapter provides reference information about the subprograms, listed in alphabetical order.


SEM_APIS.ADD_DATATYPE_INDEX

Format

SEM_APIS.ADD_DATATYPE_INDEX(

     datatype IN VARCHAR2,

     tablespace_name IN VARCHAR2 default NULL,

     parallel IN PLS_INTEGER default NULL,

     online IN BOOLEAN default FALSE,

     options IN VARCHAR2 default NULL);

Description

Adds a data type index for the specified data type to the semantic network.

Parameters

datatype

URI of the data type to index.

tablespace_name

Destination tablespace for the index.

parallel

Degree of parallelism to use when building the index.

online

TRUE allows DML operations affecting the index during creation of the index; FALSE (the default) does not allow DML operations affecting the index during creation of the index.

options

String specifying options for index creation using the form OPTION_NAME=option_value. Supported options are SRID, TOLERANCE, and DIMENSIONS, which are all associated with spatial index creation (see Section 1.6.6.2, "Indexing Spatial Data"). The option name keywords are case sensitive and must be specified in uppercase.

Usage Notes

You must have DBA privileges to call this procedure.

For more information about data type indexing, see Section 1.9.

For information about creating a data type index on RDF spatial data, see Section 1.6.6.2.

Examples

The following example creates an index on xsd:string typed literals and plain literals in the MY_TBS tablespace.

EXECUTE SEM_APIS.ADD_DATATYPE_INDEX('http://www.w3.org/2001/XMLSchema#string', tablespace_name=>'MY_TBS', parallel=>4);

SEM_APIS.ADD_SEM_INDEX

Format

SEM_APIS.ADD_SEM_INDEX(

     index_code IN VARCHAR2);

Description

Creates a semantic network index that results in creation of a nonunique B-tree index in UNUSABLE status for each of the existing models and entailments of the semantic network.

Parameters

index_code

Index code string.

Usage Notes

You must have DBA privileges to call this procedure.

For an explanation of semantic network indexes, see Section 1.8.

Examples

The following example creates a semantic network index with the index code string pcsm on the models and entailments of the semantic network.

EXECUTE SEM_APIS.ADD_SEM_INDEX('pscm');

SEM_APIS.ALTER_DATATYPE_INDEX

Format

SEM_APIS.ALTER_DATATYPE_INDEX(

     datatype IN VARCHAR2,

     command IN VARCHAR2,

     tablespace_name IN VARCHAR2 default NULL,

     parallel IN PLS_INTEGER default NULL,

     online IN BOOLEAN default FALSE);

Description

Alters a data type index.

Parameters

datatype

URI of the data type to index.

options

String specifying the command to be performed: REBUILD to rebuild the data type index, or UNUSABLE to marks the data type index as unusable. The value for this parameter is not case-sensitive.

tablespace_name

Destination tablespace for the index.

parallel

Degree of parallelism to use when rebuilding the index.

online

TRUE allows DML operations affecting the index during rebuilding of the index; FALSE (the default) does not allow DML operations affecting the index during rebuilding of the index.

Usage Notes

You must have DBA privileges to call this procedure.

For an explanation of data type indexes, see Section 1.9.

Examples

The following example rebuilds the index on xsd:string typed literals and plain literals in the MY_TBS tablespace.

EXECUTE SEM_APIS.ALTER_DATATYPE_INDEX('http://www.w3.org/2001/XMLSchema#string', command=>'REBUILD', tablespace_name=>'MY_TBS', parallel=>4);

SEM_APIS.ALTER_ENTAILMENT

Format

SEM_APIS.ALTER_ENTAILMENT(

     entailment_name IN VARCHAR2,

     command IN VARCHAR2,

     tablespace_name IN VARCHAR2,

     parallel IN NUMBER(38) DEFAULT NULL);

Description

Alters an entailment (rules index). Currently, the only action supported is to move the entailment to a specified tablespace.

Parameters

entailment_name

Name of the entailment.

command

Must be the string MOVE.

tablespace_name

Name of the destination tablespace.

parallel

Degree of parallelism to be associated with the operation. For more information about parallel execution, see Oracle Database VLDB and Partitioning Guide.

Usage Notes

For an explanation of entailments, see Section 1.3.7.

Examples

The following example moves the entailment named rdfs_rix_family to the tablespace named my_tbs.

EEXECUTE SEM_APIS.ALTER_ENTAILMENT('rdfs_rix_family', 'MOVE',  'my_tbs');

SEM_APIS.ALTER_MODEL

Format

SEM_APIS.ALTER_MODEL(

     model_name IN VARCHAR2,

     command IN VARCHAR2,

     tablespace_name IN VARCHAR2,

     parallel IN NUMBER(38) DEFAULT NULL);

Description

Alters a model. Currently, the only action supported is to move the model to a specified tablespace.

Parameters

model_name

Name of the model.

command

Must be the string MOVE.

tablespace_name

Name of the destination tablespace.

parallel

Degree of parallelism to be associated with the operation. For more information about parallel execution, see Oracle Database VLDB and Partitioning Guide.

Usage Notes

For an explanation of models, see Section 1.2, "Semantic Data Modeling" and Section 1.3, "Semantic Data in the Database".

This procedure is not supported on version-enabled RDF models, as explained in Section 6.5, "Special Considerations When Using Workspace Manager Support for RDF Data".

Examples

The following example moves the model named family to the tablespace named my_tbs.

EEXECUTE SEM_APIS.ALTER_MODEL('family', 'MOVE',  'my_tbs');

SEM_APIS.ALTER_SEM_INDEX_ON_ENTAILMENT

Format

SEM_APIS.ALTER_SEM_INDEX_ON_ENTAILMENT(

     entailment_name IN VARCHAR2,

     index_code IN VARCHAR2,

     command IN VARCHAR2,

     tablespace_name IN VARCHAR2 DEFAULT NULL,

     use_compression IN BOOLEAN DEFAULT NULL,

     parallel IN NUMBER(38) DEFAULT NULL,

     online IN BOOLEAN DEFAULT FALSE);

Description

Alters a semantic network index on an entailment.

Parameters

entailment_name

Name of the entailment.

index_code

Index code string.

command

String value containing one of the following commands: REBUILD rebuilds the semantic network index on the entailment, or UNUSABLE marks as unusable the semantic network index on the entailment. The value for this parameter is not case-sensitive.

tablespace_name

Name of the destination tablespace for the rebuild operation.

use_compression

Specifies whether compression should be used when rebuilding the index.

parallel

Degree of parallelism to be associated with the operation. For more information about parallel execution, see Oracle Database VLDB and Partitioning Guide.

online

TRUE allows DML operations affecting the index during the rebuilding of the index; FALSE (the default) does not allow DML operations affecting the index during the rebuilding of the index.

Usage Notes

For an explanation of semantic network indexes, see Section 1.8.

Examples

The following example rebuilds (and makes usable if it is unusable) the semantic network index on the entailment named rdfs_rix_family.

EXECUTE SEM_APIS.ALTER_SEM_INDEX_ON_ENTAILMENT('rdfs_rix_family', 'pscm', 'rebuild');

SEM_APIS.ALTER_SEM_INDEX_ON_MODEL

Format

SEM_APIS.ALTER_SEM_INDEX_ON_MODEL(

     model_name IN VARCHAR2,

     index_code IN VARCHAR2,

     command IN VARCHAR2,

     tablespace_name IN VARCHAR2 DEFAULT NULL,

     use_compression IN BOOLEAN DEFAULT NULL,

     parallel IN NUMBER(38) DEFAULT NULL,

     online IN BOOLEAN DEFAULT FALSE);

Description

Alters a semantic network index on a model.

Parameters

model_name

Name of the model.

index_code

Index code string.

command

String value containing one of the following commands: REBUILD rebuilds the semantic network index on the model, or UNUSABLE marks as unusable the semantic network index on the model. The value for this parameter is not case-sensitive.

tablespace_name

Name of the destination tablespace for the rebuild operation.

use_compression

Specifies whether compression should be used when rebuilding the index.

parallel

Degree of parallelism to be associated with the operation. For more information about parallel execution, see Oracle Database VLDB and Partitioning Guide.

online

TRUE allows DML operations affecting the index during the rebuilding of the index; FALSE (the default) does not allow DML operations affecting the index during the rebuilding of the index.

Usage Notes

For an explanation of semantic network indexes, see Section 1.8.

Examples

The following example rebuilds (and makes usable if it is unusable) the semantic network index on the model named family.

EXECUTE SEM_APIS.ALTER_SEM_INDEX_ON_MODEL('family', 'pscm', 'rebuild');

SEM_APIS.ANALYZE_ENTAILMENT

Format

SEM_APIS.ANALYZE_ENTAILMENT(

     entailment_name IN VARCHAR2,

     estimate_percent IN NUMBER DEFAULT to_estimate_percent_type (get_param('ESTIMATE_PERCENT')),

     method_opt IN VARCHAR2 DEFAULT get_param('METHOD_OPT'),

     degree IN NUMBER DEFAULT to_degree_type(get_param('DEGREE')),

     cascade IN BOOLEAN DEFAULT to_cascade_type(get_param('CASCADE')),

     no_invalidate IN BOOLEAN DEFAULT to_no_invalidate_type (get_param('NO_INVALIDATE')),

     force IN BOOLEAN DEFAULT FALSE);

Description

Collects statistics for a specified entailment (rules index).

Parameters

entailment_name

Name of the entailment.

estimate_percent

Percentage of rows to estimate in the internal table partition containing information about the entailment (NULL means compute). The valid range is [0.000001,100]. Use the constant DBMS_STATS.AUTO_SAMPLE_SIZE to have Oracle determine the appropriate sample size for good statistics. This is the usual default.

method_opt

Accepts either of the following options, or both in combination, for the internal table partition containing information about the entailment:

  • FOR ALL [INDEXED | HIDDEN] COLUMNS [size_clause]

  • FOR COLUMNS [size clause] column|attribute [size_clause] [,column|attribute [size_clause]...]

size_clause is defined as size_clause := SIZE {integer | REPEAT | AUTO | SKEWONLY}

column is defined as column := column_name | (extension)


- integer : Number of histogram buckets. Must be in the range [1,254].
- REPEAT : Collects histograms only on the columns that already have histograms.
- AUTO : Oracle determines the columns to collect histograms based on data distribution and the workload of the columns.
- SKEWONLY : Oracle determines the columns to collect histograms based on the data distribution of the columns.
- column_name : name of a column
- extension: Can be either a column group in the format of (column_name, column_name [, ...]) or an expression.

The usual default is FOR ALL COLUMNS SIZE AUTO.

degree

Degree of parallelism for the internal table partition containing information about the entailment. The usual default for degree is NULL, which means use the table default value specified by the DEGREE clause in the CREATE TABLE or ALTER TABLE statement. Use the constant DBMS_STATS.DEFAULT_DEGREE to specify the default value based on the initialization parameters. The AUTO_DEGREE value determines the degree of parallelism automatically. This is either 1 (serial execution) or DEFAULT_DEGREE (the system default value based on number of CPUs and initialization parameters) according to size of the object.

cascade

Gathers statistics on the indexes for the internal table partition containing information about the entailment. Use the constant DBMS_STATS.AUTO_CASCADE to have Oracle determine whether index statistics are to be collected or not. This is the usual default.

no_invalidate

Does not invalidate the dependent cursors if set to TRUE. The procedure invalidates the dependent cursors immediately if set to FALSE. Use DBMS_STATS.AUTO_INVALIDATE. to have Oracle decide when to invalidate dependent cursors. This is the usual default.

force

TRUE gathers statistics even if the entailment is locked; FALSE (the default) does not gather statistics if the entailment is locked.

Usage Notes

Index statistics collection can be parellelized except for cluster, domain, and join indexes.

This procedure internally calls the DBMS_STATS.GATHER_TABLE_STATS procedure, which collects statistics for the internal table partition that contains information about the entailment. The DBMS_STATS.GATHER_TABLE_STATS procedure is documented in Oracle Database PL/SQL Packages and Types Reference.

For information about entailments, see Section 1.3.7.

Examples

The following example collects statistics for the entailment named rdfs_rix_family.

EXECUTE SEM_APIS.ANALYZE_ENTAILMENT('rdfs_rix_family');

SEM_APIS.ANALYZE_MODEL

Format

SEM_APIS.ANALYZE_MODEL(

     model_name IN VARCHAR2,

     estimate_percent IN NUMBER DEFAULT to_estimate_percent_type (get_param('ESTIMATE_PERCENT')),

     method_opt IN VARCHAR2 DEFAULT get_param('METHOD_OPT'),

     degree IN NUMBER DEFAULT to_degree_type(get_param('DEGREE')),

     cascade IN BOOLEAN DEFAULT to_cascade_type(get_param('CASCADE')),

     no_invalidate IN BOOLEAN DEFAULT to_no_invalidate_type (get_param('NO_INVALIDATE')),

     force IN BOOLEAN DEFAULT FALSE);

Description

Collects optimizer statistics for a specified model.

Parameters

model_name

Name of the model.

estimate_percent

Percentage of rows to estimate in the internal table partition containing information about the model (NULL means compute). The valid range is [0.000001,100]. Use the constant DBMS_STATS.AUTO_SAMPLE_SIZE to have Oracle determine the appropriate sample size for good statistics. This is the usual default.

method_opt

Accepts either of the following options, or both in combination, for the internal table partition containing information about the model:

  • FOR ALL [INDEXED | HIDDEN] COLUMNS [size_clause]

  • FOR COLUMNS [size clause] column|attribute [size_clause] [,column|attribute [size_clause]...]

size_clause is defined as size_clause := SIZE {integer | REPEAT | AUTO | SKEWONLY}

column is defined as column := column_name | (extension)


- integer : Number of histogram buckets. Must be in the range [1,254].
- REPEAT : Collects histograms only on the columns that already have histograms.
- AUTO : Oracle determines the columns to collect histograms based on data distribution and the workload of the columns.
- SKEWONLY : Oracle determines the columns to collect histograms based on the data distribution of the columns.
- column_name : name of a column
- extension: Can be either a column group in the format of (column_name, column_name [, ...]) or an expression.

The usual default is FOR ALL COLUMNS SIZE AUTO.

degree

Degree of parallelism for the internal table partition containing information about the model. The usual default for degree is NULL, which means use the table default value specified by the DEGREE clause in the CREATE TABLE or ALTER TABLE statement. Use the constant DBMS_STATS.DEFAULT_DEGREE to specify the default value based on the initialization parameters. The AUTO_DEGREE value determines the degree of parallelism automatically. This is either 1 (serial execution) or DEFAULT_DEGREE (the system default value based on number of CPUs and initialization parameters) according to size of the object.

cascade

Gathers statistics on the indexes for the internal table partition containing information about the model. Use the constant DBMS_STATS.AUTO_CASCADE to have Oracle determine whether index statistics are to be collected or not. This is the usual default.

no_invalidate

Does not invalidate the dependent cursors if set to TRUE. The procedure invalidates the dependent cursors immediately if set to FALSE. Use DBMS_STATS.AUTO_INVALIDATE. to have Oracle decide when to invalidate dependent cursors. This is the usual default.

force

TRUE gathers statistics even if the model is locked; FALSE (the default) does not gather statistics if the model is locked.

Usage Notes

Index statistics collection can be parellelized except for cluster, domain, and join indexes.

This procedure internally calls the DBMS_STATS.GATHER_TABLE_STATS procedure, which collects optimizer statistics for the internal table partition that contains information about the model. The DBMS_STATS.GATHER_TABLE_STATS procedure is documented in Oracle Database PL/SQL Packages and Types Reference.

Examples

The following example collects statistics for the semantic model named family.

EXECUTE SEM_APIS.ANALYZE_MODEL('family');

SEM_APIS.BULK_LOAD_FROM_STAGING_TABLE

Format

SEM_APIS.BULK_LOAD_FROM_STAGING_TABLE(

     model_name IN VARCHAR2,

     table_owner IN VARCHAR2,

     table_name IN VARCHAR2,

     flags IN VARCHAR2 DEFAULT NULL,

     debug IN INTEGER DEFAULT NULL,

     start_comment IN VARCHAR2 DEFAULT NULL,

     end_comment IN VARCHAR2 DEFAULT NULL);

Description

Loads semantic data from a staging table.

Parameters

model_name

Name of the model.

table_owner

Name of the schema that owns the staging table that holds semantic data to be loaded.

table_name

Name of the staging table that holds semantic data to be loaded.

flags

An optional quoted string with one or more of the following keyword specifications:

  • DEL_BATCH_DUPS=USE_INSERT allows the use of an insertion-based strategy for duplicate elimination that may lead to faster processing if the input data contains many duplicates.

  • MBV_METHOD=SHADOW allows the use of a different value loading strategy that may lead to faster processing for large loads.

  • PARALLEL_CREATE_INDEX allows internal indexes to be created in parallel, which may improve the performance of the bulk load processing.

  • PARALLEL=<integer> allows much of the processing used during bulk load to be done in parallel using the specified degree of parallelism to be associated with the operation.

  • PARSE allows parsing of triples retrieved from the staging table (also parses triples containing graph names).

  • <task>_JOIN_HINT=<join_type>, where <task> can be any of the following internal tasks performed during bulk load: IZC (is zero collisions), MBV (merge batch values), or MBT (merge batch triples, used when adding triples to a non-empty model), and where <join_type> can be USE_NL and USE_HASH.

debug

(Reserved for future use)

start_comment

Optional comment about the start of the load operation.

end_comment

Optional comment about the end of the load operation.

Usage Notes

You must first load semantic data into a staging table before calling this procedure. See Section 1.7.1 for more information.

Using BULK_LOAD_FROM_STAGING_TABLE with Fine Grained Access Control (VPD and OLS)

When fine-grained access control (explained in Chapter 5) is enabled for a specific model using VPD or for the entire network using OLS, only a user with FULL access privileges to the associated policy may perform the bulk load operation. Full access privileges to a VPD policy are granted by setting the context attribute defined by the package constant sem_rdfsa.VPD_FULL_ACCESS to 1 for the namespace associated with the VPD policy (see Section 5.1.1). When OLS is enabled, full access privileges to the OLS policy are granted using the SA_USER_ADMIN.SET_USER_PRIVS procedure.

The data loaded into a model with VPD enabled is immediately available for access and does not need any maintenance operations. However, when the OLS is used, the label column in the tables storing the RDF triples must be maintained. By default, with OLS enabled, the label column in the tables storing the RDF triples is set to null. If you have FULL access, you can reset the labels for the newly inserted triples as well as any resources introduced by the new batch of triples by using appropriate subprograms (SEM_RDFSA.SET_RESOURCE_LABEL and SEM_RDFSA.SET_PREDICATE_LABEL).

Optionally, you can define a numeric column named RDF$STC_CTXT1 in the staging table and the application table, to assign the sensitivity label of the triple before the data is loaded into the desired model. Such labels are automatically applied to the corresponding triples stored in the MDSYS.RDF_LINK$ table. The labels for the newly introduced resources may still have to be applied separately before or after the load, and the system does not validate the labels assigned during bulk load operation.

The RDF$STC_CTXT1 column in the application table has no significance, and it may be dropped after the bulk load operation.

Examples

The following example loads semantic data stored in the staging table named STAGE_TABLE in schema SCOTT into the semantic model named family. The example includes some join hints.

EXECUTE SEM_APIS.BULK_LOAD_FROM_STAGING_TABLE('family', 'scott', 'stage_table', flags => 'IZC_JOIN_HINT=USE_HASH MBV_JOIN_HINT=USE_HASH');

SEM_APIS.CLEANUP_FAILED

Format

SEM_APIS.CLEANUP_FAILED(

     rdf_object_type IN VARCHAR2,

     rdf_object_name IN VARCHAR2);

Description

Drops (deletes) a specified rulebase or entailment if it is in a failed state.

Parameters

rdf_object_type

Type of the RDF object: RULEBASE for a rulebase or RULES_INDEX for an entailment (rules index).

rdf_object_name

Name of the RDF object of type rdf_object_type.

Usage Notes

This procedure checks to see if the specified RDF object is in a failed state; and if the object is in a failed state, the procedure deletes the object.

A rulebase or entailment is in a failed state if a system failure occurred during the creation of that object. You can check if a rulebase or entailment is in a failed state by checking to see if the value of the STATUS column is FAILED in the SDO_RULEBASE_INFO view (described in Section 1.3.6) or the SDO_RULES_INDEX_INFO view (described in Section 1.3.7), respectively.

If the rulebase or entailment is not in a failed state, this procedure performs no action and returns a successful status.

An exception is generated if the RDF object is currently being used.

Examples

The following example deletes the rulebase named family_rb if (and only if) that rulebase is in a failed state.

EXECUTE SEM_APIS.CLEANUP_FAILED('RULEBASE', 'family_rb');

SEM_APIS.COMPOSE_RDF_TERM

Format

SEM_APIS.COMPOSE_RDF_TERM(

     value_name IN VARCHAR2,

     value_type IN VARCHAR2,

     literal_type IN VARCHAR2,

     language_type IN VARCHAR2

     ) RETURN VARCHAR2;

or

SEM_APIS.COMPOSE_RDF_TERM(

     value_name IN VARCHAR2,

     value_type IN VARCHAR2,

     literal_type IN VARCHAR2,

     language_type IN VARCHAR2,

     long_value IN CLOB,

     options IN VARCHAR2 DEFAULT NULL,

     ) RETURN CLOB;

Description

Creates and returns an RDF term using the specified parameters.

Parameters

value_name

Value name. Must match a value in the VALUE_NAME column in the MDSYS.RDF_VALUE$ table (described in Section 1.3.2) or in the var attribute returned from SEM_MATCH table function.

value_type

The type of text information. Must match a value in the VALUE_TYPE column in the MDSYS.RDF_VALUE$ table (described in Section 1.3.2) or in the var$RDFVTYP attribute returned from SEM_MATCH table function.

literal_type

For typed literals, the type information; otherwise, null. Must either be a null value or match a value in the LITERAL_TYPE column in the MDSYS.RDF_VALUE$ table (described in Section 1.3.2) or in the var$RDFLTYP attribute returned from SEM_MATCH table function.

language_type

Language tag. Must match a value in the LANGUAGE_TYPE column in the MDSYS.RDF_VALUE$ table (described in Section 1.3.2) or in the var$RDFLANG attribute returned from SEM_MATCH table function.

long_value

The character string if the length of the lexical value is greater than 4000 bytes. Must match a value in the LONG_VALUE column in the MDSYS.RDF_VALUE$ table (described in Section 1.3.2) or in the var$RDFCLOB attribute returned from SEM_MATCH table function.

options

(Reserved for future use.)

Usage Notes

If you specify an inconsistent combination of values for the parameters, this function returns a null value. If a null value is returned but you believe that the values for the parameters are appropriate (reflecting columns from the same row in the MDSYS.RDF_VALUE$ table or from a SEM_MATCH query for the same variable), contact Oracle Support.

Examples

The following example returns, for each member of the family whose height is known, the RDF term for the height and also just the value portion of the height.

SELECT x, SEM_APIS.COMPOSE_RDF_TERM(h, h$RDFVTYP, h$RDFLTYP, h$RDFLANG)
       h_rdf_term, h
  FROM TABLE(SEM_MATCH(
    '{?x :height ?h}',
    SEM_Models('family'),
    null, 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null))
ORDER BY x;
X
--------------------------------------------------------------------------------
H_RDF_TERM
--------------------------------------------------------------------------------
H
--------------------------------------------------------------------------------
http://www.example.org/family/Cathy
"5.8"^^<http://www.w3.org/2001/XMLSchema#decimal>
5.8
 
http://www.example.org/family/Cindy
"6"^^<http://www.w3.org/2001/XMLSchema#decimal>
6
 
http://www.example.org/family/Jack
"6"^^<http://www.w3.org/2001/XMLSchema#decimal>
6
 
http://www.example.org/family/Tom
"5.75"^^<http://www.w3.org/2001/XMLSchema#decimal>
5.75
 
4 rows selected.

The following example returns the RDF terms for a few of the values stored in the MDSYS.RDF_VALUE$ table.

SELECT SEM_APIS.COMPOSE_RDF_TERM(value_name, value_type, literal_type,
     language_type)
  FROM MDSYS.RDF_VALUE$ WHERE ROWNUM < 5;

SEM_APIS.COMPOSE_RDF_TERM(VALUE_NAME,VALUE_TYPE,LITERAL_TYPE,LANGUAGE_TYPE)
--------------------------------------------------------------------------------
<http://www.w3.org/1999/02/22-rdf-syntax-ns#object>
<http://www.w3.org/1999/02/22-rdf-syntax-ns#type>
<http://www.w3.org/1999/02/22-rdf-syntax-ns#subject>
<http://www.w3.org/1999/02/22-rdf-syntax-ns#Property>

SEM_APIS.CREATE_ENTAILMENT

Format

SEM_APIS.CREATE_ENTAILMENT(

     entailment_name_in IN VARCHAR2,

     models_in IN SEM_MODELS,

     rulebases_in IN SEM_RULEBASES,

     passes IN NUMBER DEFAULT SEM_APIS.REACH_CLOSURE,

     inf_components_in IN VARCHAR2 DEFAULT NULL,

     options IN VARCHAR2 DEFAULT NULL,

     delta_in IN SEM_MODELS DEFAULT NULL,

     label_gen IN RDFSA_LABELGEN DEFAULT NULL,

     include_named_g IN SEM_GRAPHS DEFAULT NULL,

     include_default_g IN SEM_MODELS DEFAULT NULL,

     include_all_g IN SEM_MODELS DEFAULT NULL,

     inf_ng_name IN VARCHAR2 DEFAULT NULL);

Description

Creates an entailment (rules index) that can be used to perform OWL or RDFS inferencing, and optionally use user-defined rules.

Parameters

entailment_name_in

Name of the entailment to be created.

models_in

One or more model names. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25)

rulebases_in

One or more rulebase names. Its data type is SEM_RULEBASES, which has the following definition: TABLE OF VARCHAR2(25). Rules and rulebases are explained in Section 1.3.6.

passes

The number of rounds that the inference engine should run. The default value is SEM_APIS.REACH_CLOSURE, which means the inference engine will run till a closure is reached. If the number of rounds specified is less than the number of actual rounds needed to reach a closure, the status of the entailment will then be set to INCOMPLETE.

inf_components_in

A comma-delimited string of keywords representing inference components, for performing selective or component-based inferencing. If this parameter is null, the default set of inference components is used. See the Usage Notes for more information about inference components.

options

A comma-delimited string of options to control the inference process by overriding the default inference behavior. To enable an option, specify option-name=T; to disable an option, you can specify option-name=F (the default). The available option-name values are COL_COMPRESS, DISTANCE,DOP, ENTAIL_ANYWAY, HASH_PART, INC, LOCAL_NG_INF, OPT_SAMEAS, RAW8, PROOF, and USER_RULES. See the Usage Notes for explanations of each value.

delta_in

If incremental inference is in effect, specifies one or more models on which to perform incremental inference. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25)

The triples in the first model in delta_in are copied to the first model in models_in, and the entailment (rules index) in rules_index_in is updated; then the triples in the second model (if any) in delta_in are copied to the second model (if any) in models_in, and the entailment in rules_index_in is updated; and so on until all triples are copied and the entailment is updated. (The delta_in parameter has no effect if incremental inference is not enabled for the entailment.)

label_gen

An instance of MDSYS.RDFSA_LABELGEN or a subtype of it, defining the logic for generating Oracle Label Security (OLS) labels for inferred triples. What you specify for this parameter depends on whether you use the default label generator or a custom label generator:

  • If you use the default label generator, specify one of the following constants: SEM_RDFSA.LABELGEN_RULE for Use Rule Label, SEM_RDFSA.LABELGEN_SUBJECT for Use Subject Label, SEM_RDFSA.LABELGEN_PREDICATE for Use Predicate Label, SEM_RDFSA.LABELGEN_OBJECT for Use Object Label, SEM_RDFSA.LABELGEN_DOMINATING for Use Dominating Label, SEM_RDFSA.LABELGEN_ANTECED for Use Antecedent Labels. For a detailed explanation of each constant, see Section 5.2.2.4.

  • If you use a custom label generator, specify the custom label generator type. For information about creating and implementing a custom label generator, see Section 5.2.2.5.

include_named_g

Causes all triples from the specified named graphs (across all source models) to participate in named graph based global inference (NGGI, explained in Section 2.2.11.1). For example, include_named_g => sem_graphs('<urn:G1>','<urn:G2>') implies that triples from named graphs G1 and G2 will be included in NGGI.

Its data type is SEM_GRAPHS, which has the following definition: TABLE OF VARCHAR2(4000).

include_default_g

Causes all triples with a null graph name in the specified models to participate in named graph based global inference (NGGI, explained in Section 2.2.11.1). For example, include_default_g => sem_models('m1') causes all triples with a null graph name from model M1 to be included in NGGI.

include_all_g

Causes all triples, regardless of their graph name values, in the specified models to participate in named graph based global inference (NGGI, explained in Section 2.2.11.1). For example, include_all_g => sem_models('m2')causes all triples in model M2 to be included in NGGI.

inf_ng_name

Assigns the specified graph name to all the new triples inferred by the named graph based global inference (NGGI, explained in Section 2.2.11.1).

Usage Notes

For the inf_components_in parameter, you can specify any combination of the following keywords: SCOH, COMPH, DISJH, SYMMH, INVH, SPIH, MBRH, SPOH, DOMH, RANH, EQCH, EQPH, FPH, IFPH, DOM, RAN, SCO, DISJ, COMP, INV, SPO, FP, IFP, SYMM, TRANS, DIF, SAM, CHAIN, HASKEY, ONEOF, INTERSECT, INTERSECTSCOH, MBRLST, PROPDISJH, SKOSAXIOMS, SNOMED, SVFH, THINGH, THINGSAM, UNION, RDFP1, RDFP2, RDFP3, RDFP4, RDFP6, RDFP7, RDFP8AX, RDFP8BX, RDFP9, RDFP10, RDFP11, RDFP12A, RDFP12B, RDFP12C, RDFP13A, RDFP13B, RDFP13C, RDFP14A, RDFP14BX, RDFP15, RDFP16, RDFS2, RDFS3, RDFS4a, RDFS4b, RDFS5, RDFS6, RDFS7, RDFS8, RDFS9, RDFS10, RDFS11, RDFS12, RDFS13. For an explanation of the meaning of these keywords, see Table 9-1, where the keywords are listed in alphabetical order.

The default set of inference components for the OWLPrime vocabulary includes the following: SCOH, COMPH, DISJH, SYMMH, INVH, SPIH, SPOH, DOMH, RANH, EQCH, EQPH, FPH, IFPH, SAMH, DOM, RAN, SCO, DISJ, COMP, INV, SPO, FP, IFP, SYMM, TRANS, DIF, RDFP14A, RDFP14BX, RDFP15, RDFP16. However, note the following:

Table 9-1 Inferencing Keywords for inf_components_in Parameter

KeywordExplanation

CHAIN

Captures the property chain semantics defined in OWL 2. Only chains of length 2 are supported. By default, this is included in the SKOSCORE rulebase. Subproperty chaining is an OWL 2 feature, and for backward compatibility this component is not by default included in the OWLPrime rulebase. (For information about property chain handling, see Section 3.2.2.) (New as of Release 11.2.)

COMPH

Performs inference based on owl:complementOf assertions and the interaction of owl:complementOf with other language constructs.

DIF

Generates owl:differentFrom assertions based on the symmetricity of owl:differentFrom.

DISJ

Infers owl:differentFrom relationships at instance level using owl:disjointWith assertions.

DISJH

Performs inference based on owl:disjointWith assertions and their interactions with other language constructs.

DOM

Performs inference based on RDFS2.

DOMH

Performs inference based on rdfs:domain assertions and their interactions with other language constructs.

EQCH

Performs inference that are relevant to owl:equivalentClass.

EQPH

Performs inference that are relevant to owl:equivalentProperty.

FP

Performs instance-level inference using instances of owl:FunctionalProperty.

FPH

Performs inference using instances of owl:FunctionalProperty.

HASKEY

Covers the semantics behind "keys" defined in OWL 2. In OWL 2, a collection of properties can be treated as a key to a class expression. For efficiency, the size of the collection must not exceed 3. (New as of Release 11.2.)

IFP

Performs instance-level inference using instances of owl:InverseFunctionalProperty.

IFPH

Performs inference using instances of owl:InverseFunctionalProperty.

INTERSECT

Handles the core semantics of owl:intersectionOf. For example, if class C is the intersection of classes C1, C2 and C3, then C is a subclass of C1, C2, and C3. In addition, common instances of all C1, C2, and C3 are also instances of C. (New as of Release 11.2.)

INTERSECTSCOH

Handles the fact that an intersection is the maximal common subset. For example, if class C is the intersection of classes C1, C2, and C3, then any common subclass of all C1, C2, and C3 is a subclass of C. (New as of Release 11.2.)

INV

Performs instance-level inference using owl:inverseOf assertions.

INVH

Performs inference based on owl:inverseOf assertions and their interactions with other language constructs.

MBRLST

Captures the semantics that for any resource, every item in the list given as the value of the skos:memberList property is also a value of the skos:member property. (See S36 in the SKOS detailed specification.) By default, this is included in the SKOSCORE rulebase. (New as of Release 11.2.)

ONEOF

Generates classification assertions based on the definition of the enumeration classes. In OWL, class extensions can be enumerated explicitly with the owl:oneOf constructor. (New as of Release 11.2.)

PROPDISJH

Captures the interaction between owl:propertyDisjointWith and rdfs:subPropertyOf. By default, this is included in SKOSCORE rulebase. propertyDisjointWith is an OWL 2 feature, and for backward compatibility this component is not by default included in the OWLPrime rulebase. (New as of Release 11.2.)

RANH

Performs inference based on rdfs:range assertions and their interactions with other language constructs.

RDFP*

(The rules corresponding to components with a prefix of RDFP can be found in Completeness, decidability and complexity of entailment for RDF Schema and a semantic extension involving the OWL vocabulary, by H.J. Horst.)

RDFS2, ... RDFS13

RDFS2, RDFS3, RDFS4a, RDFS4b, RDFS5, RDFS6, RDFS7, RDFS8, RDFS9, RDFS10, RDFS11, RDFS12, and RDFS13 are described in Section 7.3 of RDF Semantics (http://www.w3.org/TR/rdf-mt/). Note that many of the RDFS components are not relevant for OWL inference.

SAM

Performs inference about individuals based on existing assertions for those individuals and owl:sameAs.

SAMH

Infers owl:sameAs assertions using transitivity and symmetricity of owl:sameAs.

SCO

Performs inference based on RDFS9.

SCOH

Generates the subClassOf hierarchy based on existing rdfs:subClassOf assertions. Basically, C1 rdfs:subClassOf C2 and C2 rdfs:subClassOf C3 will infer C1 rdfs:subClassOf C3 based on transitivity. SCOH is also an alias of RDFS11.

SKOSAXIOMS

Captures most of the axioms defined in the SKOS detailed specification. By default, this is included in the SKOSCORE rulebase. (New as of Release 11.2.)

SNOMED

Performs inference based on the semantics of the OWL 2 EL profile, which captures the expressiveness of SNOMED CT (Systematized Nomenclature of Medicine - Clinical Terms), which is one of the most expressive and complex medical terminologies. (New as of Release 11.2.)

SPIH

Performs inference based on interactions between rdfs:subPropertyOf and owl:inverseOf assertions.

SPO

Performs inference based on RDFS7.

SPOH

Generates rdfs:subPropertyOf hierarchy based on transitivity of rdfs:subPropertyOf. It is an alias of RDFS5.

SVFH

Handles the following semantics that involves the interaction between owl:someValuesFrom and rdfs:subClassOf. Consider two existential restriction classes C1 and C2 that both use the same restriction property. Assume further that the owl:someValuesFrom constraint class for C1 is a subclass of that for C2. Then C1 can be inferred as a subclass of C2. (New as of Release 11.2.)

SYMM

Performs instance-level inference using instances of owl:SymmetricProperty.

SYMH

Performs inference for properties of type owl:SymmetricProperty.

THINGH

Handles the semantics that any defined OWL class is a subclass of owl:Thing. The consequence of this rule is that instances of all defined OWL classes will become instances of owl:Thing. The size of the inferred graph will very likely be bigger with this component selected. (New as of Release 11.2.)

THINGSAM

Handles the semantics that instances of owl:Thing are equal to (owl:sameAs) themselves. This component is provided for the convenience of some applications. Note that an application does not have to select this inference component to figure out an individual is equal to itself; this kind of information can easily be built in the application logic. (New as of Release 11.2.)

TRANS

Calculates transitive closure for instances of owl:TransitiveProperty.

UNION

Captures the core semantics of the owl:unionOf construct. Basically, the union class is a superclass of all member classes. For backward compatibility, this component is not by default included in the OWLPrime rulebase. (New as of Release 11.2.)


To deselect a component, use the component name followed by a minus (-) sign. For example, SCOH- deselects inference of the subClassOf hierarchy.

For the options parameter, you can enable the following options to override the default inferencing behavior:

For the delta_in parameter, inference performance is best if the value is small compared to the overall size of those models. In a typical scenario, the best results might be achieved when the delta contains fewer than 10,000 triples; however, some tests have shown significant inference performance improvements with deltas as large as 100,000 triples.

For the label_gen parameter, if you want to use the default OLS label generator, specify the appropriate SEM_RDFSA package constant value from Table 9-2.

Table 9-2 SEM_RDFSA Package Constants for label_gen Parameter

ConstantDescription

SEM_RDFSA.LABELGEN_SUBJECT

Label generator that applies the label associated with the inferred triple's subject as the triple's label.

SEM_RDFSA.LABELGEN_PREDICATE

Label generator that applies the label associated with the inferred triple's subject as the triple's label.

SEM_RDFSA.LABELGEN_OBJECT

Label generator that applies the label associated with the inferred triple's subject as the triple's label.

SEM_RDFSA.LABELGEN_RULE

Label generator that applies the label associated with the rule that directly produced the inferred triple as the triple's label. If you specify this option, you must also specify PROOF=T in the options parameter.

SEM_RDFSA.LABELGEN_DOMINATING

Label generator that computes a dominating label of all the available labels for the triple's components (subject, predicate, object, and rule), and applies it as the label for the inferred triple.


Fine-Grained Access Control (VPD and OLS) Considerations

When fine-grained access control is enabled for a specific model using VPD or for the entire network using OLS, only a user with FULL access privileges to the associated policy may create an entailment. Full access privileges to a VPD policy are granted by setting the context attribute defined by the package constant sem_rdfsa.VPD_FULL_ACCESS to 1 for the namespace associated with the VPD policy (see Section 5.1.1, "VPD Policy for RDF Data"). When OLS is enabled, full access privileges to the OLS policy are granted using the SA_USER_ADMIN.SET_USER_PRIVS procedure. VPD policy is automatically enforced for any entailment created using at least one VPD enabled RDF model. As discussed in Section 5.1.1, the RDF metadata associated with the VPD policy may need to be maintained using the newly inferred data to ensure that the predefined VPD policies are not circumvented using inferred data.

Inferred triples accessed through generated labels might not be same as conceptual triples inferred directly from the user accessible triples and rules. The labels generated using a subset of triple components may be weaker than intended. For example, one of the antecedents for the inferred triple may have a higher label than any of the components of the triple. When the label is generated based on just the triple components, end users with no access to one of the antecedents may still have access to the inferred triple. Even when the antecedents are used for custom label generation, the generated label may be stronger than intended. The inference process is not exhaustive, and information pertaining to any alternate ways of inferring the same triple is not available. So, the label generated using a given set of antecedents may be too strong, because the user with access to all the triples in the alternate path could infer the triple with lower access.

Even when generating a label that dominates all its components and antecedents, the label may not be precise. This is the case when labels considered for dominating relationship have non-overlapping group information. For example, consider two labels L:C:NY and L:C:NH where L is a level, C is a component and NY and NH are two groups. A simple label that dominates these two labels is L:C:NY,NH, and a true supremum for the two labels is L:C:US, where US is parent group for both NY and NH. Unfortunately, neither of these two dominating labels is precise for the triple inferred from the triples with first two labels. If L:C:NY,NH is used for the inferred triple, a user with membership in either of these groups has access to the inferred triple, whereas the same user does not have access to one of its antecedents. On the other hand, if L:C:US is used for the inferred triple, a user with membership in both the groups and not in the US group will not be able to access the inferred triple, whereas that user could infer the triple by directly accessing its components and antecedents.

Because of these unique challenges with inferred triples, extra caution must be taken when choosing or implementing the label generator.

See also the OLS example in the Examples section.

Examples

The following example creates an entailment named OWLTST_IDX using the OWLPrime rulebase, and it causes proof to be generated for inferred triples.

EXECUTE sem_apis.create_entailment('owltst_idx', sem_models('owltst'), sem_rulebases('OWLPRIME'), SEM_APIS.REACH_CLOSURE, null, 'PROOF=T');

The following example assumes an OLS environment. It creates a rulebase with a rule, and it creates an entailment.

-- Create an entailment with a rule. -- 
exec sdo_rdf_inference.create_entailment('contracts_rb');
 
insert into mdsys.rdfr_contracts_rb values (
  'projectLedBy', '(?x :drivenBy ?y) (?y :hasVP ?z)',  NULL,
  '(?x :isLedBy ?z)',
  SDO_RDF_Aliases(SDO_RDF_Alias('','http://www.myorg.com/pred/')));
 
-- Assign sensitivity label for the predicate to be inferred. -- 
-- Yhe predicate label may be set globally or it can be assign to --
-- the one or the models used to infer the data – e.g: CONTRACTS. 
begin
  sem_rdfsa.set_predicate_label(
         model_name   => 'rdf$global',
         predicate    => 'http://www.myorg.com/pred/isLedBy',
         label_string => 'TS:US_SPCL');
end;
/
 
-- Create index with a specific label generator. -- 
begin
  sem_apis.create_entailment(
         entailment_name_in => 'contracts_inf',
         models_in          => SDO_RDF_Models('contracts'),
         rulebases_in       => SDO_RDF_Rulebases('contracts_rb'),
         options            => 'USER_RULES=T',
         label_gen          => sem_rdfsa.LABELGEN_PREDICATE);
end;
/
 
-- Check for any label exceptions and update them accordingly. -- 
update mdsys.rdfi_contracts_inf set ctxt1 = 1100 where ctxt1 = -1;
 
-- The new entailment is now ready for use in SEM_MATCH queries. --

SEM_APIS.CREATE_RULEBASE

Format

SEM_APIS.CREATE_RULEBASE(

     rulebase_name IN VARCHAR2);

Description

Creates a rulebase.

Parameters

rulebase_name

Name of the rulebase.

Usage Notes

This procedure creates a user-defined rulebase. After creating the rulebase, you can add rules to it. To cause the rules in the rulebase to be applied in a query of RDF data, you can specify the rulebase in the call to the SEM_MATCH table function.

Rules and rulebases are explained in Section 1.3.6. The SEM_MATCH table function is described in Section 1.6,

Examples

The following example creates a rulebase named family_rb. (It is an excerpt from Example 1-43 in Section 1.11.2.)

EXECUTE SEM_APIS.CREATE_RULEBASE('family_rb');

SEM_APIS.CREATE_SEM_MODEL

Format

SEM_APIS.CREATE_SEM_MODEL(

     model_name IN VARCHAR2,

     table_name IN VARCHAR2,

     column_name IN VARCHAR2,

     model_tablespace IN VARCHAR2 DEFAULT NULL);

Description

Creates a semantic technology model.

Parameters

model_name

Name of the model.

table_name

Name of the table to hold references to semantic technology data for this model.

column_name

Name of the column of type SDO_RDF_TRIPLE_S in table_name.

model_tablespace

Name of the tablespace for the tables and other database objects used by Oracle to support this model. The default value is the tablespace that was specified in the call to the SEM_APIS.CREATE_SEM_NETWORK procedure.

Usage Notes

You must create the table to hold references to semantic technology data before calling this procedure to create the semantic technology model. For more information, see Section 1.10.

This procedure adds the model to the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

This procedure is the only supported way to create a model. Do not use SQL INSERT statements with the MDSYS.SEM_MODEL$ view.

To delete a model, use the SEM_APIS.DROP_SEM_MODEL procedure.

Examples

The following example creates a semantic technology model named articles. References to the triple data for the model will be stored in the TRIPLE column of the ARTICLES_RDF_DATA table. (This example is an excerpt from Example 1-42 in Section 1.11.2.)

EXECUTE SEM_APIS.CREATE_SEM_MODEL('articles', 'articles_rdf_data', 'triple');

The definition of the ARTICLES_RDF_DATA table is as follows:

CREATE TABLE articles_rdf_data (id NUMBER, triple SDO_RDF_TRIPLE_S);

SEM_APIS.CREATE_SEM_NETWORK

Format

SEM_APIS.CREATE_SEM_NETWORK(

     tablespace_name IN VARCHAR2

Description

Creates structures for persistent storage of semantic data.

Parameters

tablespace_name

Name of the tablespace to be used for tables created by this procedure. This tablespace will be the default for all models that you create, although you can override the default when you create a model by specifying the model_tablespace parameter in the call to the SEM_APIS.CREATE_SEM_MODEL procedure.

Usage Notes

This procedure creates system tables and other database objects used for semantic technology support.

You should create a tablespace for the semantic technology system tables and specify the tablespace name in the call to this procedure. (You should not specify the SYSTEM tablespace.) The size needed for the tablespace that you create will depend on the amount of semantic technology data you plan to store.

You must connect to the database as a user with DBA privileges in order to call this procedure, and you should call the procedure only once for the database.

To drop these structures for persistent storage of semantic data, you must connect as a user with DBA privileges and call the SEM_APIS.DROP_SEM_NETWORK procedure.

Examples

The following example creates a tablespace for semantic technology system tables and creates structures for persistent storage of semantic data in this tablespace.

CREATE TABLESPACE rdf_tblspace
 DATAFILE '/oradata/orcl/rdf_tblspace.dat' SIZE 1024M REUSE
 AUTOEXTEND ON NEXT 256M MAXSIZE UNLIMITED
 SEGMENT SPACE MANAGEMENT AUTO;
. . .
EXECUTE SEM_APIS.CREATE_SEM_NETWORK('rdf_tblspace');

SEM_APIS.CREATE_SOURCE_EXTERNAL_TABLE

Format

SEM_APIS.CREATE_SOURCE_EXTERNAL_TABLE(

     source_table IN VARCHAR2,

     def_directory IN VARCHAR2,

     log_directory IN VARCHAR2 DEFAULT NULL,

     bad_directory IN VARCHAR2 DEFAULT NULL,

     log_file IN VARCHAR2 DEFAULT NULL,

     bad_file IN VARCHAR2 DEFAULT NULL,

     parallel IN INTEGER DEFAULT NULL,

     source_table_owner IN VARCHAR2 DEFAULT NULL,

     flags IN VARCHAR2 DEFAULT NULL);

Description

Creates an external table to map an N-Triple or N-Quad format file into a table.

Parameters

source_table

Name of the external table to be created.

def_directory

Database directory where the input files are located. To load from this staging table, you must have READ privilege on this directory.

log_directory

Database directory where the log files will be generated when loading from the external table. If not specified, the value of the def_directory parameter is used. When loading from the external table, you must have WRITE privilege on this directory.

bad_directory

Database directory where the bad files will be generated when loading from the external table. If not specified, the value of the def_directory parameter is used. When loading from the external table, you must have WRITE privilege on this directory.

log_file

Name of the log file. If not specified, the name will be .generated automatically during a load operation.

bad_file

Name of the bad file. If not specified, the name will be .generated automatically during a load operation.

parallel

Degree of parallelism to associate with the external table being created.

source_table_owner

Owner for the external table being created. If not specified, the invoker becomes the owner.

flags

(Reserved for future use)

Usage Notes

For more information and an example, see Section 1.7.1.1.2, "Loading N-Quad Format Data into a Staging Table Using an External Table".

Examples

The following example creates a source external table. (This example is an excerpt from Example 1-36 in Section 1.7.1.1.2.)

BEGIN
  sem_apis.create_source_external_table(
    source_table    => 'stage_table_source'
   ,def_directory   => 'DATA_DIR'
   ,bad_file        => 'CLOBrows.bad'
   );
END;

SEM_APIS.CREATE_VIRTUAL_MODEL

Format

SEM_APIS.CREATE_VIRTUAL_MODEL(

     vm_name IN VARCHAR2,

     models IN SEM_MODELS,

     rulebases IN SEM_RULEBASES DEFAULT NULL,

     options IN VARCHAR2 DEFAULT NULL);

Description

Creates a virtual model containing the specified semantic models and rulebases.

Parameters

vm_name

Name of the virtual model to be created.

models

One or more semantic model names. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25). At least one semantic model must be specified.

rulebases

One or more rulebase names. Its data type is SEM_RULEBASES, which has the following definition: TABLE OF VARCHAR2(25). If this parameter is null, no rulebases are included in the virtual model definition. Rules and rulebases are explained in Section 1.3.6.

options

(Reserved for future use.)

Usage Notes

For an explanation of virtual models, including usage information, see Section 1.3.8.

An entailment must exist for each specified combination of semantic model and rulebase.

To create a virtual model, you must either be (A) the owner of each specified model and any corresponding entailments, or (B) a user with DBA privileges.

This procedure creates views with names in the following format:

To use the example in Section 1.3.8 of a virtual model vm1 created from models m1, m2, m3, and with an entailment created for m1, m2 ,and m3 using the OWLPrime rulebase, this procedure will create the following two views (assuming that m1, m2, and m3, and the OWLPRIME entailment have internal model_id values 1, 2, 3, 4):

CREATE VIEW MDSYS.SEMV_VM1 AS
  SELECT start_node_id, p_value_id, canon_end_node_id, end_node_id
  FROM MDSYS.rdf_link$
  WHERE model_id IN (1, 2, 3, 4);
 
CREATE VIEW MDSYS.SEMU_VM1 AS
  SELECT start_node_id, p_value_id, canon_end_node_id, MAX(end_node_id)
  FROM MDSYS.rdf_link$
  WHERE model_id IN (1, 2, 3, 4)
  GROUP BY start_node_id, p_value_id, canon_end_node_id;

The user that invokes this procedure will be the owner of the virtual model and will have SELECT WITH GRANT privileges on the SEMU_vm_name and SEMV_vm_name views. To query the corresponding virtual model, a user must have select privileges on these views.

Examples

The following example creates a virtual model named VM1.

EXECUTE sem_apis.create_virtual_model('VM1', sem_models('model_1', 'model_2'), sem_rulebases('OWLPRIME'));

SEM_APIS.DISABLE_CHANGE_TRACKING

Format

SEM_APIS.DISABLE_CHANGE_TRACKING(

     models_in IN SEM_MODELS);

Description

Disables change tracking for a specified set of models.

Parameters

models_in

One or more model names. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25)

Usage Notes

Disabling change tracking on a model automatically disables incremental inference on all entailment that use the model.

To use this procedure, you must be the owner of the specified model, and incremental inference must have been previously enabled.

For an explanation of incremental inference, including usage information, see Section 2.2.9.

Examples

The following example disables change tracking for the family model.

EXECUTE sem_apis.disable_change_tracking(sem_models('family'));

SEM_APIS.DISABLE_INC_INFERENCE

Format

SEM_APIS.DISABLE_INC_INFERENCE(

     entailment_name IN VARCHAR2);

Description

Disables incremental inference for a specified entailment (rules index).

Parameters

entailment_name

Name of the entailment for which to disable incremental inference.

Usage Notes

To use this procedure, you must be the owner of the specified entailment, and incremental inference must have been previously enabled by the SEM_APIS.ENABLE_INC_INFERENCE procedure.

Calling this procedure automatically disables change tracking for all models owned by the invoking user that were having changes tracked only because of this particular inference.

For an explanation of incremental inference, including usage information, see Section 2.2.9.

Examples

The following example enables incremental inference for the entailment named RDFS_RIX_FAMILY.

EXECUTE sem_apis.disable_inc_inference('rdfs_rix_family');

SEM_APIS.DROP_DATATYPE_INDEX

Format

SEM_APIS.DROP_DATATYPE_INDEX(

     datatype IN VARCHAR2,

     force_drop IN BOOLEAN default FALSE);

Description

Drops (deletes) an existing data type index.

Parameters

datatype

URI of the data type for the index to drop.

force_drop

TRUE forces the index to be dropped if an error occurs during the processing of the statement; FALSE (the default) does not drop the index if an error occurs during the processing of the statement.

Usage Notes

You must have DBA privileges to call this procedure.

For an explanation of data type indexes, see Section 1.9.

Examples

The following example drops the data type index for xsd:string typed literals and plain literals.

EXECUTE SEM_APIS.DROP_DATATYPE_INDEX('http://www.w3.org/2001/XMLSchema#string');

SEM_APIS.DROP_ENTAILMENT

Format

SEM_APIS.DROP_ENTAILMENT(

     entailment_name_in IN VARCHAR2,

     named_g_in IN SEM_GRAPHS DEFAULT NULL,

     dop IN INT DEFAULT 1);

Description

Drops (deletes) an entailment (rules index).

Parameters

entailment_name_in

Name of the entailment to be deleted.

named_g_in

Causes only the triples with the specified graph names in the entailment to be deleted. A null value (the default) drops the entire entailment.

For example, named_g_in => sem_graphs('<urn:G1>','<urn:G2>') drops only the triples in entailment with graph names G1 and G2; the rest of the entailment graph is not dropped.

dop

Degree of parallelism for a parallel execution of triple deletion. Applies only if the named_g_in parameter is not null.

Usage Notes

You can use this procedure to delete an entailment that you created using the SEM_APIS.CREATE_ENTAILMENT procedure.

If you drop only a subset of the entailment with specified named graphs (that is, when named_g_in is not null) on an entailment with a VALID or INCOMPLETE status, then the resulting status of the entailment after the drop is set to INCOMPLETE.

Examples

The following example deletes a entailment named OWLTST_IDX.

EXECUTE sem_apis.drop_entailment('owltst_idx');

The following example deletes only inferred triples with graph names G1 and G2 that belong to the entailment named OWLNG_IDX. Any inferred triples in the default graph and other named graphs remain in the entailment.

EXECUTE sem_apis.drop_entailment('owlng_idx',sem_graphs('<urn:G1>','<urn:G2>'));

SEM_APIS.DROP_RULEBASE

Format

SEM_APIS.DROP_RULEBASE(

     rulebase_name IN VARCHAR2);

Description

Deletes a rulebase.

Parameters

rulebase_name

Name of the rulebase.

Usage Notes

This procedure deletes the specified rulebase, making it no longer available for use in calls to the SEM_MATCH table function. For information about rulebases, see Section 1.3.6.

Only the creator of a rulebase can delete the rulebase.

Examples

The following example drops the rulebase named family_rb.

EXECUTE SEM_APIS.DROP_RULEBASE('family_rb');

SEM_APIS.DROP_SEM_INDEX

Format

SEM_APIS.DROP_SEM_INDEX(

     index_code IN VARCHAR2);

Description

Drops a semantic network index on the models and entailments of the semantic network.

Parameters

index_code

Index code string. Must match the index_code value that was specified in an earlier call to the SEM_APIS.ADD_SEM_INDEX procedure.

Usage Notes

For an explanation of semantic network indexes, see Section 1.8.

Examples

The following example drops a semantic network index with the index code string pcsm on the models and entailments of the semantic network.

EXECUTE SEM_APIS.DROP_SEM_INDEX('pscm');

SEM_APIS.DROP_SEM_MODEL

Format

SEM_APIS.DROP_SEM_MODEL(

     model_name IN VARCHAR2);

Description

Drops (deletes) a semantic technology model.

Parameters

model_name

Name of the model.

Usage Notes

This procedure deletes the model from the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

This procedure is the only supported way to delete a model. Do not use SQL DELETE statements with the MDSYS.SEM_MODEL$ view.

Only the creator of a model can delete the model.

Examples

The following example drops the semantic technology model named articles.

EXECUTE SEM_APIS.DROP_SEM_MODEL('articles');

SEM_APIS.DROP_SEM_NETWORK

Format

SEM_APIS.DROP_SEM_NETWORK(

     cascade IN BOOLEAN DEFAULT FALSE);

Description

Removes structures used for persistent storage of semantic data.

Parameters

cascade

TRUE drops any existing semantic technology models and rulebases, and removes structures used for persistent storage of semantic data; FALSE (the default) causes the operation to fail if any semantic technology models or rulebases exist.

Usage Notes

To remove structures used for persistent storage of semantic data, you must connect as a user with DBA privileges and call this procedure.

If any version-enabled models exist, this procedure will fail regardless of the value of the cascade parameter.

Examples

The following example removes structures used for persistent storage of semantic data.

EXECUTE SEM_APIS.DROP_SEM_NETWORK;

SEM_APIS.DROP_USER_INFERENCE_OBJS

Format

SEM_APIS.DROP_USER_INFERENCE_OBJS(

     uname IN VARCHAR2);

Description

Drops (deletes) all rulebases and entailments owned by a specified database user.

Parameters

uname

Name of a database user. (This value is case-sensitive; for example, HERMAN and herman are considered different users.)

Usage Notes

You must have sufficient privileges to delete rules and rulebases for the specified user.

This procedure does not delete the database user. It deletes only RDF rulebases and entailments owned by that user.

Examples

The following example deletes all rulebases and entailments owned by user SCOTT.

EXECUTE SEM_APIS.DROP_USER_INFERENCE_OBJS('SCOTT');
 
PL/SQL procedure successfully completed.

SEM_APIS.DROP_VIRTUAL_MODEL

Format

SEM_APIS.DROP_VIRTUAL_MODEL(

     vm_name IN VARCHAR2);

Description

Drops (deletes) a virtual model.

Parameters

vm_name

Name of the virtual model to be deleted.

Usage Notes

You can use this procedure to delete a virtual model that you created using the SEM_APIS.CREATE_VIRTUAL_MODEL procedure. A virtual model is deleted automatically if any of its component models, rulebases, or entailment are deleted.

To use this procedure, you must be the owner of the specified virtual model.

For an explanation of virtual models, including usage information, see Section 1.3.8.

Examples

The following example deletes a virtual model named VM1.

EXECUTE sem_apis.drop_virtual_model('VM1');

SEM_APIS.ENABLE_CHANGE_TRACKING

Format

SEM_APIS.ENABLE_CHANGE_TRACKING(

     models_in IN SEM_MODELS);

Description

Enables change tracking for a specified set of models.

Parameters

models_in

One or more model names. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25)

Usage Notes

Change tracking must be enabled on a model before incremental inference can be enabled on any entailments that use the model.

To use this procedure, you must be the owner of the specified model or models.

If the owner of an entailment is also an owner of any underlying models, then enabling incremental inference on the entailment (by calling the SEM_APIS.ENABLE_INC_INFERENCE procedure) automatically enables change tracking on those models owned by that user.

To disable change tracking for a set of models, use the SEM_APIS.DISABLE_CHANGE_TRACKING procedure.

For an explanation of incremental inference, including usage information, see Section 2.2.9.

Examples

The following example enables change tracking for the family model.

EXECUTE sem_apis.enable_change_tracking(sem_models('family'));

SEM_APIS.ENABLE_INC_INFERENCE

Format

SEM_APIS.ENABLE_INC_INFERENCE(

     entailment_name IN VARCHAR2);

Description

Enables incremental inference for a specified entailment (rules index).

Parameters

entailment_name

Name of the entailment for which to enable incremental inference.

Usage Notes

To use this procedure, you must be the owner of the specified entailment.

Before this procedure is executed, all underlying models involved in the entailment must have change tracking enabled. If the owner of the entailment is also an owner of any underlying models, calling this procedure automatically enables change tracking on those models. However, if some underlying model are not owned by the owner of the entailment, the appropriate model owners must first call the SEM_APIS.ENABLE_CHANGE_TRACKING procedure to enable change tracking on those models.

To disable incremental inference for an entailment, use the SEM_APIS.DISABLE_INC_INFERENCE procedure.

For an explanation of incremental inference, including usage information, see Section 2.2.9.

Examples

The following example enables incremental inference for the entailment named RDFS_RIX_FAMILY.

EXECUTE sem_apis.enable_inc_inference('rdfs_rix_family');

SEM_APIS.GET_CHANGE_TRACKING_INFO

Format

SEM_APIS.GET_CHANGE_TRACKING_INFO(

     model_name IN VARCHAR2,

     enabled OUT BOOLEAN,

     tracking_start_time OUT TIMESTAMP);

Description

Returns change tracking information for a model.

Parameters

model_name

Name of the semantic technology model.

enabled

Boolean value returned by the procedure: TRUE if change tracking is enabled for the model, or FALSE if change tacking is not enabled for the model.

timestamp

Timestamp indicating when change tracking was enabled for the model (if it is enabled).

Usage Notes

The model_name value must match a value in the MODEL_NAME column in the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

To enable change tracking for a set of models, use the SEM_APIS.ENABLE_CHANGE_TRACKING procedure.

For an explanation of incremental inference, including usage information, see Section 2.2.9.

Examples

The following example displays change tracking information for a model.

DECLARE 
  bEnabled  boolean;
  tsEnabled  timestamp;
  
BEGIN
  EXECUTE IMMEDIATE 'create table m1 (t SDO_RDF_TRIPLE_S)';
  sem_apis.create_sem_model('m1','m1','t');
 
  sem_apis.enable_change_tracking(sem_models('m1'));
 
  sem_apis.get_change_tracking_info('m1', bEnabled,  tsEnabled);
  dbms_output.put_line('is enabled:' || case when bEnabled then 'true' else 'false' end);
  dbms_output.put_line('enabled at:' || tsEnabled);
END;
/

SEM_APIS.GET_INC_INF_INFO

Format

SEM_APIS.GET_INC_INF_INFO(

     entailment_name IN VARCHAR2,

     enabled OUT BOOLEAN,

     prev_inf_start_time OUT TIMESTAMP);

Description

Returns incremental inference information for an entailment.

Parameters

entailment_name

Name of the entailment.

enabled

Boolean value returned by the procedure: TRUE if incremental inference is enabled for the entailment, or FALSE if incremental inference is not enabled for the entailment.

timestamp

Timestamp indicating when the entailment was most recently updated (if incremental inference is enabled).

Usage Notes

To enable incremental inference for an entailment, use the SEM_APIS.ENABLE_INC_INFERENCE procedure.

For an explanation of incremental inference, including usage information, see Section 2.2.9.

Examples

The following example displays incremental inference information for an entailment.

DECLARE 
  bEnabled boolean;
  tsEnabled timestamp;
  
DECLARE 
  EXECUTE IMMEDIATE 'create table m1 (t SDO_RDF_TRIPLE_S)';
  sem_apis.create_sem_model('m1','m1','t');
 
  sem_apis.create_entailment('m1_inf',sem_models('m1'), 
sem_rulebases('owlprime'),null,null,'INC=T');
 
  sem_apis.get_inc_inf_info('m1_inf', bEnabled,  tsEnabled);
  dbms_output.put_line('is enabled:' || case when bEnabled then 'true' else 'false' 
  end);
  dbms_output.put_line('enabled at:' || tsEnabled);
END
/

SEM_APIS.GET_MODEL_ID

Format

SEM_APIS.GET_MODEL_ID(

     model_name IN VARCHAR2

     ) RETURN NUMBER;

Description

Returns the model ID number of a semantic technology model.

Parameters

model_name

Name of the semantic technology model.

Usage Notes

The model_name value must match a value in the MODEL_NAME column in the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

Examples

The following example returns the model ID number for the model named articles. (This example is an excerpt from Example 1-42 in Section 1.11.2.)

SELECT SEM_APIS.GET_MODEL_ID('articles') AS model_id FROM DUAL;
 
  MODEL_ID
----------
         1

SEM_APIS.GET_MODEL_NAME

Format

SEM_APIS.GET_MODEL_NAME(

     model_id IN NUMBER

     ) RETURN VARCHAR2;

Description

Returns the model name of a semantic technology model.

Parameters

model_id

ID number of the semantic technology model.

Usage Notes

The model_id value must match a value in the MODEL_ID column in the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

Examples

The following example returns the model ID number for the model with the ID value of 1. This example is an excerpt from Example 1-42 in Section 1.11.2.)

SQL> SELECT SEM_APIS.GET_MODEL_NAME(1) AS model_name FROM DUAL;
 
MODEL_NAME                                                       
--------------------------------------------------------------------------------
ARTICLES                                                           

SEM_APIS.GET_TRIPLE_ID

Format

SEM_APIS.GET_TRIPLE_ID(

     model_id IN NUMBER,

     subject IN VARCHAR2,

     property IN VARCHAR2,

     object IN VARCHAR2

     ) RETURN VARCHAR2;

or

SEM_APIS.GET_TRIPLE_ID(

     model_name IN VARCHAR2,

     subject IN VARCHAR2,

     property IN VARCHAR2,

     object IN VARCHAR2

     ) RETURN VARCHAR2;

Description

Returns the ID of a triple in the specified semantic technology model, or a null value if the triple does not exist.

Parameters

model_id

ID number of the semantic technology model. Must match a value in the MODEL_ID column of the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

model_name

Name of the semantic technology model. Must match a value in the MODEL_NAME column of the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

subject

Subject. Must match a value in the VALUE_NAME column of the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

property

Property. Must match a value in the VALUE_NAME column of the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

object

Object. Must match a value in the VALUE_NAME column of the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

Usage Notes

This function has two formats, enabling you to specify the semantic technology model by its model number or its name.

Examples

The following example returns the ID number of a triple. (This example is an excerpt from Example 1-42 in Section 1.11.2.)

SELECT SEM_APIS.GET_TRIPLE_ID(
  'articles',
  'http://nature.example.com/Article2',
  'http://purl.org/dc/terms/references',
  'http://nature.example.com/Article3') AS RDF_triple_id FROM DUAL;
 
RDF_TRIPLE_ID
--------------------------------------------------------------------------------
2_9F2BFF05DA0672E_90D25A8B08C653A_46854582F25E8AC5

SEM_APIS.GETV$DATETIMETZVAL

Format

SEM_APIS.GETV$DATETIMETZVAL(

     value_type IN VARCHAR2,

     vname_prefix IN VARCHAR2,

     vname_suffix IN VARCHAR2,

     literal_type IN VARCHAR2,

     language_type IN VARCHAR2,

     ) RETURN NUMBER;

Description

Returns a TIMESTAMP WITH TIME ZONE value for xsd:dateTime typed literals, and returns a null value for all other RDF terms. Greenwich Mean Time is used as the default time zone for xsd:dateTime values without time zones.

Parameters

value_type

Type of the RDF term.

vname_prefix

Prefix value of the RDF term.

vname_suffix

Suffix value of the RDF term.

literal_type

Literal type of the RDF term.

language_type

Language type of the RDF term.

Usage Notes

For better performance, consider creating a function-based index on this function. For more information, see Section 1.6.7.2.

Examples

The following example returns TIMESTAMP WITH TIME ZONE values for all xsd:dateTime literals in the MDSYS.RDF_VALUE$ table:

SELECT SEM_APIS.GETV$DATETIMETZVAL(value_type, vname_prefix, vname_suffix, 
  literal_type, language_type) 
  FROM MDSYS.RDF_VALUE$;

SEM_APIS.GETV$DATETZVAL

Format

SEM_APIS.GETV$DATETZVAL(

     value_type IN VARCHAR2,

     vname_prefix IN VARCHAR2,

     vname_suffix IN VARCHAR2,

     literal_type IN VARCHAR2,

     language_type IN VARCHAR2,

     ) RETURN TIMESTAMP WITH TIME ZONE;

Description

Returns a TIMESTAMP WITH TIME ZONE value for xsd:date typed literals, and returns a null value for all other RDF terms. Greenwich Mean Time is used as the default time zone for xsd:date values without time zones.

Parameters

value_type

Type of the RDF term.

vname_prefix

Prefix value of the RDF term.

vname_suffix

Suffix value of the RDF term.

literal_type

Literal type of the RDF term.

language_type

Language type of the RDF term.

Usage Notes

For better performance, consider creating a function-based index on this function. For more information, see Section 1.6.7.2.

Examples

The following example returns TIMESTAMP WITH TIME ZONE values for all xsd:date literals in the MDSYS.RDF_VALUE$ table:

SELECT SEM_APIS.GETV$DATETZVAL(value_type, vname_prefix, vname_suffix, 
  literal_type, language_type) 
  FROM MDSYS.RDF_VALUE$;

SEM_APIS.GETV$NUMERICVAL

Format

SEM_APIS.GETV$NUMERICVAL(

     value_type IN VARCHAR2,

     vname_prefix IN VARCHAR2,

     vname_suffix IN VARCHAR2,

     literal_type IN VARCHAR2,

     language_type IN VARCHAR2,

     ) RETURN NUMBER;

Description

Returns a numeric value for XML Schema numeric typed literals, and returns a null value for all other RDF terms.

Parameters

value_type

Type of the RDF term.

vname_prefix

Prefix value of the RDF term.

vname_suffix

Suffix value of the RDF term.

literal_type

Literal type of the RDF term.

language_type

Language type of the RDF term.

Usage Notes

For better performance, consider creating a function-based index on this function. For more information, see Section 1.6.7.2.

Examples

The following example returns numeric values for all numeric literals in the MDSYS.RDF_VALUE$ table:

SELECT SEM_APIS.GETV$NUMERICVAL(value_type, vname_prefix, vname_suffix, 
  literal_type, language_type) 
  FROM MDSYS.RDF_VALUE$;

SEM_APIS.GETV$STRINGVAL

Format

SEM_APIS.GETV$STRINGVAL(

     value_type IN VARCHAR2,

     vname_prefix IN VARCHAR2,

     vname_suffix IN VARCHAR2,

     literal_type IN VARCHAR2,

     language_type IN VARCHAR2,

     ) RETURN TIMESTAMP WITH TIME ZONE;

Description

Returns a VARCHAR2 string of the lexical form of plain literals and xsd:string typed literals, and returns a null value for all other RDF terms. CHR(0) is returned for empty literals.

Parameters

value_type

Type of the RDF term.

vname_prefix

Prefix value of the RDF term.

vname_suffix

Suffix value of the RDF term.

literal_type

Literal type of the RDF term.

language_type

Language type of the RDF term.

Usage Notes

For better performance, consider creating a function-based index on this function. For more information, see Section 1.6.7.2.

Examples

The following example returns lexical values for all plain literals and xsd:string literals in the MDSYS.RDF_VALUE$ table:

SELECT SEM_APIS.GETV$STRINGVAL(value_type, vname_prefix, vname_suffix, 
  literal_type, language_type) 
  FROM MDSYS.RDF_VALUE$;

SEM_APIS.GETV$TIMETZVAL

Format

SEM_APIS.GETV$TIMETZVAL(

     value_type IN VARCHAR2,

     vname_prefix IN VARCHAR2,

     vname_suffix IN VARCHAR2,

     literal_type IN VARCHAR2,

     language_type IN VARCHAR2,

     ) RETURN TIMESTAMP WITH TIME ZONE;

Description

Returns a TIMESTAMP WITH TIME ZONE value for xsd:time typed literals, and returns a null value for all other RDF terms. Greenwich Mean Time is used as the default time zone for xsd:time values without time zones. 2009-06-26 is used as the default date in all the generated TIMESTAMP WITH TIME ZONE values.

Parameters

value_type

Type of the RDF term.

vname_prefix

Prefix value of the RDF term.

vname_suffix

Suffix value of the RDF term.

literal_type

Literal type of the RDF term.

language_type

Language type of the RDF term.

Usage Notes

For better performance, consider creating a function-based index on this function. For more information, see Section 1.6.7.2.

Because xsd:time values include only a time but not a date, the returned TIMESTAMP WITH TIME ZONE values (which include a date component) have 2009-06-26 added as the date. This is done so that the returned values can be indexed internally, and so that the date is the same for all of them.

Examples

The following example returns TIMESTAMP WITH TIME ZONE values (using the default 2009-06-26 for the date) for all xsd:time literals in the MDSYS.RDF_VALUE$ table. (

SELECT SEM_APIS.GETV$DATETIMETZVAL(value_type, vname_prefix, vname_suffix, 
  literal_type, language_type) 
  FROM MDSYS.RDF_VALUE$;

SEM_APIS.IS_TRIPLE

Format

SEM_APIS.IS_TRIPLE(

     model_id IN NUMBER,

     subject IN VARCHAR2,

     property IN VARCHAR2,

     object IN VARCHAR2) RETURN VARCHAR2;

or

SEM_APIS.IS_TRIPLE(

     model_name IN VARCHAR2,

     subject IN VARCHAR2,

     property IN VARCHAR2,

     object IN VARCHAR2) RETURN VARCHAR2;

Description

Checks if a statement is an existing triple in the specified model in the database.

Parameters

model_id

ID number of the semantic technology model. Must match a value in the MODEL_ID column of the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

model_name

Name of the semantic technology model. Must match a value in the MODEL_NAME column of the MDSYS.SEM_MODEL$ view, which is described in Section 1.3.1.

subject

Subject. Must match a value in the VALUE_NAME column of the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

property

Property. Must match a value in the VALUE_NAME column of the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

object

Object. Must match a value in the VALUE_NAME column of the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

Usage Notes

This function returns the string value FALSE, TRUE, or TRUE (EXACT):

Examples

The following example checks if a statement is a triple in the database. In this case, there is an exact match. (This example is an excerpt from Example 1-42 in Section 1.11.2.)

SELECT SEM_APIS.IS_TRIPLE(
  'articles',
  'http://nature.example.com/Article2',
  'http://purl.org/dc/terms/references',
  'http://nature.example.com/Article3') AS is_triple FROM DUAL;
 
IS_TRIPLE                                                                       
--------------------------------------------------------------------------------
TRUE (EXACT)

SEM_APIS.LOAD_INTO_STAGING_TABLE

Format

SEM_APIS.LOAD_INTO_STAGING_TABLE(

     stagong_table IN VARCHAR2,

     source_table IN VARCHAR2,

     input_format IN VARCHAR2 DEFAULT NULL,

     parallel IN INTEGER DEFAULT NULL,

     staging_table_owner IN VARCHAR2 DEFAULT NULL,

     source_table_owner IN VARCHAR DEFAULT NULL,

     flags IN VARCHAR DEFAULT NULL);

Description

Loads data into a staging table from an external table mapped to an N-Triple or N-Quad format input file.

Parameters

staging_table

Name of the staging table.

source_table

Name of the source external table.

input_format

Format of the input file mapped by the source external table: N-TRIPLE or N-QUAD

parallel

Degree of parallelism to use during the load.

staging_table_owner

Owner for the staging table being created. If not specified, the invoker is assumed to be the owner.

source_table_owner

Owner for the source table. If not specified, the invoker is assumed to be the owner.

flags

(Reserved for future use)

Usage Notes

For more information and an example, see Section 1.7.1.1.2, "Loading N-Quad Format Data into a Staging Table Using an External Table".

Examples

The following example loads the staging table. (This example is an excerpt from Example 1-36 in Section 1.7.1.1.2.)

BEGIN
  sem_apis.load_into_staging_table(
    staging_table => 'STAGE_TABLE'
   ,source_table  => 'stage_table_source'
   ,input_format  => 'N-QUAD');
END;

SEM_APIS.LOOKUP_ENTAILMENT

Format

SEM_APIS.LOOKUP_ENTAILMENT (

     models IN SEM_MODELS,

     rulebases IN SEM_RULEBASES

     ) RETURN VARCHAR2;

Description

Returns the name of the entailment (rules index) based on the specified models and rulebases.

Parameters

models

One or more model names. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25)

rulebases

One or more rulebase names. Its data type is SEM_RULEBASES, which has the following definition: TABLE OF VARCHAR2(25)Rules and rulebases are explained in Section 1.3.6.

Usage Notes

For a rulebase index to be returned, it must be based on all specified models and rulebases.

Examples

The following example finds the entailment that is based on the family model and the RDFS and family_rb rulebases. (It is an excerpt from Example 1-43 in Section 1.11.2.)

SELECT SEM_APIS.LOOKUP_ENTAILMENT(SEM_MODELS('family'),
  SEM_RULEBASES('RDFS','family_rb')) AS lookup_entailment FROM DUAL;

LOOKUP_ENTAILMENT
--------------------------------------------------------------------------------
RDFS_RIX_FAMILY

SEM_APIS.MERGE_MODELS

Format

SEM_APIS.MERGE_MODELS(

     source_model IN VARCHAR2,

     destination_model IN VARCHAR2,

     rebuild_apptab_index IN BOOLEAN DEFAULT TRUE,

     drop_source_model IN BOOLEAN DEFAULT FALSE,

     options IN VARCHAR2 DEFAULT NULL);

Description

Inserts the content from a source model into a destination model, and updates the destination application table.

Parameters

source_model

Name of the source model.

destination_model

Name of the destination model.

rebuild_apptab_index

TRUE causes indexes on the destination application table to be rebuilt after the models are merged; FALSE does not rebuild any indexes.

drop_source_model

TRUE causes the source model (source_model) to be deleted after the models are merged; FALSE (the default) does not delete the source model.

options

A comma-delimited string of options that overrides the default behavior of the procedure. Currently, only the DOP (degree of parallelism) option is supported, to enable parallel execution of this procedure and to specify the degree of parallelism to be associated with the operation.

Usage Notes

Before you merge any models, if you are using positional parameters, check to be sure that you are specifying the correct models for the first and second parameters (source model for the first, destination model for the second). This is especially important if you plan to specify drop_source_model=TRUE.

If appropriate, make copies of the destination model or both models before performing the merge. To make a copy of a model, use SEM_APIS.CREATE_SEM_MODEL to create an empty model with the desired name for the copy, and use SEM_APIS.MERGE_MODELS to populate the newly created copy as the destination model.

Some common uses for this procedure include the following:

On a multi-core or multi-cpu machine, the DOP (degree of parallelism) option can be beneficial. See Examples for an example that uses the DOP option.

If the source model is large, you may want to update the optimizer statistics on the destination after the merge operation by calling the SEM_APIS.ANALYZE_MODEL procedure.

The following considerations apply to the use of this procedure:

Examples

The following example inserts the contents of model M1 into M2.

EXECUTE SEM_APIS.MERGE_MODELS('M1', 'M2');

The following example inserts the contents of model M1 into M2, and it specifies a degree of parallelism of 4 (up to four parallel threads for execution of the merge operation).

EXECUTE SEM_APIS.MERGE_MODELS('M1', 'M2', null, null, 'DOP=4');

SEM_APIS.REFRESH_SEM_NETWORK_INDEX_INFO

Format

SEM_APIS.REFRESH_SEM_NETWORK_INDEX_INFO(

     options IN VARCHAR2 DEFAULT NULL);

Description

Refreshes the information about semantic network indexes.

Parameters

options

(Reserved for future use)

Usage Notes

This procedure updates the information in the MDSYS.SEM_NETWORK_INDEX_INFO view, which is described in Section 1.8.1.

Examples

The following example refreshes the information about semantic network indexes.

EXECUTE sem_apis.refresh_sem_network_index_info;

SEM_APIS.REMOVE_DUPLICATES

Format

SEM_APIS.REMOVE_DUPLICATES(

     model_name IN VARCHAR2,

     threshold IN FLOAT DEFAULT 0.3,

     rebuild_apptab_index IN BOOLEAN DEFAULT TRUE);

Description

Removes duplicate triples from a model.

Parameters

model_name

Name of the model.

threshold

A value to determine how numerous triples must be in order for the removal operation to be performed. This procedure removes triples only if the number of triples in the model exceeds the following formula: (total-triples - total-unique-triples + 0.01) / (total-unique-triples + 0.01). For the default value of 0.3 and a model containing 1000 total triples (including duplicates), duplicate triples would be removed only if the number of duplicates exceeds approximately 230.

The lower the threshold value, the fewer duplicates are needed for the procedure to remove duplicates; the higher the threshold value, the more duplicates are needed for the procedure to remove duplicates.

rebuild_apptab_index

TRUE (the default) causes all usable indexes on tables that were affected by this operation to be rebuilt after the duplicate triples are removed; FALSE does not rebuild any indexes.

Usage Notes

When duplicate triples are removed, all information in the removed rows is lost, including information in columns other than the triple column.

This procedure is not supported on virtual models (explained in Section 1.3.8).

This procedure is not supported on version-enabled models (explained in Chapter 6).

If the model is empty, or if it contains no duplicate triples or not enough duplicate triples (as computed using the threshold value), this procedure does not perform any removal operations.

If there are not enough duplicates (as computed using the threshold value) to perform the operation, an informational message is displayed.

Examples

The following example removes duplicate triples in the model named family. It accepts the default threshold value of 0.3 and (by default) rebuilds indexes after the duplicates are removed.

EXECUTE SEM_APIS.REMOVE_DUPLICATES('family');

SEM_APIS.RENAME_ENTAILMENT

Format

SEM_APIS.RENAME_ENTAILMENT(

     old_name IN VARCHAR2,

     new_name IN VARCHAR2);

Description

Renames an entailment (rules index).

Parameters

old_name

Name of the existing entailment to be renamed.

new_name

New name for the entailment.

Usage Notes

This procedure is not supported on version-enabled RDF models, as explained in Section 6.5, "Special Considerations When Using Workspace Manager Support for RDF Data".

Examples

The following example renames a entailment named OWLTST_IDX to MY_OWLTST_IDX.

EXECUTE sem_apis.rename_entailment('owltst_idx', 'my_owltst_idx');

SEM_APIS.RENAME_MODEL

Format

SEM_APIS.RENAME_MODEL(

     old_name IN VARCHAR2,

     new_name IN VARCHAR2);

Description

Renames a model.

Parameters

old_name

Name of the existing model to be renamed.

new_name

New name for the model.

Usage Notes

The following considerations apply to the use of this procedure:

Contrast this procedure with SEM_APIS.SWAP_NAMES, which swaps (exchanges) the names of two existing models.

Examples

The following example renames a model named MODEL1 to MODEL2.

EXECUTE sem_apis.rename_model('model1', 'model2');

SEM_APIS.SWAP_NAMES

Format

SEM_APIS.SWAP_NAMES(

     model1 IN VARCHAR2,

     model2 IN VARCHAR2);

Description

Swaps (exchanges) the names of two existing models.

Parameters

model1

Name of a model.

model2

Name of another model.

Usage Notes

As a result of this procedure, the name of model model1 is changed to the (old) name of model2, and the name of model model2 is changed to the (old) name of model1.

The order of the names does not affect the result. For example, you could specify TEST for model1 and PRODUCTION for model2, or PRODUCTION for model1 and TEST for model2, and the result will be the same.

This procedure is not supported on version-enabled RDF models, as explained in Section 6.5, "Special Considerations When Using Workspace Manager Support for RDF Data".

Contrast this procedure with SEM_APIS.RENAME_MODEL, which renames an existing model.

Examples

The following example changes the name of the (old) TEST model to PRODUCTION, and the name of the (old) PRODUCTION model to TEST.

EXECUTE sem_apis.swap_names('test', 'production');

SEM_APIS.VALIDATE_ENTAILMENT

Format

SEM_APIS.VALIDATE_ENTAILMENT(

     models_in IN SEM_MODELS,

     rulebases_in IN SEM_RULEBASES,

     criteria_in IN VARCHAR2 DEFAULT NULL,

     max_conflict IN NUMBER DEFAULT 100,

     options IN VARCHAR2 DEFAULT NULL

     ) RETURN RDF_LONGVARCHARARRAY;

Description

Validates entailments (rules indexes) that can be used to perform OWL or RDFS inferencing for one or more models.

Parameters

models_in

One or more model names. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25)

rulebases_in

One or more rulebase names. Its data type is SEM_RULEBASES, which has the following definition: TABLE OF VARCHAR2(25). Rules and rulebases are explained in Section 1.3.6.

criteria_in

A comma-delimited string of validation checks to run. If you do not specify this parameter, by default all of the following checks are run:

  • UNSAT: Find unsatisfiable classes.

  • EMPTY: Find instances that belong to unsatisfiable classes.

  • SYNTAX_S: Find triples whose subject is neither URI nor blank node.

  • SYNTAX_P: Find triples whose predicate is not URI.

  • SELF_DIF: Find individuals that are different from themselves.

  • INST: Find individuals that simultaneously belong to two disjoint classes.

  • SAM_DIF: Find pairs of individuals that are same (owl:sameAs) and different (owl:differentFrom) at the same time.

To specify fewer checks, specify a string with only the checks to be performed. For example, criteria_in => 'UNSAT' causes the validation process to search only for unsatisfiable classes.

max_conflict

The maximum number of conflicts to find before the validation process stops. The default value is 100.

options

(Not currently used. Reserved for Oracle use.).

Usage Notes

This procedure can be used to detect inconsistencies in the original entailment. For more information, see Section 2.2.5.

This procedure returns a null value if no errors are detected or (if errors are detected) an object of type RDF_LONGVARCHARARRAY, which has the following definition: VARRAY(32767) OF VARCHAR2(4000)

To create an entailment, use the SEM_APIS.CREATE_ENTAILMENT procedure.

Examples

For an example of this procedure, see Example 2-5 in Section 2.2.5.


SEM_APIS.VALIDATE_MODEL

Format

SEM_APIS.VALIDATE_MODEL(

     models_in IN SEM_MODELS,

     criteria_in IN VARCHAR2 DEFAULT NULL,

     max_conflict IN NUMBER DEFAULT 100,

     options IN VARCHAR2 DEFAULT NULL

     ) RETURN RDF_LONGVARCHARARRAY;

Description

Validates one or more models.

Parameters

models_in

One or more model names. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25)

criteria_in

A comma-delimited string of validation checks to run. If you do not specify this parameter, by default all of the following checks are run:

  • UNSAT: Find unsatisfiable classes.

  • EMPTY: Find instances that belong to unsatisfiable classes.

  • SYNTAX_S: Find triples whose subject is neither URI nor blank node.

  • SYNTAX_P: Find triples whose predicate is not URI.

  • SELF_DIF: Find individuals that are different from themselves.

  • INST: Find individuals that simultaneously belong to two disjoint classes.

  • SAM_DIF: Find pairs of individuals that are same (owl:sameAs) and different (owl:differentFrom) at the same time.

To specify fewer checks, specify a string with only the checks to be performed. For example, criteria_in => 'UNSAT' causes the validation process to search only for unsatisfiable classes.

max_conflict

The maximum number of conflicts to find before the validation process stops. The default value is 100.

options

(Not currently used. Reserved for Oracle use.).

Usage Notes

This procedure can be used to detect inconsistencies in the original data model. For more information, see Section 2.2.5.

This procedure returns a null value if no errors are detected or (if errors are detected) an object of type RDF_LONGVARCHARARRAY, which has the following definition: VARRAY(32767) OF VARCHAR2(4000)

Examples

The following example validates the model named family.

SELECT SEM_APIS.VALIDATE_MODEL(SEM_MODELS('family')) FROM DUAL;

SEM_APIS.VALUE_NAME_PREFIX

Format

SEM_APIS.VALUE_NAME_PREFIX (

     value_name IN VARCHAR2,

     value_type IN VARCHAR2

     ) RETURN VARCHAR2;

Description

Returns the value in the VNAME_PREFIX column for the specified value name and value type pair in the MDSYS.RDF_VALUE$ table.

Parameters

value_name

Value name. Must match a value in the VALUE_NAME column in the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

value_type

Value type. Must match a value in the VALUE_TYPE column in the MDSYS.RDF_VALUE$ table, which is described in <a href="sdo_rdf_concepts.htm#CHDEJDCA">Section 1.3.2.

Usage Notes

This function usually causes an index on the MDSYS.RDF_VALUE$ table to be used for processing a lookup for values, and thus can make a query run faster.

Examples

The following query returns value name portions of all the lexical values in MDSYS.RDF_VALUE$ table with a prefix value same as that returned by the VALUE_NAME_PREFIX function. This query uses an index on the MDSYS.RDF_VALUE$ table, thereby providing efficient lookup.

SELECT value_name FROM MDSYS.RDF_VALUE$
  WHERE vname_prefix = SEM_APIS.VALUE_NAME_PREFIX(
    'http://www.w3.org/1999/02/22-rdf-syntax-ns#type','UR');
 
VALUE_NAME
--------------------------------------------------------------------------------
http://www.w3.org/1999/02/22-rdf-syntax-ns#Alt
http://www.w3.org/1999/02/22-rdf-syntax-ns#Bag
http://www.w3.org/1999/02/22-rdf-syntax-ns#List
http://www.w3.org/1999/02/22-rdf-syntax-ns#Property
http://www.w3.org/1999/02/22-rdf-syntax-ns#Seq
http://www.w3.org/1999/02/22-rdf-syntax-ns#Statement
http://www.w3.org/1999/02/22-rdf-syntax-ns#XMLLiteral
http://www.w3.org/1999/02/22-rdf-syntax-ns#first
http://www.w3.org/1999/02/22-rdf-syntax-ns#nil
http://www.w3.org/1999/02/22-rdf-syntax-ns#object
http://www.w3.org/1999/02/22-rdf-syntax-ns#predicate
http://www.w3.org/1999/02/22-rdf-syntax-ns#rest
http://www.w3.org/1999/02/22-rdf-syntax-ns#subject
http://www.w3.org/1999/02/22-rdf-syntax-ns#type
http://www.w3.org/1999/02/22-rdf-syntax-ns#value
 
15 rows selected.

SEM_APIS.VALUE_NAME_SUFFIX

Format

SEM_APIS.VALUE_NAME_SUFFIX (

     value_name IN VARCHAR2,

     value_type IN VARCHAR2

     ) RETURN VARCHAR2;

Description

Returns the value in the VNAME_SUFFIX column for the specified value name and value type pair in the MDSYS.RDF_VALUE$ table.

Parameters

value_name

Value name. Must match a value in the VALUE_NAME column in the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

value_type

Value type. Must match a value in the VALUE_TYPE column in the MDSYS.RDF_VALUE$ table, which is described in Section 1.3.2.

Usage Notes

This function usually causes an index on the MDSYS.RDF_VALUE$ table to be used for processing a lookup for values, and thus can make a query run faster.

Examples

The following query returns value name portions of all the lexical values in MDSYS.RDF_VALUE$ table with a suffix value same as that returned by the VALUE_NAME_SUFFIX function. This query uses an index on the MDSYS.RDF_VALUE$ table, thereby providing efficient lookup.

SELECT value_name FROM MDSYS.RDF_VALUE$
  WHERE vname_suffix = SEM_APIS.VALUE_NAME_SUFFIX(
    'http://www.w3.org/1999/02/22-rdf-syntax-ns#type','UR');
 
VALUE_NAME
--------------------------------------------------------------------------------
http://www.w3.org/1999/02/22-rdf-syntax-ns#type
PKX$PK.AOEBPS/cover.htmO Cover

Oracle Corporation

PK[pTOPK.AOEBPS/sem_prtref.htm  Reference and Supplementary Information

Part II

Reference and Supplementary Information

This document has the following parts:

Part II contains the following chapters with reference and supplementary information:

To understand the examples in the reference chapters, you must understand the conceptual and data type information in Chapter 1, "Oracle Database Semantic Technologies Overview" and Chapter 2, "OWL Concepts".

PKǤ PK.AOEBPS/title.htm. Oracle Database Semantic Technologies Developer's Guide, 11g Release 2 (11.2)

Oracle® Database

Semantic Technologies Developer's Guide

11g Release 2 (11.2)

E25609-03

May 2012

Provides usage and reference information about Oracle Database support for semantic technologies, including storage, inference, and query capabilities for data and ontologies based on Resource Description Framework (RDF), RDF Schema (RDFS), and Web Ontology Language (OWL).


Oracle Database Semantic Technologies Developer's Guide, 11g Release 2 (11.2)

E25609-03

Copyright © 2005, 2012, Oracle and/or its affiliates. All rights reserved.

Primary Author:  Chuck Murray

Contributors:  Eugene Inseok Chong, Souri Das, Vladimir Kolovski, Matt Perry, Jags Srinivasan, Seema Sundara, Zhe (Alan) Wu, Aravind Yalamanchi

This software and related documentation are provided under a license agreement containing restrictions on use and disclosure and are protected by intellectual property laws. Except as expressly permitted in your license agreement or allowed by law, you may not use, copy, reproduce, translate, broadcast, modify, license, transmit, distribute, exhibit, perform, publish, or display any part, in any form, or by any means. Reverse engineering, disassembly, or decompilation of this software, unless required by law for interoperability, is prohibited.

The information contained herein is subject to change without notice and is not warranted to be error-free. If you find any errors, please report them to us in writing.

If this is software or related documentation that is delivered to the U.S. Government or anyone licensing it on behalf of the U.S. Government, the following notice is applicable:

U.S. GOVERNMENT RIGHTS Programs, software, databases, and related documentation and technical data delivered to U.S. Government customers are "commercial computer software" or "commercial technical data" pursuant to the applicable Federal Acquisition Regulation and agency-specific supplemental regulations. As such, the use, duplication, disclosure, modification, and adaptation shall be subject to the restrictions and license terms set forth in the applicable Government contract, and, to the extent applicable by the terms of the Government contract, the additional rights set forth in FAR 52.227-19, Commercial Computer Software License (December 2007). Oracle America, Inc., 500 Oracle Parkway, Redwood City, CA 94065.

This software or hardware is developed for general use in a variety of information management applications. It is not developed or intended for use in any inherently dangerous applications, including applications that may create a risk of personal injury. If you use this software or hardware in dangerous applications, then you shall be responsible to take all appropriate fail-safe, backup, redundancy, and other measures to ensure its safe use. Oracle Corporation and its affiliates disclaim any liability for any damages caused by use of this software or hardware in dangerous applications.

Oracle and Java are registered trademarks of Oracle and/or its affiliates. Other names may be trademarks of their respective owners.

Intel and Intel Xeon are trademarks or registered trademarks of Intel Corporation. All SPARC trademarks are used under license and are trademarks or registered trademarks of SPARC International, Inc. AMD, Opteron, the AMD logo, and the AMD Opteron logo are trademarks or registered trademarks of Advanced Micro Devices. UNIX is a registered trademark of The Open Group.

This software or hardware and documentation may provide access to or information on content, products, and services from third parties. Oracle Corporation and its affiliates are not responsible for and expressly disclaim all warranties of any kind with respect to third-party content, products, and services. Oracle Corporation and its affiliates will not be responsible for any loss, costs, or damages incurred due to your access to or use of third-party content, products, or services.

PKz>PK.A OEBPS/loe.htm/ List of Examples

List of Examples

PKph(///PK.AOEBPS/sem_jena.htm Jena Adapter for Oracle Database

7 Jena Adapter for Oracle Database

The Jena Adapter for Oracle Database (referred to here as the Jena Adapter) provides a Java-based interface to Oracle Database Semantic Technologies by implementing the well-known Jena Graph, Model, and DatasetGraph APIs. (Jena is an open source framework, and its license and copyright conditions are described in http://jena.sourceforge.net/license.html.)

The DatasetGraph APIs are for managing named graph data, also referred to as quads. In addition, the Jena Adapter provides network analytical functions on top of semantic data through integrating with the Oracle Spatial network data model.

This chapter assumes that you are familiar with major concepts explained in Chapter 1, "Oracle Database Semantic Technologies Overview" and Chapter 2, "OWL Concepts". It also assumes that you are familiar with the overall capabilities and use of the Jena Java framework. For information about the Jena framework, see http://jena.sourceforge.net/, especially the Jena Documentation page. If you use the network analytical function, you should also be familiar with the Oracle Spatial network data model, which is documented in Oracle Spatial Topology and Network Data Models Developer's Guide.

The Jena Adapter extends the semantic data management capabilities of Oracle Database Release 11.2 RDF/OWL.

This chapter includes the following major topics:


Disclaimer:

The current Jena Adapter release has been tested against Jena 2.6.4, ARQ 2.8.8, and Joseki 3.4.4. Because of the nature of open source projects, you should not use this Jena Adapter with later versions of Jena, ARQ, or Joseki.

7.1 Setting Up the Software Environment

To use the Jena Adapter, you must first ensure that the system environment has the necessary software, including Oracle Database 11g Release 2 with the Spatial and Partitioning options and with Semantic Technologies support enabled, Jena version 2.6.4, the Jena Adapter, and JDK 1.6. You can set up the software environment by performing these actions:

  1. Install Oracle Database Release 11.2 Enterprise Edition with the Oracle Spatial and Partitioning Options.

  2. If you have not yet installed Release 11.2.0.3 or later, install the 11.2.0.2 Patch Set for Oracle Database Server (https://updates.oracle.com/ARULink/PatchDetails/process_form?patch_num=10098816).

  3. Enable the support for Semantic Technologies, as explained in Section A.1.

  4. Install Jena (version 2.6.4): download the .zip file from http://sourceforge.net/projects/jena/files/Jena/Jena-2.6.4/jena-2.6.4.zip/download and unzip it. (The directory or folder into which you unzip it will be referred to as <Jena_DIR>.)

    The Java package will be unpacked into <Jena_DIR>.

    Note that Jena 2.6.4 comes with ARQ version 2.8.7 (arq-2.8.7.jar); however, this version of the Jena Adapter actually requires a newer ARQ version (arq-2.8.8.jar). You can download arq-2.8.8.jar from http://sourceforge.net/projects/jena/files/ARQ/ARQ-2.8.8/arq-2.8.8.zip/download and unzip it to a temporary directory. Remove the arq-2.8.7.jar file from <Jena_DIR>/Jena-2.6.4/lib/, and copy arq-2.8.8.jar from the temporary directory into <Jena_DIR>/Jena-2.6.4/lib/.

  5. Download the Jena Adapter (jena_adaptor_for_release11.2.0.3.zip) from the Oracle Database Semantic Technologies page (http://www.oracle.com/technetwork/database/options/semantic-tech/and click Downloads), and unzip it into a temporary directory, such as (on a Linux system) /tmp/jena_adapter. (If this temporary directory does not already exist, create it before the unzip operation.)

    The Jena Adapter directories and files have the following structure:

     examples/
       examples/Test10.java
       examples/Test11.java
       examples/Test12.java
       examples/Test13.java
       examples/Test14.java
       examples/Test15.java
       examples/Test16.java
       examples/Test17.java
       examples/Test18.java
       examples/Test19.java
       examples/Test20.java
       examples/Test6.java
       examples/Test7.java
       examples/Test8.java
       examples/Test9.java
       examples/Test.java
    jar/
       jar/sdordfclient.jar
    javadoc/
      javadoc/javadoc.zip
    joseki/
       joseki/index.html
       joseki/application.xml
       joseki/update.html
       joseki/xml-to-html.xsl
       joseki/joseki-config.ttl
    sparqlgateway/ 
      default.xslt
      noop.xslt
      qb1.sparql
      . . .
      browse.jsp
      index.html
      . . .
      application.xml
      WEB-INF/
        WEB-INF/web.xml
        WEB-INF/weblogic.xml
      WEB-INF/lib/
      StyleSheets/
        StyleSheets/paginator.css
        StyleSheets/sg.css
        StyleSheets/sgmin.css
      Scripts/
        Scripts/load.js
        Scripts/paginator.js
        Scripts/tooltip.js
      admin/
        admin/sparql.jsp
        admin/xslt.jsp
    web/
      web/web.xml
    
  6. Copy ojdbc6.jar into <Jena_DIR>/lib (Linux) or <Jena_DIR>\lib (Windows). (ojdbc6.jar is in $ORACLE_HOME/jdbc/lib or %ORACLE_HOME%\jdbc\lib.)

  7. Copy sdordf.jar into <Jena_DIR>/lib (Linux) or <Jena_DIR>\lib (Windows). (sdordf.jar is in $ORACLE_HOME/md/jlib or %ORACLE_HOME%\md\jlib.)

  8. If JDK 1.6 is not already installed, install it.

  9. If the JAVA_HOME environment variable does not already refer to the JDK 1.6 installation, define it accordingly. For example:

    setenv JAVA_HOME /usr/local/packages/jdk16/
    
  10. If the SPARQL service to support the SPARQL protocol is not set up, set it up as explained in Section 7.2.

After setting up the software environment, ensure that your Semantic Technologies environment can enable you to use the Jena Adapter to perform queries, as explained in Section 7.3.

7.2 Setting Up the SPARQL Service

Setting up a SPARQL endpoint using the Jena Adapter involves downloading Joseki, an open source HTTP engine that supports the SPARQL protocol and SPARQL queries. This section explains how to set up a SPARQL service using a servlet deployed in WebLogic Server. The number and complexity of the steps reflect the fact that Oracle is not permitted to bundle all the dependent third-party libraries in a .war or .ear file.

  1. Download and Install Oracle WebLogic Server 11g Release 1 (10.3.1). For details, see http://www.oracle.com/technology/products/weblogic/ and http://www.oracle.com/technetwork/middleware/ias/downloads/wls-main-097127.html.

  2. Ensure that you have Java 6 installed, because it is required by Joseki 3.4.4.

  3. Download Joseki 3.4.4 (joseki-3.4.4.zip) from http://sourceforge.net/projects/joseki/files/Joseki-SPARQL/.

  4. Unpack joseki-3.4.4.zip into a temporary directory. For example:

    mkdir /tmp/joseki
    cp joseki-3.4.4.zip /tmp/joseki
    cd /tmp/joseki
    unzip joseki-3.4.4.zip
    
  5. Ensure that you have downloaded and unzipped the Jena Adapter for Oracle Database, as explained in Section 7.1.

  6. Create a directory named joseki.war at the same level as the jena_adapter directory, and go to it. For example:

    mkdir /tmp/joseki.war
    cd /tmp/joseki.war
    
  7. Copy necessary files into the directory created in the preceding step:

    cp /tmp/jena_adapter/joseki/*  /tmp/joseki.war
    cp -rf /tmp/joseki/Joseki-3.4.4/webapps/joseki/StyleSheets  /tmp/joseki.war
    
  8. Create directories and copy necessary files into them, as follows:

    mkdir /tmp/joseki.war/WEB-INF
    cp /tmp/jena_adapter/web/*  /tmp/joseki.war/WEB-INF
     
    mkdir /tmp/joseki.war/WEB-INF/lib
    cp /tmp/joseki/Joseki-3.4.4/lib/*.jar   /tmp/joseki.war/WEB-INF/lib
    cp /tmp/jena_adapter/jar/*.jar   /tmp/joseki.war/WEB-INF/lib
    ##
    ## Assume ORACLE_HOME points to the home directory of a Release 11.2.0.3 Oracle Database.
    ##
    cp $ORACLE_HOME/md/jlib/sdordf.jar /tmp/joseki.war/WEB-INF/lib
    cp $ORACLE_HOME/jdbc/lib/ojdbc6.jar /tmp/joseki.war/WEB-INF/lib
    
  9. Using the WebLogic Server Administration console, create a J2EE data source named OracleSemDS. During the data source creation, you can specify a user and password for the database schema that contains the relevant semantic data against which SPARQL queries are to be executed.

    If you need help in creating this data source, see Section 7.2.1, "Creating the Required Data Source Using WebLogic Server".

  10. Go to the autodeploy directory of WebLogic Server and copy files, as follows. (For information about auto-deploying applications in development domains, see: http://download.oracle.com/docs/cd/E11035_01/wls100/deployment/autodeploy.html)

    cd <domain_name>/autodeploy
    cp -rf  /tmp/joseki.war  <domain_name>/autodeploy
    

    In the preceding example, <domain_name> is the name of a WebLogic Server domain.

    Note that while you can run a WebLogic Server domain in two different modes, development and production, only development mode allows you use the auto-deployment feature.

  11. Check the files and the directory structure to see if they reflect the following:

    .
    |-- META-INF
    |   `-- MANIFEST.MF
    |-- StyleSheets
    |   `-- joseki.css
    |-- WEB-INF
    |   |-- lib
    |   |   |-- arq-2.8.8-tests.jar
    |   |   |-- arq-2.8.8.jar
    |   |   |-- icu4j-3.4.4.jar
    |   |   |-- iri-0.8.jar
    |   |   |-- jena-2.6.4-tests.jar
    |   |   |-- jena-2.6.4.jar
    |   |   |-- jetty-6.1.25.jar
    |   |   |-- jetty-util-6.1.25.jar
    |   |   |-- joseki-3.4.4.jar
    |   |   |-- junit-4.5.jar
    |   |   |-- log4j-1.2.14.jar
    |   |   |-- lucene-core-2.3.1.jar
    |   |   |-- ojdbc6.jar
    |   |   |-- sdb-1.3.4.jar
    |   |   |-- sdordf.jar
    |   |   |-- sdordfclient.jar
    |   |   |-- servlet-api-2.5-20081211.jar
    |   |   |-- slf4j-api-1.5.8.jar
    |   |   |-- slf4j-log4j12-1.5.8.jar
    |   |   |-- stax-api-1.0.1.jar
    |   |   |-- tdb-0.8.10.jar
    |   |   |-- wstx-asl-3.2.9.jar
    |   |   `-- xercesImpl-2.7.1.jar
    |   `-- web.xml
    |-- application.xml
    |-- index.html
    |-- joseki-config.ttl
    |-- update.html
    `-- xml-to-html.xsl
    
  12. If you want to build a .war file from the /tmp/joseki.war directory (note that a .war file is required if you want to deploy Joseki to an OC4J container), enter the following commands:

    cd /tmp/joseki.war
    jar cvf /tmp/joseki_app.war *
    
  13. Start or restart WebLogic Server.

  14. Verify your deployment by using your Web browser to connect to a URL in the following format (assume that the Web application is deployed at port 7001): http://<hostname>:7001/joseki

    You should see a page titled Oracle SPARQL Service Endpoint using Joseki, and the first text box should contain an example SPARQL query.

  15. Click Submit Query.

    You should see a page titled Oracle SPARQL Endpoint Query Results. There may or may not be any results, depending on the underlying semantic model against which the query is executed.

By default, the joseki-config.ttl file contains an oracle:Dataset definition using a model named M_NAMED_GRAPHS. The following snippet shows the configuration. The oracle:allGraphs predicate denotes that the SPARQL service endpoint will serve queries using all graphs stored in the M_NAMED_GRAPHS model.

<#oracle> rdf:type oracle:Dataset;
    joseki:poolSize     1 ;         ## Number of concurrent connections allowed to this dataset.
    oracle:connection
    [ a oracle:OracleConnection ;
    ];
    oracle:allGraphs [ oracle:firstModel "M_NAMED_GRAPHS" ] .

The M_NAMED_GRAPHS model will be created automatically (if it does not already exist) upon the first SPARQL query request. You can add a few example triples and quads to test the named graph functions; for example:

SQL> CONNECT username/password
SQL> INSERT INTO m_named_graphs_tpl VALUES(sdo_rdf_triple_s('m_named_graphs','<urn:s>','<urn:p>','<urn:o>'));
SQL> INSERT INTO m_named_graphs_tpl VALUES(sdo_rdf_triple_s('m_named_graphs:<urn:G1>','<urn:g1_s>','<urn:g1_p>','<urn:g1_o>'));
SQL> INSERT INTO m_named_graphs_tpl VALUES(sdo_rdf_triple_s('m_named_graphs:<urn:G2>','<urn:g2_s>','<urn:g2_p>','<urn:g2_o>'));
SQL> COMMIT;

After inserting the rows, go to http://<hostname>:7001/joseki, type the following SPARQL query, and click Submit Query:

SELECT ?g ?s ?p ?o
WHERE 
  { GRAPH ?g { ?s ?p ?o} }

The result should be an HTML table with four columns and two sets of result bindings.

The http://<hostname>:7001/joseki page also contains a JSON Output option. If this option is selected (enabled), the SPARQL query response is converted to JSON format.

7.2.1 Creating the Required Data Source Using WebLogic Server

If you need help creating the required J2EE data source using the WebLogic Server admin console, you can follow these steps:

  1. Login to: http://<hostname>:7001/console

  2. In the Domain Structure panel, click Services.

  3. Click JDBC

  4. Click Data Sources.

  5. In the Summary of JDBC Data Sources panel, click New under the Data Sources table.

  6. In the Create a New JDBC Data Source panel, enter or select the following values.

    Name: OracleSemDS

    JNDI Name: OracleSemDS

    Database Type: Oracle

    Database Driver: Oracle's Driver (Thin) Versions: 9.0.1,9.2.0,10,11

  7. Click Next twice.

  8. In the Connection Properties panel, enter the appropriate values for the Database Name, Host Name, Port, Database User Name (schema that contains semantic data), Password fields.

  9. Click Next.

  10. Select (check) the target server or servers to which you want to deploy this OracleSemDS data source.

  11. Click Finish.

    You should see a message that all changes have been activated and no restart is necessary.

7.2.2 Configuring the SPARQL Service

By default, the SPARQL Service endpoint assumes that the queries are to be executed against a semantic model with a pre-set name. This semantic model is owned by the schema specified in the J2EE data source with JNDI name OracleSemDS. Note that you do not need to create this model explicitly using PL/SQL or Java; if the model does not exist in the network, it will be automatically created, along with the necessary application table and index.


Note:

Effective with the Jena Adapter release in November 2011, the application table index (<model_name>_idx) definition is changed to accommodate named graph data (quads).

For existing models created by an older version of the Jena Adapter, you can migrate the application table index name and definition by using the static OracleUtils.migrateApplicationTableIndex(oracle, graph, dop) method in the oracle.spatial.rdf.client.jena package. (See the Javadoc for more information.) Note that the new index definition is critical to the performance of DML operations against the application table.


However, you must configure the SPARQL service by editing the joseki-config.ttl configuration file, which is in <domain_name>/autodeploy/joseki.war.

The supplied joseki-config.ttl file includes a section similar to the following for the Oracle data set:

#
## Datasets
#
[] ja:loadClass "oracle.spatial.rdf.client.jena.assembler.OracleAssemblerVocab" .
 
oracle:Dataset  rdfs:subClassOf  ja:RDFDataset .
 
<#oracle> rdf:type oracle:Dataset;
    joseki:poolSize     1 ;         ## Number of concurrent connections allowed to this dataset.
    oracle:connection
    [ a oracle:OracleConnection ;
    ];
    oracle:defaultModel [ oracle:firstModel "TEST_MODEL" ] .

In this section of the file, you can:

  • Modify the joseki:poolSize value, which specifies the number of concurrent connections allowed to this Oracle data set (<#oracle> rdf:type oracle:Dataset;), which points to various RDF models in the database.

  • Modify the name (or the object value of oracle:firstModel predicate) of the defaultModel, to use a different semantic model for queries. You can also specify multiple models, and one or more rulebases for this defaultModel.

    For example, the following specifies two models (named ABOX and TBOX) and an OWLPRIME rulebase for the default model. Note that models specified using the oracle:modelName predicate must exist; they will not be created automatically.

    <#oracle> rdf:type oracle:Dataset;
        joseki:poolSize 1 ; ## Number of concurrent connections allowed to this dataset.
        oracle:connection
        [ a oracle:OracleConnection ;
        ];
        oracle:defaultModel [ oracle:firstModel "ABOX";
                                          oracle:modelName "TBOX";
                                          oracle:rulebaseName "OWLPRIME" ] .
    
  • Specify named graphs in the dataset. For example, you can create a named graph called <http://G1> based on two Oracle models and an entailment, as follows.

    <#oracle> rdf:type oracle:Dataset;
        joseki:poolSize 1 ; ## Number of concurrent connections allowed to this dataset.
          oracle:connection
          [ a oracle:OracleConnection ;
          ];
          oracle:namedModel [ oracle:firstModel "ABOX";
                              oracle:modelName "TBOX";
                              oracle:rulebaseName "OWLPRIME";
                              oracle:namedModelURI <http://G1> ]  .
    

    The object of namedModel can take the same specifications as defaultModel, so virtual models are supported here as well (see also the next item).

  • Use a virtual model for queries by adding oracle:useVM "TRUE", as shown in the following example. Note that if the specified virtual model does not exist, it will automatically be created on demand.

    <#oracle> rdf:type oracle:Dataset;
        joseki:poolSize 1 ; ## Number of concurrent connections allowed to this dataset.
        oracle:connection
        [ a oracle:OracleConnection ;
        ];
        oracle:defaultModel [ oracle:firstModel "ABOX";
                                         oracle:modelName "TBOX";
                                         oracle:rulebaseName "OWLPRIME";
                                         oracle:useVM "TRUE"
        ] .
    

    For more information, see Section 7.10.1, "Virtual Models Support".

  • Specify a virtual model as the default model to answer SPARQL queries by using the predicate oracle:virtualModelName, as shown in the following example with a virtual model named TRIPLE_DATA_VM_0:

    oracle:defaultModel [ oracle:virtualModelName "TRIPLE_DATA_VM_0" ] .
    

    If the underlying data consists of quads, you can use oracle:virtualModelName with oracle:allGraphs. The presence of oracle:allGraphs causes an instantiation of DatasetGraphOracleSem objects to answer named graph queries. An example is as follows:

    oracle:allGraphs [ oracle:virtualModelName "QUAD_DATA_VM_0" ] .
    

    Note that when a virtual model name is specified as the default graph, the endpoint can serve only query requests; SPARQL Update operations are not supported.

  • Set the queryOptions and inferenceMaintenance properties to change the query behavior and inference update mode. (See the Javadoc for information about QueryOptions and InferenceMaintenanceMode.)

    By default, QueryOptions.ALLOW_QUERY_INVALID_AND_DUP and InferenceMaintenanceMode.NO_UPDATE are set, for maximum query flexibility and efficiency.

7.2.2.1 Client Identifiers

For every database connection created or used by the Jena Adapter, a client identifier is associated with the connection. The client identifier can be helpful, especially in a Real Application Cluster (Oracle RAC) environment, for isolating Jena Adapter-related activities from other database activities when you are doing performance analysis and tuning.

By default, the client identifier assigned is JenaAdapter. However, you can specify a different value by setting the Java VM clientIdentifier property using the following format:

-Doracle.spatial.rdf.client.jena.clientIdentifier=<identificationString>

To start the tracing of only Jena Adapter-related activities on the database side, you can use the DBMS_MONITOR.CLIENT_ID_TRACE_ENABLE procedure. For example:

SQL> EXECUTE DBMS_MONITOR.CLIENT_ID_TRACE_ENABLE('JenaAdapter', true, true);

7.2.2.2 Using OLTP Compression for Application Tables and Staging Tables

By default, the Jena Adapter creates the application tables and any staging tables (the latter used for bulk loading, as explained in Section 7.11) using basic table compression with the following syntax:

CREATE TABLE .... (... column definitions ...) ... compress;

However, if you are licensed to use the Oracle Advanced Compression option no the database, you can set the following JVM property to turn on OLTP compression, which compresses data during all DML operations against the underlying application tables and staging tables:

-Doracle.spatial.rdf.client.jena.advancedCompression="compress for oltp"

7.2.3 Terminating Long-Running SPARQL Queries

Because some applications need to be able to terminate long-running SPARQL queries, an abort framework has been introduced with the Jena Adapter and the Joseki setup. Basically, for queries that may take a long time to run, you must stamp each with a unique query ID (qid) value.

For example, the following SPARQL query selects out the subject of all triples. A query ID (qid) is set so that this query can be terminated upon request.

PREFIX ORACLE_SEM_FS_NS:  <http://example.com/semtech#qid=8761>
SELECT ?subject WHERE {?subject ?property ?object }

The qid attribute value is of long integer type. You can choose a value for the qid for a particular query based on your own application needs.

To terminate a SPARQL query that has been submitted with a qid value, applications can send an abort request to a servlet in the following format and specify a matching QID value

http://<hostname>:7001/joseki/querymgt?abortqid=8761

7.2.4 N-Triples Encoding for Non-ASCII Characters

For any non-ASCII characters in the lexical representation of RDF resources, \uHHHH N-Triples encoding is used when the characters are inserted into the Oracle database. (For details about N-Triples encoding, see http://www.w3.org/TR/rdf-testcases/#ntrip_grammar.) Encoding of the constant resources in a SPARQL query is handled in a similar fashion.

Using \uHHHH N-Triples encoding enables support for international characters, such as a mix of Norwegian and Swedish characters, in the Oracle database even if a supported Unicode character set is not being used.

7.3 Setting Up the Semantic Technologies Environment

To use the Jena Adapter to perform queries, you can connect as any user (with suitable privileges) and use any models in the semantic network. If your Semantic Technologies environment already meets the requirements, you can go directly to compiling and running Java code that uses the Jena Adapter. If your Semantic Technologies environment is not yet set up to be able to use the Jena Adapter, you can perform actions similar to the following example steps:

  1. Connect as SYS with the SYSDBA role:

    sqlplus sys/<password-for-sys> as sysdba
    
  2. Create a tablespace for the system tables. For example:

    CREATE TABLESPACE rdf_users datafile 'rdf_users01.dbf' 
        size 128M reuse autoextend on next 64M 
        maxsize unlimited segment space management auto;
    
  3. Create the semantic network. For example:

    EXECUTE sem_apis.create_sem_network('RDF_USERS');
    
  4. Create a database user (for connecting to the database to use the semantic network and the Jena Adapter). For example:

    CREATE USER rdfusr IDENTIFIED BY <password-for-udfusr>
                       DEFAULT TABLESPACE rdf_users;
    
  5. Grant the necessary privileges to this database user. For example:

    GRANT connect, resource TO rdfusr;
    
  6. To use the Jena Adapter with your own semantic data, perform the appropriate steps to store data, create a model, and create database indexes, as explained in Section 1.10, "Quick Start for Using Semantic Data". Then perform queries by compiling and running Java code; see Section 7.15 for information about example queries.

    To use the Jena Adapter with supplied example data, see Section 7.15.

7.4 SEM_MATCH and Jena Adapter Queries Compared

There are two ways to query semantic data stored in Oracle Database: SEM_MATCH-based SQL statements and SPARQL queries through the Jena Adapter. Queries using each approach are similar in appearance, but there are important behavioral differences. To ensure consistent application behavior, you must understand the differences and use care when dealing with query results coming from SEM_MATCH queries and SPARQL queries.

The following simple examples show the two approaches.

Query 1 (SEM_MATCH-based)

select s, p, o
    from table(sem_match('{?s ?p ?o}', sem_models('Test_Model'), ....))

Query 2 (SPARQL query through the Jena Adapter)

select ?s ?p ?o
where {?s ?p ?o}

These two queries perform the same kind of functions; however, there are some important differences. Query 1 (SEM_MATCH-based):

Query 2 (SPARQL query executed through the Jena Adapter) also reads all triples out of Test_Model (assume it executed a call to ModelOracleSem referring to the same underlying Test_Model). However, Query 2:

Blank node handling is another difference between the two approaches:

The maximum length for the name of a semantic model created using the Jena Adapter API is 22 characters.

7.5 Optimized Handling of SPARQL Queries

This section describes some performance-related features of the Jena Adapter that can enhance SPARQL query processing. These features are performed automatically by default.

This section assumes that you are familiar with SPARQL, including the CONSTRUCT feature and property paths.

7.5.1 Compilation of SPARQL queries to a single SEM_MATCH Call

SPARQL queries involving DISTINCT, OPTIONAL, FILTER, UNION, ORDER BY, and LIMIT are converted to a single Oracle SEM_MATCH table function. If a query cannot be converted directly to SEM_MATCH because it uses SPARQL features not supported by SEM_MATCH (for example, CONSTRUCT), the Jena Adapter employs a hybrid approach and tries to execute the largest portion of the query using a single SEM_MATCH function while executing the rest using the Jena ARQ query engine.

For example, the following SPARQL query is directly translated to a single SEM_MATCH table function:

PREFIX dc:  <http://purl.org/dc/elements/1.1/> 
PREFIX rdf:  <http://www.w3.org/1999/02/22-rdf-syntax-ns#> 
PREFIX foaf: <http://xmlns.com/foaf/0.1/> 
SELECT ?person ?name 
  WHERE {
                 {?alice foaf:knows ?person . }
               UNION { 
                 ?person ?p ?name. OPTIONAL { ?person ?x ?name1 } 
                     }
        }

However, the following example query is not directly translatable to a single SEM_MATCH table function because of the CONSTRUCT keyword:

PREFIX vcard: <http://www.w3.org/2001/vcard-rdf/3.0#> 
CONSTRUCT  { <http://example.org/person#Alice> vcard:FN ?obj } 
  WHERE  { { ?x <http://pred/a> ?obj.}
         UNION
         { ?x <http://pred/b> ?obj.}  }

In this case, the Jena Adapter converts the inner UNION query into a single SEM_MATCH table function, and then passes on the result set to the Jena ARQ query engine for further evaluation.

7.5.2 Optimized Handling of Property Paths

As defined in Jena, a property path is a possible route through an RDF graph between two graph nodes. Property paths are an extension of SPARQL and are more expressive than basic graph pattern queries, because regular expressions can be used over properties for pattern matching RDF graphs. For more information about property paths, see the documentation for the Jena ARQ query engine.

The Jena Adapter supports all Jena property path types through the integration with the Jena ARQ query engine, but it converts some common path types directly to native SQL hierarchical queries (not based on SEM_MATCH) to improve performance. The following types of property paths are directly converted to SQL by the Jena Adapter when dealing with triple data:

  • Predicate alternatives: (p1 | p2 | … | pn) where pi is a property URI

  • Predicate sequences: (p1 / p2 / … / pn) where pi is a property URI

  • Reverse paths : ( ^ p ) where p is a predicate URI

  • Complex paths: p+, p*, p{0, n} where p could be an alternative, sequence, reverse path, or property URI

Path expressions that cannot be captured in this grammar are not translated directly to SQL by the Jena Adapter, and they are answered using the Jena query engine.

The following example contains a code snippet using a property path expression with path sequences:

String m = "PROP_PATH";
 
ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, m);
 
GraphOracleSem graph = new GraphOracleSem(oracle, m);    
    
// populate the RDF Graph
    graph.add(Triple.create(Node.createURI("http://a"),
    Node.createURI("http://p1"),
    Node.createURI("http://b")));
 
graph.add(Triple.create(Node.createURI("http://b"),
 Node.createURI("http://p2"),
 Node.createURI("http://c")));
 
graph.add(Triple.create(Node.createURI("http://c"),
 Node.createURI("http://p5"),
 Node.createURI("http://d")));
 
String query =
" SELECT ?s  " +
" WHERE {?s (<http://p1>/<http://p2>/<http://p5>)+ <http://d>.}";
   
QueryExecution qexec = 
      QueryExecutionFactory.create(QueryFactory.create(query, 
 Syntax.syntaxARQ), model);
 
try {
  ResultSet results = qexec.execSelect();
  ResultSetFormatter.out(System.out, results);
}
finally {
  if (qexec != null)
    qexec.close();
}
     
OracleUtils.dropSemanticModel(oracle, m);
model.close();

7.6 Additions to the SPARQL Syntax to Support Other Features

The Jena Adapter allows you to pass in hints and additional query options. It implements these capabilities by overloading the SPARQL namespace prefix syntax by using Oracle-specific namespaces that contain query options. The namespaces are in the form PREFIX ORACLE_SEM_xx_NS, where xx indicates the type of feature (such as HT for hint or AP for additional predicate)

7.6.1 SQL Hints

SQL hints can be passed to a SEM_MATCH query including a line in the following form:

PREFIX ORACLE_SEM_HT_NS: <http://oracle.com/semtech#hint>

Where hint can be any hint supported by SEM_MATCH. For example:

PREFIX ORACLE_SEM_HT_NS: <http://oracle.com/semtech#leading(t0,t1)> 
SELECT ?book ?title ?isbn     
WHERE { ?book <http://title> ?title. ?book <http://ISBN> ?isbn }

In this example, t0,t1 refers to the first and second patterns in the query.

Note the slight difference in specifying hints when compared to SEM_MATCH. Due to restrictions of namespace value syntax, a comma (,) must be used to separate t0 and t1 (or other hint components) instead of a space.

For more information about using SQL hints, see Section 1.6, "Using the SEM_MATCH Table Function to Query Semantic Data", specifically the material about the HINT0 keyword in the options attribute.

7.6.2 Using Bind Variables in SPARQL Queries

In Oracle Database, using bind variables can reduce query parsing time and increase query efficiency and concurrency. Bind variable support in SPARQL queries is provided through namespace pragma specifications similar to ORACLE_SEM_FS_NS.

Consider a case where an application runs two SPARQL queries, where the second (Query 2) depends on the partial or complete results of the first (Query 1). Some approaches that do not involve bind variables include:

  • Iterating through results of Query 1 and generating a set of queries. (However, this approach requires as many queries as the number of results of Query 1.)

  • Constructing a SPARQL filter expression based on results of Query 1.

  • Treating Query 1 as a subquery.

Another approach in this case is to use bind variables, as in the following sample scenario:

Query 1:
 
  SELECT ?x
    WHERE { ... <some complex query> ... };
 
 
Query 2:
 
  SELECT ?subject ?x
    WHERE {?subject <urn:related> ?x .};

The following example shows Query 2 with the syntax for using bind variables with the Jena Adapter:

PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#no_fall_back,s2s>
PREFIX ORACLE_SEM_UEAP_NS: <http://oracle.com/semtech#x$RDFVID%20in(?,?,?)>
PREFIX ORACLE_SEM_UEPJ_NS: <http://oracle.com/semtech#x$RDFVID>
PREFIX ORACLE_SEM_UEBV_NS: <http://oracle.com/semtech#1,2,3>
SELECT ?subject ?x
WHERE {
  ?subject <urn:related>  ?x
};

This syntax includes using the following namespaces:

  • ORACLE_SEM_UEAP_NS is like ORACLE_SEM_AP_NS, but the value portion of ORACLE_SEM_UEAP_NS is URL Encoded. Before the value portion is used, it must be URL decoded, and then it will be treated as an additional predicate to the SPARQL query.

    In this example, after URL decoding, the value portion (following the # character) of this ORACLE_SEM_UEAP_NS prefix becomes "x$RDFVID in(?,?,?)". The three question marks imply a binding to three values coming from Query 1.

  • ORACLE_SEM_UEPJ_NS specifies the additional projections involved. In this case, because ORACLE_SEM_UEAP_NS references the x$RDFVID column, which does not appear in the SELECT clause of the query, it must be specified. Multiple projections are separated by commas.

  • ORACLE_SEM_UEBV_NS specifies the list of bind values that are URL encoded first, and then concatenated and delimited by commas.

Conceptually, the preceding example query is equivalent to the following non-SPARQL syntax query, in which 1, 2, and 3 are treated as bind values:

SELECT ?subject ?x
  WHERE {
    ?subject <urn:related>  ?x
  }
  AND ?x$RDFVID in (1,2,3);

In the preceding SPARQL example of Query 2, the three integers 1, 2, and 3 come from Query 1. You can use the oext:build-uri-for-id function to generate such internal integer IDs for RDF resources. The following example gets the internal integer IDs from Query 1:

PREFIX oext: <http://oracle.com/semtech/jena-adaptor/ext/function#>
SELECT ?x  (oext:build-uri-for-id(?x) as ?xid)
WHERE { ... <some complex query> ... };

The values of ?xid have the form of <rdfvid:integer-value>. The application can strip out the angle brackets and the "rdfvid:" strings to get the integer values and pass them to Query 2.

Consider another case, with a single query structure but potentially many different constants. For example, the following SPARQL query finds the hobby for each user who has a hobby and who logs in to an application. Obviously, different users will provide different <uri> values to this SPARQL query, because users of the application are represented using different URIs.

SELECT ?hobby
  WHERE { <uri> <urn:hasHobby> ?hobby };

One approach, which would not use bind variables, is to generate a different SPARQL query for each different <uri> value. For example, user Jane Doe might trigger the execution of the following SPARQL query:

SELECT ?hobby WHERE {
<http://www.example.com/Jane_Doe> <urn:hasHobby> ?hobby };

However, another approach is to use bind variables, as in the following example specifying user Jane Doe:

PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#no_fall_back,s2s>
PREFIX ORACLE_SEM_UEAP_NS: <http://oracle.com/semtech#subject$RDFVID%20in(ORACLE_ORARDF_RES2VID(?))>
PREFIX ORACLE_SEM_UEPJ_NS: <http://oracle.com/semtech#subject$RDFVID>
PREFIX ORACLE_SEM_UEBV_NS: <http://oracle.com/semtech#http%3a%2f%2fwww.example.com%2fJohn_Doe>
SELECT ?subject ?hobby
  WHERE {
    ?subject <urn:hasHobby>  ?hobby
  };

Conceptually, the preceding example query is equivalent to the following non-SPARQL syntax query, in which http://www.example.com/Jane_Doe is treated as a bind variable:

SELECT ?subject ?hobby
WHERE {
  ?subject <urn:hasHobby>  ?hobby
}
AND ?subject$RDFVID in (ORACLE_ORARDF_RES2VID('http://www.example.com/Jane_Doe'));

In this example, ORACLE_ORARDF_RES2VID is a function that translates URIs and literals into their internal integer ID representation. This function is created automatically when the Jena Adapter is used to connect to an Oracle database.

7.6.3 Additional WHERE Clause Predicates

The SEM_MATCH filter attribute can specify additional selection criteria as a string in the form of a WHERE clause without the WHERE keyword. Additional WHERE clause predicates can be passed to a SEM_MATCH query including a line in the following form:

PREFIX ORACLE_SEM_AP_NS: <http://oracle.com/semtech#pred>

Where pred reflects the WHERE clause content to be appended to the query. For example:

PREFIX rdfs: <http://www.w3.org/2000/01/rdf-schema#>
PREFIX ORACLE_SEM_AP_NS:<http://www.oracle.com/semtech#label$RDFLANG='fr'>  
SELECT DISTINCT ?inst ?label
  WHERE { ?inst a <http://someCLass>. ?inst rdfs:label ?label . }
  ORDER BY (?label) LIMIT 20

In this example, a restriction is added to the query that the language type of the label variable must be 'fr'.

7.6.4 Additional Query Options

Additional query options can be passed to a SEM_MATCH query including a line in the following form:

PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#option>

Where option reflects a query option (or multiple query options delimited by commas) to be appended to the query. For example:

PREFIX ORACLE_SEM_FS_NS:   
<http://oracle.com/semtech#timeout=3,dop=4,INF_ONLY,ORDERED,ALLOW_DUP=T>
SELECT * WHERE {?subject ?property ?object }

The following query options are supported:

  • ALLOW_DUP=t chooses a faster way to query multiple semantic models, although duplicate results may occur.

  • BEST_EFFORT_QUERY=t, when used with the TIMEOUT=n option, returns all matches found in n seconds for the SPARQL query.

  • DEGREE=n specifies, at the statement level, the degree of parallelism (n) for the query. With multi-core or multi-CPU processors, experimenting with different DOP values (such as 4 or 8) may improve performance.

    Contrast DEGREE with DOP, which specifies parallelism at the session level. DEGREE is recommended over DOP for use with the Jena Adapter, because DEGREE involves less processing overhead.

  • DOP=n specifies, at the session level, the degree of parallelism (n) for the query. With multi-core or multi-CPU processors, experimenting with different DOP values (such as 4 or 8) may improve performance.

  • INF_ONLY causes only the inferred model to be queried.

  • JENA_EXECUTOR disables the compilation of SPARQL queries to SEM_MATCH (or native SQL); instead, the Jena native query executor will be used.

  • JOIN=n specifies how results from a SPARQL SERVICE call to a federated query can be joined with other parts of the query. For information about federated queries and the JOIN option, see Section 7.6.4.1.

  • NO_FALL_BACK causes the underlying query execution engine not to fall back on the Jena execution mechanism if a SQL exception occurs.

  • ODS=n specifies, at the statement level, the level of dynamic sampling. (For an explanation of dynamic sampling, see the section about estimating statistics with dynamic sampling in Oracle Database Performance Tuning Guide.) Valid values for n are 1 through 10. For example, you could try ODS=3 for complex queries.

  • ORDERED is translated to a LEADING SQL hint for the query triple pattern joins, while performing the necessary RDF_VALUE$ joins last.

  • PLAIN_SQL_OPT=F disables the native compilation of queries directly to SQL.

  • QID=n specifies a query ID number; this feature can be used to cancel the query if it is not responding.

  • RESULT_CACHE uses the Oracle RESULT_CACHE directive for the query.

  • REWRITE=F disables ODCI_Table_Rewrite for the SEM_MATCH table function.

  • SKIP_CLOB=T causes CLOB values not to be returned for the query.

  • S2S (SPARQL to pure SQL) causes the underlying SEM_MATCH-based query or queries generated based on the SPARQL query to be further converted into SQL queries without using the SEM_MATCH table function. The resulting SQL queries are executed by the Oracle cost-based optimizer, and the results are processed by the Jena Adapter before being passed on to the client. For more information about the S2S option, including benefits and usage information, see Section 7.6.4.2.

    S2S is enabled by default for all SPARQL queries. If you want to disable S2S, set the following JVM system property:

    -Doracle.spatial.rdf.client.jena.defaultS2S=false
    
  • TIMEOUT=n (query timeout) specifies the number of seconds (n) that the query will run until it is terminated. The underlying SQL generated from a SPARQL query can return many matches and can use features like subqueries and assignments, all of which can take considerable time. The TIMEOUT and BEST_EFFORT_QUERY=t options can be used to prevent what you consider excessive processing time for the query.

7.6.4.1 JOIN Option and Federated Queries

A SPARQL federated query, as described in W3C documents, is a query "over distributed data" that entails "querying one source and using the acquired information to constrain queries of the next source." For more information, see SPARQL 1.1 Federation Extensions (http://www.w3.org/2009/sparql/docs/fed/service).

You can use the JOIN option (described in Section 7.6.4) and the SERVICE keyword in a federated query that uses the Jena Adapter. For example, assume the following query:

SELECT ?s ?s1 ?o
 WHERE { ?s1 ?p1 ?s .
                    {
                     SERVICE <http://sparql.org/books> { ?s ?p ?o }
                    }
                 }

If the local query portion (?s1 ?p1 ?s,) is very selective, you can specify join=2, as shown in the following query:

PREFIX ORACLE_SEM_FS_NS:   <http://oracle.com/semtech#join=2>
SELECT ?s ?s1 ?o
 WHERE { ?s1 ?p1 ?s .
                    {
                     SERVICE <http://sparql.org/books> { ?s ?p ?o }
                    }
                 }

In this case, the local query portion (?s1 ?p1 ?s,) is executed locally against the Oracle database. Each binding of ?s from the results is then pushed into the SERVICE part (remote query portion), and a call is made to the service endpoint specified. Conceptually, this approach is somewhat like nested loop join.

If the remote query portion (?s ?s1 ?o) is very selective, you can specify join=3, as shown in the following query, so that the remote portion is executed first and results are used to drive the execution of local portion:

PREFIX ORACLE_SEM_FS_NS:   <http://oracle.com/semtech#join=3>
SELECT ?s ?s1 ?o
 WHERE { ?s1 ?p1 ?s .
                    {
                     SERVICE <http://sparql.org/books> { ?s ?p ?o }
                    }
                  }

In this case, a single call is made to the remote service endpoint and each binding of ?s triggers a local query. As with join=2, this approach is conceptually a nested loop based join, but the difference is that the order is switched.

If neither the local query portion nor the remote query portion is very selective, then we can choose join=1, as shown in the following query:

PREFIX ORACLE_SEM_FS_NS:   <http://oracle.com/semtech#join=1>
SELECT ?s ?s1 ?o
 WHERE { ?s1 ?p1 ?s .
                    {
                     SERVICE <http://sparql.org/books> { ?s ?p ?o }
                    }
                }

In this case, the remote query portion and the local portion are executed independently, and the results are joined together by Jena. Conceptually, this approach is somewhat like a hash join.

For debugging or tracing federated queries, you can use the HTTP Analyzer in Oracle JDeveloper to see the underlying SERVICE calls.

7.6.4.2 S2S Option Benefits and Usage Information

The S2S option, described in Section 7.6.4, provides the following potential benefits:

  • It works well with the RESULT_CACHE option to improve query performance. Using the S2S and RESULT_CACHE options is especially helpful for queries that are executed frequently.

  • It reduces the parsing time of the SEM_MATCH table function, which can be helpful for applications that involve many dynamically generated SPARQL queries.

  • It eliminates the limit of 4000 bytes for the query body (the first parameter of the SEM_MATCH table function), which means that longer, more complex queries are supported.

The S2S option causes an internal in-memory cache to be used for translated SQL query statements. The default size of this internal cache is 1024 (that is, 1024 SQL queries); however, you can adjust the size by using the following Java VM property:

-Doracle.spatial.rdf.client.jena.queryCacheSize=<size>

7.6.5 Midtier Resource Caching

When semantic data is stored, all of the resource values are hashed into IDs, which are stored in the triples table. The mappings from value IDs to full resource values are stored in the MDSYS.RDF_VALUE$ table. At query time, for each selected variable, Oracle Database must perform a join with the RDF_VALUE$ table to retrieve the resource.

However, to reduce the number of joins, you can use the midtier cache option, which causes an in-memory cache on the middle tier to be used for storing mappings between value IDs and resource values. To use this feature, include the following PREFIX pragma in the SPARQL query:

PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#midtier_cache>

To control the maximum size (in bytes) of the in-memory cache, use the oracle.spatial.rdf.client.jena.cacheMaxSize system property. The default cache maximum size is 1GB.

Note that midtier resource caching is most effective for queries using ORDER BY or DISTINCT (or both) constructs, or queries with multiple projection variables. Midtier cache can be combined with the other options specified in Section 7.6.4.

If you want to pre-populate the cache with all of the resources in a model, use the GraphOracleSem.populateCache or DatasetGraphOracleSem.populateCache method. Both methods take a parameter specifying the number of threads used to build the internal midtier cache. Running either method in parallel can significantly increase the cache building performance on a machine with multiple CPUs (cores).

7.7 Functions Supported in SPARQL Queries through the Jena Adapter

SPARQL queries through the Jena Adapter can use the following kinds of functions:

7.7.1 Functions in the ARQ Function Library

SPARQL queries through the Jena Adapter can use functions in the function library of the Jena ARQ query engine. These queries are executed in the middle tier.

The following examples use the upper-case and namespace functions. In these examples, the prefix fn is <http://www.w3.org/2005/xpath-functions#> and the prefix afn is <http://jena.hpl.hp.com/ARQ/function#>.

PREFIX  fn: <http://www.w3.org/2005/xpath-functions#>
PREFIX  afn: <http://jena.hpl.hp.com/ARQ/function#>
SELECT (fn:upper-case(?object) as ?object1)
WHERE { ?subject dc:title ?object }

PREFIX  fn: <http://www.w3.org/2005/xpath-functions#>
PREFIX  afn: <http://jena.hpl.hp.com/ARQ/function#>
SELECT ?subject (afn:namespace(?object) as ?object1)
WHERE { ?subject <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> ?object } 

7.7.2 Native Oracle Database Functions for Projected Variables

SPARQL queries through the Jena Adapter can use native Oracle Database functions for projected variables. These queries and the functions are executed inside the database. Note that the functions described in this section should not be used together with ARQ functions (described in Section 7.7.1).

This section lists the supported native functions and provides some examples. In the examples, the prefix oext is <http://oracle.com/semtech/jena-adaptor/ext/function#>.


Note:

In the preceding URL, note the spelling jena-adaptor, which is retained for compatibility with existing applications and which must be used in queries. The adapter spelling is used in regular text, to follow Oracle documentation style guidelines.

  • oext:upper-literal converts literal values (except for long literals) to uppercase. For example:

    PREFIX  oext: <http://oracle.com/semtech/jena-adaptor/ext/function#>
    SELECT (oext:upper-literal(?object) as ?object1)
    WHERE { ?subject dc:title ?object }
    
  • oext:lower-literal converts literal values (except for long literals) to lowercase. For example:

    PREFIX  oext: <http://oracle.com/semtech/jena-adaptor/ext/function#>
    SELECT (oext:lower-literal(?object) as ?object1)
    WHERE { ?subject dc:title ?object }
    
  • oext:build-uri-for-id converts the value ID of a URI, bNode, or literal into a URI form. For example:

    PREFIX  oext: <http://oracle.com/semtech/jena-adaptor/ext/function#>
    SELECT (oext:build-uri-for-id(?object) as ?object1)
    WHERE { ?subject dc:title ?object }
    

    An example of the output might be: <rdfvid:1716368199350136353>

    One use of this function is to allow Java applications to maintain in memory a mapping of those value IDs to the lexical form of URIs, bNodes, or literals. The MDSYS.RDF_VALUE$ table provides such a mapping in Oracle Database.

    For a given variable ?var, if only oext:build-uri-for-id(?var) is projected, the query performance is likely to be faster because fewer internal table join operations are needed to answer the query.

  • oext:literal-strlen returns the length of literal values (except for long literals). For example:

    PREFIX  oext: <http://oracle.com/semtech/jena-adaptor/ext/function#>
    SELECT (oext:literal-strlen(?object) as ?objlen)
    WHERE { ?subject dc:title ?object }
    

7.7.3 User-Defined Functions

SPARQL queries through the Jena Adapter can use user-defined functions that are stored in the database.

In the following example, assume that you want to define a string length function (my_strlen) that handles long literals (CLOB) as well as short literals. On the SPARQL query side, this function can be referenced under the namespace of ouext, which is http://oracle.com/semtech/jena-adaptor/ext/user-def-function#.

PREFIX  ouext: <http://oracle.com/semtech/jena-adaptor/ext/user-def-function#>
SELECT ?subject ?object (ouext:my_strlen(?object) as ?obj1)
WHERE { ?subject dc:title ?object }

Inside the database, functions including my_strlen, my_strlen_cl, my_strlen_la, my_strlen_lt, and my_strlen_vt are defined to implement this capability. Conceptually, the return values of these functions are mapped as shown in Table 7-1.

Table 7-1 Functions and Return Values for my_strlen Example

Function NameReturn Value

my_strlen

<VAR>

my_strlen_cl

<VAR>$RDFCLOB

my_strlen_la

<VAR>$RDFLANG

my_strlen_lt

<VAR>$RDFLTYP

my_strlen_vt

<VAR>$RDFVTYP


A set of functions (five in all) is used to implement a user-defined function that can be referenced from SPARQL, because this aligns with the internal representation of an RDF resource (in MDSYS.RDF_VALUE$). There are five major columns describing an RDF resource in terms of its value, language, literal type, long value, and value type, and these five columns can be selected out using SEM_MATCH. In this context, a user-defined function simply converts one RDF resource that is represented by five columns to another RDF resource.

These functions are defined as follows:

create or replace function my_strlen(rdfvtyp in varchar2,
                              rdfltyp in varchar2,
                              rdflang in varchar2,
                              rdfclob in clob,
                              value   in varchar2
                              ) return varchar2
 as
   ret_val  varchar2(4000);
 begin
   -- value
   if (rdfvtyp = 'LIT') then
     if (rdfclob is null) then
       return length(value);
     else
       return dbms_lob.getlength(rdfclob);
     end if;
   else
     -- Assign -1 for non-literal values so that application can
     -- easily differentiate
     return '-1';
   end if;
 end;
 /
 
 create or replace function my_strlen_cl(rdfvtyp in varchar2,
                              rdfltyp in varchar2,
                              rdflang in varchar2,
                              rdfclob in clob,
                              value   in varchar2
                              ) return clob
 as
 begin
   return null;
 end;
 /
 
 create or replace function my_strlen_la(rdfvtyp in varchar2,
                              rdfltyp in varchar2,
                              rdflang in varchar2,
                              rdfclob in clob,
                              value   in varchar2
                              ) return varchar2
 as
 begin
   return null;
 end;
 /
 
 create or replace function my_strlen_lt(rdfvtyp in varchar2,
                              rdfltyp in varchar2,
                              rdflang in varchar2,
                              rdfclob in clob,
                              value   in varchar2
                              ) return varchar2
 as
   ret_val  varchar2(4000);
 begin
   -- literal type
   return 'http://www.w3.org/2001/XMLSchema#integer';
 end;
 /
 
 create or replace function my_strlen_vt(rdfvtyp in varchar2,
                              rdfltyp in varchar2,
                              rdflang in varchar2,
                              rdfclob in clob,
                              value   in varchar2
                              ) return varchar2
 as
   ret_val  varchar2(3);
 begin
   return 'LIT';
 end;
 /

User-defined functions can also accept a parameter of VARCHAR2 type. The following five functions together define a my_shorten_str function that accepts an integer (in VARCHAR2 form) for the substring length and returns the substring. (The substring in this example is 12 characters, and it must not be greater than 4000 bytes.)

-- SPARQL query that returns the first 12 characters of literal values.
-- 
PREFIX  ouext: <http://oracle.com/semtech/jena-adaptor/ext/user-def-function#>
SELECT (ouext:my_shorten_str(?object, "12") as ?obj1) ?subject
WHERE { ?subject dc:title ?object }
 
create or replace function my_shorten_str(rdfvtyp in varchar2,
                            rdfltyp in varchar2,
                            rdflang in varchar2,
                            rdfclob in clob,
                            value   in varchar2,
                            arg     in varchar2
                            ) return varchar2
as
 ret_val  varchar2(4000);
begin
 -- value
 if (rdfvtyp = 'LIT') then
   if (rdfclob is null) then
     return substr(value, 1, to_number(arg));
   else
     return dbms_lob.substr(rdfclob, to_number(arg), 1);
   end if;
 else
   return null;
 end if;
end;
/
 
create or replace function my_shorten_str_cl(rdfvtyp in varchar2,
                            rdfltyp in varchar2,
                            rdflang in varchar2,
                            rdfclob in clob,
                            value   in varchar2,
                            arg     in varchar2
                            ) return clob
as
 ret_val  clob;
begin
 -- lob
 return null;
end;
/
 
create or replace function my_shorten_str_la(rdfvtyp in varchar2,
                            rdfltyp in varchar2,
                            rdflang in varchar2,
                            rdfclob in clob,
                            value   in varchar2,
                            arg     in varchar2
                            ) return varchar2
as
 ret_val  varchar2(4000);
begin
 -- lang
 if (rdfvtyp = 'LIT') then
   return rdflang;
 else
   return null;
 end if;
end;
/
 
create or replace function my_shorten_str_lt(rdfvtyp in varchar2,
                            rdfltyp in varchar2,
                            rdflang in varchar2,
                            rdfclob in clob,
                            value   in varchar2,
                            arg     in varchar2
                            ) return varchar2
as
 ret_val  varchar2(4000);
begin
 -- literal type
 ret_val := rdfltyp;
 return ret_val;
end;
/
 
create or replace function my_shorten_str_vt(rdfvtyp in varchar2,
                            rdfltyp in varchar2,
                            rdflang in varchar2,
                            rdfclob in clob,
                            value   in varchar2,
                            arg     in varchar2
                            ) return varchar2
as
 ret_val  varchar2(3);
begin
 return 'LIT';
end;
/

7.8 SPARQL Update Support

The Jena Adapter supports SPARQL Update (http://www.w3.org/TR/sparql11-update/), also referred to as SPARUL. The primary programming APIs involve the Jena class UpdateAction (in package com.hp.hpl.jena.update) and the Jena Adapter classes GraphOracleSem and DatasetGraphOracleSem. Example 7-1 shows a SPARQL Update operation removes all triples in named graph <http://example/graph> from the relevant model stored in the database.

Example 7-1 Simple SPARQL Update

GraphOracleSem graphOracleSem = .... ;
DatasetGraphOracleSem dsgos = DatasetGraphOracleSem.createFrom(graphOracleSem);
 
// SPARQL Update operation
String szUpdateAction = "DROP GRAPH <http://example/graph>";
 
// Execute the Update against a DatasetGraph instance (can be a Jena Model as well)
UpdateAction.parseExecute(szUpdateAction, dsgos);

Note that Oracle Database does not keep any information about an empty named graph. This implies if you invoke CREATE GRAPH <graph_name> without adding any triples into this graph, then no additional rows in the application table or the underlying RDF_LINK$ table will be created. To an Oracle database, you can safely skip the CREATE GRAPH step, as is the case in Example 7-1.

Example 7-2 shows a SPARQL Update operation (from ARQ 2.8.8) involving multiple insert and delete operations.

Example 7-2 SPARQL Update with Insert and Delete Operations

PREFIX : <http://example/>
CREATE GRAPH <http://example/graph> ;
INSERT DATA { :r :p 123 } ;
INSERT DATA { :r :p 1066 } ;
DELETE DATA { :r :p 1066 } ;
INSERT DATA {
  GRAPH <http://example/graph> { :r :p 123 . :r :p 1066 }
} ;
DELETE DATA {
  GRAPH <http://example/graph>  { :r :p 123 }
}

After running the update operation in Example 7-2 against an empty DatasetGraphOracleSem, running the SPARQL query SELECT ?s ?p ?o WHERE {?s ?p ?o} generates the following response:

-----------------------------------------------------------------------------------------------
| s                  | p                  | o                                                 |
===============================================================================================
| <http://example/r> | <http://example/p> | "123"^^<http://www.w3.org/2001/XMLSchema#decimal> |
-----------------------------------------------------------------------------------------------

Using the same data, running the SPARQL query SELECT ?g ?s ?p ?o where {GRAPH ?g {?s ?p ?o}} generates the following response:

-------------------------------------------------------------------------------------------------------------------------
| g                      | s                  | p                  | o                                                  |
=========================================================================================================================
| <http://example/graph> | <http://example/r> | <http://example/p> | "1066"^^<http://www.w3.org/2001/XMLSchema#decimal> |
-------------------------------------------------------------------------------------------------------------------------

In addition to using the Java API for SPARQL Update operations, you can configure Joseki to accept SPARQL Update operations by removing the comment (##) characters at the start of the following lines in the joseki-config.ttl file.

## <#serviceUpdate>
##     rdf:type            joseki:Service ;
##     rdfs:label          "SPARQL/Update" ;
##     joseki:serviceRef   "update/service" ;
##     # dataset part
##     joseki:dataset      <#oracle>;
##     # Service part.    
##     # This processor will not allow either the protocol,
##     # nor the query, to specify the dataset.
##     joseki:processor    joseki:ProcessorSPARQLUpdate
##     .
## 
## <#serviceRead>
##     rdf:type            joseki:Service ;
##     rdfs:label          "SPARQL" ;
##     joseki:serviceRef   "sparql/read" ;
##     # dataset part
##     joseki:dataset      <#oracle> ;     ## Same dataset
##     # Service part. 
##     # This processor will not allow either the protocol,
##     # nor the query, to specify the dataset.
##     joseki:processor    joseki:ProcessorSPARQL_FixedDS ;
##     .

After you edit the joseki-config.ttl file, you must restart the Joseki Web application. You can then try a simple update operation, as follows:

  1. In your browser, go to: http://<hostname>:7001/joseki/update.html

  2. Type or paste the following into the text box:

    PREFIX : <http://example/>
    INSERT DATA {
      GRAPH <http://example/g1> { :r :p 455 }
    }
    
  3. Click Perform SPARQL Update.

To verify that the update operation was successful, go to http://<hostname>:7001/joseki and enter the following query:

SELECT *
WHERE
  {  GRAPH <http://example/g1> {?s ?p ?o}};

The response should contain the following triple:

<http://example/r>     <http://example/p>    "455"^^<http://www.w3.org/2001/XMLSchema#decimal>

A SPARQL Update can also be sent using an HTTP POST operation to the http://<hostname>:7001/joseki/update/service. For example, you can use curl (http://en.wikipedia.org/wiki/CURL) to send an HTTP POST request to perform the update operation:

curl --data "request=PREFIX%20%3A%20%3Chttp%3A%2F%2Fexample%2F%3E%20%0AINSERT%20DATA%20%7B%0A%20%20GRAPH%20%3Chttp%3A%2F%2Fexample%2Fg1%3E%20%7B%20%3Ar%20%3Ap%20888%20%7D%0A%7D%0A"  http://hostname:7001/joseki/update/service

In the preceding example, the URL encoded string is a simple INSERT operation into a named graph. After decoding, it reads as follows:

PREFIX : <http://example/>
INSERT DATA {
  GRAPH <http://example/g1> { :r :p 888 }

7.9 Analytical Functions for RDF Data

You can perform analytical functions on RDF data by using the SemNetworkAnalyst class in the oracle.spatial.rdf.client.jena package. This support integrates the Oracle Spatial network data model (NDM) logic with the underlying RDF data structures. Therefore, to use analytical functions on RDF data, you must be familiar with the Oracle Spatial NDM, which is documented in Oracle Spatial Topology and Network Data Models Developer's Guide.

The required NDM Java libraries, including sdonm.jar and sdoutl.jar, are under the directory $ORACLE_HOME/md/jlib. Note that xmlparserv2.jar (under $ORACLE_HOME/xdk/lib) must be included in the classpath definition.

Example 7-3 uses the SemNetworkAnalyst class, which internally uses the NDM NetworkAnalyst API

Example 7-3 Performing Analytical functions on RDF Data

Oracle oracle = new Oracle(jdbcUrl, user, password);
GraphOracleSem graph = new GraphOracleSem(oracle, modelName);
 
Node nodeA = Node.createURI("http://A");
Node nodeB = Node.createURI("http://B");
Node nodeC = Node.createURI("http://C");
Node nodeD = Node.createURI("http://D");
Node nodeE = Node.createURI("http://E");
Node nodeF = Node.createURI("http://F");
Node nodeG = Node.createURI("http://G");
Node nodeX = Node.createURI("http://X");
 
// An anonymous node
Node ano = Node.createAnon(new AnonId("m1"));
 
Node relL = Node.createURI("http://likes");
Node relD = Node.createURI("http://dislikes");
Node relK = Node.createURI("http://knows");
Node relC = Node.createURI("http://differs");
 
graph.add(new Triple(nodeA, relL, nodeB));
graph.add(new Triple(nodeA, relC, nodeD));
graph.add(new Triple(nodeB, relL, nodeC));
graph.add(new Triple(nodeA, relD, nodeC));
 
graph.add(new Triple(nodeB, relD, ano));
graph.add(new Triple(nodeC, relL, nodeD));
graph.add(new Triple(nodeC, relK, nodeE));
graph.add(new Triple(ano,   relL, nodeD));
graph.add(new Triple(ano,   relL, nodeF));
graph.add(new Triple(ano,   relD, nodeB));
 
// X only likes itself
graph.add(new Triple(nodeX, relL, nodeX));
 
graph.commitTransaction();
HashMap<Node, Double> costMap = new HashMap<Node, Double>();
costMap.put(relL, Double.valueOf((double)0.5));
costMap.put(relD, Double.valueOf((double)1.5));
costMap.put(relC, Double.valueOf((double)5.5));
 
graph.setDOP(4); // this allows the underlying LINK/NODE tables
                 // and indexes to be created in parallel.
 
SemNetworkAnalyst sna = SemNetworkAnalyst.getInstance(
    graph,   // network data source
    true,    // directed graph
    true,    // cleanup existing NODE and LINK table
    costMap
    );
 
psOut.println("From nodeA to nodeC");
Node[] nodeArray = sna.shortestPathDijkstra(nodeA, nodeC);
printNodeArray(nodeArray, psOut);
 
psOut.println("From nodeA to nodeD"); 
nodeArray = sna.shortestPathDijkstra( nodeA, nodeD);
printNodeArray(nodeArray, psOut);
 
psOut.println("From nodeA to nodeF");
nodeArray = sna.shortestPathAStar(nodeA, nodeF);
printNodeArray(nodeArray, psOut);
 
psOut.println("From ano to nodeC");
nodeArray = sna.shortestPathAStar(ano, nodeC);
printNodeArray(nodeArray, psOut);
 
psOut.println("From ano to nodeX");
nodeArray = sna.shortestPathAStar(ano, nodeX);
printNodeArray(nodeArray, psOut);
 
graph.close();
oracle.dispose();
...
...
   
// A helper function to print out a path
public static void printNodeArray(Node[] nodeArray, PrintStream psOut)
{
  if (nodeArray == null) {
    psOut.println("Node Array is null");
    return;
  }
  if (nodeArray.length == 0) {psOut.println("Node Array is empty"); }
  int iFlag = 0;
  psOut.println("printNodeArray: full path starts");
  for (int iHops = 0; iHops < nodeArray.length; iHops++) {
    psOut.println("printNodeArray: full path item " + iHops + " = "
        + ((iFlag == 0) ? "[n] ":"[e] ") + nodeArray[iHops]);
    iFlag = 1 - iFlag;
  }
}

In Example 7-3:

The output of Example 7-3 is as follows. In this output, the shortest paths are listed for the given start and end nodes. Note that the return value of sna.shortestPathAStar(ano, nodeX) is null because there is no path between these two nodes.

From nodeA to nodeC
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://A           ## "n" denotes Node             
printNodeArray: full path item 1 = [e] http://likes       ## "e" denotes Edge (Link)
printNodeArray: full path item 2 = [n] http://B
printNodeArray: full path item 3 = [e] http://likes
printNodeArray: full path item 4 = [n] http://C
 
From nodeA to nodeD
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://A
printNodeArray: full path item 1 = [e] http://likes
printNodeArray: full path item 2 = [n] http://B
printNodeArray: full path item 3 = [e] http://likes
printNodeArray: full path item 4 = [n] http://C
printNodeArray: full path item 5 = [e] http://likes
printNodeArray: full path item 6 = [n] http://D
 
From nodeA to nodeF
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://A
printNodeArray: full path item 1 = [e] http://likes
printNodeArray: full path item 2 = [n] http://B
printNodeArray: full path item 3 = [e] http://dislikes
printNodeArray: full path item 4 = [n] m1
printNodeArray: full path item 5 = [e] http://likes
printNodeArray: full path item 6 = [n] http://F
 
From ano to nodeC
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] m1
printNodeArray: full path item 1 = [e] http://dislikes
printNodeArray: full path item 2 = [n] http://B
printNodeArray: full path item 3 = [e] http://likes
printNodeArray: full path item 4 = [n] http://C
 
From ano to nodeX
Node Array is null

The underlying RDF graph view (SEMM_<model_name> or RDFM_<model_name>) cannot be used directly by NDM functions, and so SemNetworkAnalyst creates necessary tables that contain the nodes and links that are derived from a given RDF graph. These tables are not updated automatically when the RDF graph changes; rather, you can set the cleanup parameter in SemNetworkAnalyst.getInstance to true, to remove old node and link tables and to rebuild updated tables.

Example 7-4 implements the NDM' nearestNeighbors function on top of semantic data. This gets a NetworkAnalyst object from the SemNetworkAnalyst instance, gets the node ID, creates PointOnNet objects, and processes LogicalSubPath objects.

Example 7-4 Implementing NDM nearestNeighbors Function on Top of Semantic Data

%cat TestNearestNeighbor.java 
 
import java.io.*;
import java.util.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import com.hp.hpl.jena.util.iterator.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.update.*;
import com.hp.hpl.jena.sparql.core.DataSourceImpl;
 
import oracle.spatial.rdf.client.jena.*;
 
import oracle.spatial.rdf.client.jena.SemNetworkAnalyst;
import oracle.spatial.network.lod.LODGoalNode;
import oracle.spatial.network.lod.LODNetworkConstraint;
import oracle.spatial.network.lod.NetworkAnalyst;
import oracle.spatial.network.lod.PointOnNet;
import oracle.spatial.network.lod.LogicalSubPath;
 
 
/**
 * This class implements a nearestNeighbors function on top of semantic data
 * using public APIs provided in SemNetworkAnalyst and Oracle Spatial NDM
 */
public class TestNearestNeighbor
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
 
    PrintStream psOut = System.out;
 
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    
    String szModelName = "test_nn";
    // First construct a TBox and load a few axioms
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, szModelName);
    String insertString =  
      " PREFIX my:  <http://my.com/> " +
      " INSERT DATA "                             +
      " { my:A   my:likes my:B .                " +
      "   my:A   my:likes my:C .                " +
      "   my:A   my:knows my:D .                " +
      "   my:A   my:dislikes my:X .             " +
      "   my:A   my:dislikes my:Y .             " +
      "   my:C   my:likes my:E .                " +
      "   my:C   my:likes my:F .                " +
      "   my:C   my:dislikes my:M .             " +
      "   my:D   my:likes my:G .                " +
      "   my:D   my:likes my:H .                " +
      "   my:F   my:likes my:M .                " +
      " }   ";
    UpdateAction.parseExecute(insertString,  model);
 
    GraphOracleSem g = model.getGraph();
    g.commitTransaction();
    g.setDOP(4);
 
    HashMap<Node, Double> costMap = new HashMap<Node, Double>();
    costMap.put(Node.createURI("http://my.com/likes"),    Double.valueOf(1.0));
    costMap.put(Node.createURI("http://my.com/dislikes"), Double.valueOf(4.0));
    costMap.put(Node.createURI("http://my.com/knows"),    Double.valueOf(2.0));
 
    SemNetworkAnalyst sna = SemNetworkAnalyst.getInstance(
        g,     // source RDF graph
        true,  // directed graph
        true,  // cleanup old Node/Link tables
        costMap
        );
 
    Node nodeStart = Node.createURI("http://my.com/A");
    long origNodeID = sna.getNodeID(nodeStart);
 
    long[] lIDs = {origNodeID};
 
    // translate from the original ID
    long nodeID = (sna.mapNodeIDs(lIDs))[0]; 
 
    NetworkAnalyst networkAnalyst = sna.getEmbeddedNetworkAnalyst();
 
    LogicalSubPath[] lsps = networkAnalyst.nearestNeighbors(
      new PointOnNet(nodeID),      // startPoint
      6,                           // numberOfNeighbors
      1,                           // searchLinkLevel
      1,                           // targetLinkLevel
      (LODNetworkConstraint) null, // constraint
      (LODGoalNode) null           // goalNodeFilter
      );
 
    if (lsps != null) {
      for (int idx = 0; idx < lsps.length; idx++) {
        LogicalSubPath lsp = lsps[idx];
        Node[] nodePath = sna.processLogicalSubPath(lsp, nodeStart);
        psOut.println("Path " + idx);
        printNodeArray(nodePath, psOut);
      }
    }
 
    g.close();
    sna.close();
    oracle.dispose();
  }
 
 
  public static void printNodeArray(Node[] nodeArray, PrintStream psOut)
  {
    if (nodeArray == null) {
      psOut.println("Node Array is null");
      return;
    }
    if (nodeArray.length == 0) {
      psOut.println("Node Array is empty");
    }
    int iFlag = 0;
    psOut.println("printNodeArray: full path starts");
    for (int iHops = 0; iHops < nodeArray.length; iHops++) {
      psOut.println("printNodeArray: full path item " + iHops + " = "
          + ((iFlag == 0) ? "[n] ":"[e] ") + nodeArray[iHops]);
      iFlag = 1 - iFlag;
    }
  }
}

The output of Example 7-4 is as follows.

Path 0
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://my.com/A
printNodeArray: full path item 1 = [e] http://my.com/likes
printNodeArray: full path item 2 = [n] http://my.com/C
 
Path 1
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://my.com/A
printNodeArray: full path item 1 = [e] http://my.com/likes
printNodeArray: full path item 2 = [n] http://my.com/B
 
Path 2
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://my.com/A
printNodeArray: full path item 1 = [e] http://my.com/knows
printNodeArray: full path item 2 = [n] http://my.com/D
 
Path 3
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://my.com/A
printNodeArray: full path item 1 = [e] http://my.com/likes
printNodeArray: full path item 2 = [n] http://my.com/C
printNodeArray: full path item 3 = [e] http://my.com/likes
printNodeArray: full path item 4 = [n] http://my.com/E
 
Path 4
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://my.com/A
printNodeArray: full path item 1 = [e] http://my.com/likes
printNodeArray: full path item 2 = [n] http://my.com/C
printNodeArray: full path item 3 = [e] http://my.com/likes
printNodeArray: full path item 4 = [n] http://my.com/F
 
Path 5
printNodeArray: full path starts
printNodeArray: full path item 0 = [n] http://my.com/A
printNodeArray: full path item 1 = [e] http://my.com/knows
printNodeArray: full path item 2 = [n] http://my.com/D
printNodeArray: full path item 3 = [e] http://my.com/likes
printNodeArray: full path item 4 = [n] http://my.com/H

7.9.1 Generating Contextual Information about a Path in a Graph

It is sometimes useful to see contextual information about a path in a graph, in addition to the path itself. The buildSurroundingSubGraph method in the SemNetworkAnalyst class can output a DOT file (graph description language file, extension .gv) into the specified Writer object. For each node in the path, up to ten direct neighbors are used to produce a surrounding subgraph for the path. Example 7-5 shows the usage, specifically the output from the analytical functions used in Example 7-3, "Performing Analytical functions on RDF Data".

Example 7-5 Generating a DOT File with Contextual Information

nodeArray = sna.shortestPathDijkstra(nodeA, nodeD);
printNodeArray(nodeArray, psOut);
 
FileWriter dotWriter = new FileWriter("Shortest_Path_A_to_D.gv");
sna.buildSurroundingSubGraph(nodeArray, dotWriter);

The generated output DOT file from Example 7-5 is straightforward, as shown in the following example:

% cat Shortest_Path_A_to_D.gv
digraph { rankdir = LR; charset="utf-8"; 
 
"Rhttp://A" [ label="http://A" shape=rectangle,color=red,style = filled, ];
"Rhttp://B" [ label="http://B" shape=rectangle,color=red,style = filled, ];
"Rhttp://A" -> "Rhttp://B" [ label="http://likes"  color=red, style=bold, ];
"Rhttp://C" [ label="http://C" shape=rectangle,color=red,style = filled, ];
"Rhttp://A" -> "Rhttp://C" [ label="http://dislikes" ];
"Rhttp://D" [ label="http://D" shape=rectangle,color=red,style = filled, ];
"Rhttp://A" -> "Rhttp://D" [ label="http://differs" ];
"Rhttp://B" -> "Rhttp://C" [ label="http://likes"  color=red, style=bold, ];
"Rm1" [ label="m1" shape=ellipse,color=blue, ];
"Rhttp://B" -> "Rm1" [ label="http://dislikes" ];
"Rm1" -> "Rhttp://B" [ label="http://dislikes" ];
"Rhttp://C" -> "Rhttp://D" [ label="http://likes"  color=red, style=bold, ];
"Rhttp://E" [ label="http://E" shape=ellipse,color=blue, ];
"Rhttp://C" -> "Rhttp://E" [ label="http://knows" ];
"Rm1" -> "Rhttp://D" [ label="http://likes" ];
}

You can also use methods in the SemNetworkAnalyst and GraphOracleSem classes to produce more sophisticated visualization of the analytical function output.

You can convert the preceding DOT file into a variety of image formats. Figure 7-1 is an image representing the information in the preceding DOT file.

Figure 7-1 Visual Representation of Analytical Function Output

Description of Figure 7-1 follows
Description of "Figure 7-1 Visual Representation of Analytical Function Output"

7.10 Support for Server-Side APIs

This section describes some of the Oracle Database Semantic Technologies features that are exposed by the Jena Adapter. For comprehensive documentation of the API calls that support the available features, see the Jena Adapter reference information (Javadoc). For additional information about the server-side features exposed by the Jena Adapter, see the relevant chapters in this manual.

7.10.1 Virtual Models Support

Virtual models (explained in Section 1.3.8) are specified in the GraphOracleSem constructor, and they are handled transparently. If a virtual model exists for the model-rulebase combination, it is used in query answering; if such a virtual model does not exist, it is created in the database.


Note:

Virtual model support through the Jena Adapter is available only with Oracle Database Release 11.2 or later.

The following example reuses an existing virtual model.

String modelName = "EX";
String m1 = "EX_1";
 
ModelOracleSem defaultModel = 
  ModelOracleSem.createOracleSemModel(oracle, modelName);
 
// create these models in case they don't exist
ModelOracleSem model1 = ModelOracleSem.createOracleSemModel(oracle, m1);
 
String vmName = "VM_" + modelName;
 
 
//create a virtual model containing EX and EX_1
oracle.executeCall(
"begin sem_apis.create_virtual_model(?,sem_models('"+ m1 + "','"+ modelName+"'),null); end;",vmName);
 
String[] modelNames = {m1};
String[] rulebaseNames = {};
 
Attachment attachment = Attachment.createInstance(modelNames, rulebaseNames, 
InferenceMaintenanceMode.NO_UPDATE, QueryOptions.ALLOW_QUERY_VALID_AND_DUP);
 
// vmName is passed to the constructor, so GraphOracleSem will use the virtual 
// model named vmname (if the current user has read privileges on it)
GraphOracleSem graph = new GraphOracleSem(oracle, modelName, attachment, vmName);
graph.add(Triple.create(Node.createURI("urn:alice"),
                        Node.createURI("http://xmlns.com/foaf/0.1/mbox"),
                        Node.createURI("mailto:alice@example")));
ModelOracleSem model = new ModelOracleSem(graph);          
 
String queryString =
 
   " SELECT ?subject ?object WHERE { ?subject ?p ?object } ";
 
Query query = QueryFactory.create(queryString) ;
QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
 
try {
   ResultSet results = qexec.execSelect() ;
   for ( ; results.hasNext() ; ) {
      QuerySolution soln = results.nextSolution() ;
      psOut.println("soln " + soln);
   }
} 
finally { 
   qexec.close() ; 
}
 
OracleUtils.dropSemanticModel(oracle, modelName);
OracleUtils.dropSemanticModel(oracle, m1);
 
oracle.dispose();

You can also use the GraphOracleSem constructor to create a virtual model, as in the following example:

GraphOracleSem graph = new GraphOracleSem(oracle, modelName, attachment, true);

In this example, the fourth parameter (true) specifies that a virtual model needs to be created for the specified modelName and attachment.

7.10.2 Connection Pooling Support

Oracle Database Connection Pooling is provided through the Jena Adapter OraclePool class. Once this class is initialized, it can return Oracle objects out of its pool of available connections. Oracle objects are essentially database connection wrappers. After dispose is called on the Oracle object, the connection is returned to the pool. More information about using OraclePool can be found in the API reference information (Javadoc).

The following example sets up an OraclePool object with five (5) initial connections.

public static void main(String[] args) throws Exception
  {      
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
 
    // test with connection properties 
    java.util.Properties prop = new java.util.Properties();
    prop.setProperty("MinLimit", "2");     // the cache size is 2 at least 
    prop.setProperty("MaxLimit", "10");
    prop.setProperty("InitialLimit", "2"); // create 2 connections at startup
    prop.setProperty("InactivityTimeout", "1800");    //  seconds
    prop.setProperty("AbandonedConnectionTimeout", "900");  //  seconds
    prop.setProperty("MaxStatementsLimit", "10");
    prop.setProperty("PropertyCheckInterval", "60"); // seconds
 
    System.out.println("Creating OraclePool");
    OraclePool op = new OraclePool(szJdbcURL, szUser, szPasswd, prop, 
               "OracleSemConnPool");
    System.out.println("Done creating OraclePool");
 
    // grab an Oracle and do something with it
    System.out.println("Getting an Oracle from OraclePool");
    Oracle oracle = op.getOracle();
    System.out.println("Done");
    System.out.println("Is logical connection:" +
        oracle.getConnection().isLogicalConnection());
    GraphOracleSem g = new GraphOracleSem(oracle, szModelName);
    g.add(Triple.create(Node.createURI("u:John"), 
                        Node.createURI("u:parentOf"), 
                        Node.createURI("u:Mary")));
    g.close();
    // return the Oracle back to the pool
    oracle.dispose();
    
    // grab another Oracle and do something else 
    System.out.println("Getting an Oracle from OraclePool");
    oracle = op.getOracle();
    System.out.println("Done");
    System.out.println("Is logical connection:" +
        oracle.getConnection().isLogicalConnection());
    g = new GraphOracleSem(oracle, szModelName);
    g.add(Triple.create(Node.createURI("u:John"), 
                        Node.createURI("u:parentOf"), 
                        Node.createURI("u:Jack")));
    g.close();
    
    OracleUtils.dropSemanticModel(oracle, szModelName); 
    
    // return the Oracle object back to the pool
    oracle.dispose();
}

7.10.3 Semantic Model PL/SQL Interfaces

Several semantic PL/SQL subprograms are available through the Jena Adapter. Table 7-2 lists the subprograms and their corresponding Java class and methods.

Table 7-2 PL/SQL Subprograms and Corresponding Jena Adapter Java Class and Methods

PL/SQL SubprogramCorresponding Java Class and Methods

SEM_APIS.DROP_SEM_MODEL


OracleUtils.dropSemanticModel

SEM_APIS.MERGE_MODELS


OracleUtils.mergeModels

SEM_APIS.SWAP_NAMES


OracleUtils.swapNames

SEM_APIS.REMOVE_DUPLICATES


OracleUtils.removeDuplicates

SEM_APIS.RENAME_MODEL


OracleUtils.renameModels


For information about these PL/SQL utility subprograms, see the reference information in Chapter 9. For information about the corresponding Java class and methods, see the Jena Adapter API Reference documentation (Javadoc).

7.10.4 Inference Options

You can add options to entailment calls by using the following methods in the Attachment class (in package oracle.spatial.rdf.client.jena):

public void setUseLocalInference(boolean useLocalInference)
public boolean getUseLocalInference()
 
public void setDefGraphForLocalInference(String defaultGraphName)
public String getDefGraphForLocalInference()
 
public String getInferenceOption()
public void setInferenceOption(String inferenceOption)

For more information about these methods, see the Javadoc.

Example 7-6 enables parallel inference (with a degree of 4) and RAW format when creating an entailment. The example also uses the performInference method to create the entailment (comparable to using the SEM_APIS.CREATE_ENTAILMENT PL/SQL procedure).

Example 7-6 Specifying Inference Options

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import com.hp.hpl.jena.util.iterator.*;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.update.*;
import com.hp.hpl.jena.sparql.core.DataSourceImpl;
 
public class TestNewInference
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
 
    PrintStream psOut = System.out;
 
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    
    String szTBoxName = "test_new_tbox";
    {
      // First construct a TBox and load a few axioms
      ModelOracleSem modelTBox = ModelOracleSem.createOracleSemModel(oracle, szTBoxName);
      String insertString =  
        " PREFIX my:  <http://my.com/> " +
        " PREFIX rdfs:  <http://www.w3.org/2000/01/rdf-schema#> " +
        " INSERT DATA "                                     +
        " { my:C1  rdfs:subClassOf my:C2 .                " +
        "   my:C2  rdfs:subClassOf my:C3 .                " +
        "   my:C3  rdfs:subClassOf my:C4 .                " +
        " }   ";
      UpdateAction.parseExecute(insertString,  modelTBox);
      modelTBox.close();
    }
 
    String szABoxName = "test_new_abox";
    {
      // Construct an ABox and load a few quads
      ModelOracleSem modelABox = ModelOracleSem.createOracleSemModel(oracle, szABoxName);
      DatasetGraphOracleSem dataset = DatasetGraphOracleSem.createFrom(modelABox.getGraph());
      modelABox.close();
 
      String insertString =  
        " PREFIX my:    <http://my.com/> " +
        " PREFIX rdf:   <http://www.w3.org/1999/02/22-rdf-syntax-ns#> " +
        " INSERT DATA                                                 " +
        " { GRAPH my:G1 { my:I1  rdf:type my:C1 .                     " +
        "                 my:I2  rdf:type my:C2 .                     " +
        "               }                                             " +
        " };                                                          " +
        " INSERT DATA                                                 " +
        " { GRAPH my:G2 { my:J1  rdf:type my:C3 .                     " +
        "               }                                             " +
        " }    ";
      UpdateAction.parseExecute(insertString,  dataset);
      dataset.close();
    }
 
 
    String[] attachedModels = new String[1];
    attachedModels[0] = szTBoxName;
 
    String[] attachedRBs = {"OWL2RL"};
 
    Attachment attachment = Attachment.createInstance(
        attachedModels, attachedRBs,
        InferenceMaintenanceMode.NO_UPDATE,
        QueryOptions.ALLOW_QUERY_INVALID);
 
    // We are going to run named graph based local inference
    attachment.setUseLocalInference(true);
 
    // Set the default graph (TBox)
    attachment.setDefGraphForLocalInference(szTBoxName);
 
    // Set the inference option to use parallel inference 
    // with a degree of 4, and RAW format.
    attachment.setInferenceOption("DOP=4,RAW8=T");
 
    GraphOracleSem graph = new GraphOracleSem(
        oracle, 
        szABoxName, 
        attachment
        );
    DatasetGraphOracleSem dsgos = DatasetGraphOracleSem.createFrom(graph);
    graph.close();
 
    // Invoke create_entailment PL/SQL API
    dsgos.performInference();
 
    psOut.println("TestNewInference: # of inferred graph " + 
        Long.toString(dsgos.getInferredGraphSize()));
 
    String queryString = 
      " SELECT ?g ?s ?p ?o WHERE {  GRAPH ?g {?s ?p ?o } } " ;
 
    Query query = QueryFactory.create(queryString, Syntax.syntaxARQ);
    QueryExecution qexec = QueryExecutionFactory.create(
        query, DataSourceImpl.wrap(dsgos));
    ResultSet results = qexec.execSelect();
 
    ResultSetFormatter.out(psOut, results);
 
    dsgos.close();
    oracle.dispose();
  }
}

The output of Example 7-6 is as follows.

TestNewInference: # of inferred graph 9
 
--------------------------------------------------------------------------------------------------------------------
| g                  | s                  | p                                                 | o                  |
====================================================================================================================
| <http://my.com/G1> | <http://my.com/I2> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C3> |
| <http://my.com/G1> | <http://my.com/I2> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C2> |
| <http://my.com/G1> | <http://my.com/I2> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C4> |
| <http://my.com/G1> | <http://my.com/I1> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C3> |
| <http://my.com/G1> | <http://my.com/I1> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C1> |
| <http://my.com/G1> | <http://my.com/I1> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C2> |
| <http://my.com/G1> | <http://my.com/I1> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C4> |
| <http://my.com/G2> | <http://my.com/J1> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C3> |
| <http://my.com/G2> | <http://my.com/J1> | <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> | <http://my.com/C4> |
--------------------------------------------------------------------------------------------------------------------

For information about using OWL inferencing, see Section 2.2.

7.10.5 PelletInfGraph Class Support Deprecated

The support for the PelletInfGraph class within the Jena Adapter is deprecated. You should instead use the more optimized Oracle/Pellet integration through the PelletDb OWL 2 reasoner for Oracle Database 11g. (For information about PelletDb, see http://clarkparsia.com/pelletdb/.)

7.11 Bulk Loading Using the Jena Adapter

To load thousands to hundreds of thousands of RDF/OWL data files into an Oracle database, you can use the prepareBulk and completeBulk methods in the OracleBulkUpdateHandler Java class to simplify the task.

The addInBulk method in the OracleBulkUpdateHandler class can load triples of a graph or model into an Oracle database in bulk loading style. If the graph or model is a Jena in-memory graph or model, the operation is limited by the size of the physical memory. The prepareBulk method bypasses the Jena in-memory graph or model and takes a direct input stream to an RDF data file, parses the data, and load the triples into an underlying staging table. If the staging table and an accompanying table for storing long literals do not already exist, they are created automatically.

The prepareBulk method can be invoked multiple times to load multiple data files into the same underlying staging table. It can also be invoked concurrently, assuming the hardware system is balanced and there are multiple CPU cores and sufficient I/O capacity.

Once all the data files are processed by the prepareBulk method, you can invoke completeBulk to load all the data into the semantic network.

Example 7-7 shows how to load all data files in directory dir_1 into the underlying staging table. Long literals are supported and will be stored in a separate table. The data files can be compressed using GZIP to save storage space, and the prepareBulk method can detect automatically if a data file is compressed using GZIP or not.

Example 7-7 Loading Data into the Staging Table (prepareBulk)

Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
GraphOracleSem graph = new GraphOracleSem(oracle, szModelName);
 
PrintStream psOut = System.out;

String dirname = "dir_1";
File fileDir = new File(dirname);
String[] szAllFiles = fileDir.list();

// loop through all the files in a directory
for (int idx = 0; idx < szAllFiles.length; idx++) {
  String szIndFileName = dirname + File.separator + szAllFiles[idx];
  psOut.println("process to [ID = " + idx + " ] file " + szIndFileName);
  psOut.flush();
 
  try {
    InputStream is = new FileInputStream(szIndFileName);
    graph.getBulkUpdateHandler().prepareBulk(
        is,                    // input stream
        "http://example.com",  // base URI
        "RDF/XML",             // data file type: can be RDF/XML, N-TRIPLE, etc.
        "SEMTS",               // tablespace
        null,                  // flags
        null,                  // listener
        null                   // staging table name.
        );
    is.close();
  }
  catch (Throwable t) {
    psOut.println("Hit exception " + t.getMessage());
  }
}
 
graph.close();
oracle.dispose();

The code in Example 7-7, starting from creating a new Oracle object and ending with disposing of the Oracle object, can be executed in parallel. Assume there is a quad-core CPU and enough I/O capacity on the database hardware system; you can divide up all the data files and save them into four separate directories: dir_1, dir_2, dir_3, and dir_4. Four Java threads of processes can be started to work on those directories separately and concurrently. (For more information, see Section 7.11.1, "Using prepareBulk in Parallel (Multithreaded) Mode".)

After all data files are processed, you can invoke, just once, the completeBulk method to load the data from staging table into the semantic network, as shown in Example 7-8. Triples with long literals will be loaded also.

Example 7-8 Loading Data from the Staging Table into the Semantic Network (completeBulk)

graph.getBulkUpdateHandler().completeBulk(
  null,  // flags for invoking SEM_APIS.bulk_load_from_staging_table
  null   // staging table name
);

The prepareBulk method can also take a Jena model as an input data source, in which case triples in that Jena model are loaded into the underlying staging table. For more information, see the Javadoc.

In addition to loading triples from Jena models and data files, the prepareBulk method supports RDFa, as shown in Example 7-9. (RDFa is explained in http://www.w3.org/TR/xhtml-rdfa-primer/.)

Example 7-9 Using prepareBulk with RDFa

graph.getBulkUpdateHandler().prepareBulk(
  rdfaUrl,   // url to a web page using RDFa
  "SEMTS",   // tablespace
  null,      // flags
  null,      // listener
  null       // staging table name
);

To parse RDFa, the relevant java-rdfa libraries must be included in the classpath. No additional setup or API calls are required. (For information about java-rdfa, see http://www.rootdev.net/maven/projects/java-rdfa/ and the other topics there under Project Information.)

Note that if the rdfaUrl is located outside a firewall, you may need to set the following HTTP Proxy-related Java VM properties:

-Dhttp.proxyPort=...
-Dhttp.proxyHost=...

The preceding examples in this section load triple data into a single graph. Loading quad data that may span across multiple named graphs (such as data in NQUADS format) requires the use of the DatasetGraphOracleSem class. The DatasetGraphOracleSem class does not use the BulkUpdateHandler API, but does provide a similar prepareBulk and completeBulk interface, as shown in Example 7-10.

Example 7-10 Loading Quads into a DatasetGraph

Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
 
// Can only create DatasetGraphOracleSem from an existing GraphOracleSem
GraphOracleSem graph = new GraphOracleSem(oracle, szModelName);
DatasetGraphOracleSem dataset = DatasetGraphOracleSem.createFrom(graph);
 
// Don't need graph anymore, close it to free resources
graph.close();
 
try {
    InputStream is = new FileInputStream(szFileName);
    // load NQUADS file into a staging table. This file can be gzipp'ed.
    dataset.prepareBulk(
        is,                // input stream
        "http://my.base/", // base URI
        "N-QUADS",         // data file type; can be "TRIG"
        "SEMTS",           // tablespace
        null,              // flags
        null,              // listener
        null,              // staging table name
        false              // truncate staging table before load
    );
    // Load quads from staging table into the dataset
    dataset.completeBulk(
        null, // flags; can be "PARSE PARALLEL_CREATE_INDEX PARALLEL=4 
              //                mbv_method=shadow" on a quad core machine
        null  // staging table name
    );
} 
catch (Throwable t) {
    System.out.println("Hit exception " + t.getMessage());
}
finally {
    dataset.close();
    oracle.dispose();
}

7.11.1 Using prepareBulk in Parallel (Multithreaded) Mode

Example 7-7, "Loading Data into the Staging Table (prepareBulk)" provided a way to load, sequentially, a set of files under a file system directory to an Oracle Database table (staging table). Example 7-11 loads, concurrently, a set of files to an Oracle table (staging table). The degree of parallelism is controlled by the input parameter iMaxThreads.

On a balanced hardware setup with 4 or more CPU cores, setting iMaxThreads to 8 (or 16) can improve significantly the speed of prepareBulk operation when there are many data files to be processed.

Example 7-11 Using prepareBulk with iMaxThreads

public void testPrepareInParallel(String jdbcUrl, String user,
                       String password, String modelName,
                       String lang,
                       String tbs,
                       String dirname,
                       int iMaxThreads,
                       PrintStream psOut)
   throws SQLException, IOException,  InterruptedException
 {
   File dir = new File(dirname);
   File[] files = dir.listFiles();

   // create a set of physical Oracle connections and graph objects
   Oracle[] oracles = new Oracle[iMaxThreads];
   GraphOracleSem[] graphs = new GraphOracleSem[iMaxThreads];
   for (int idx = 0; idx < iMaxThreads; idx++) {
     oracles[idx] = new Oracle(jdbcUrl, user, password);
     graphs[idx] = new GraphOracleSem(oracles[idx], modelName);
   }

   PrepareWorker[] workers = new PrepareWorker[iMaxThreads];
   Thread[] threads = new Thread[iMaxThreads];
   for (int idx = 0; idx < iMaxThreads; idx++) {
     workers[idx] = new PrepareWorker(
         graphs[idx],
         files,
         idx,
         iMaxThreads,
         lang,
         tbs,
         psOut
         );
     threads[idx] = new Thread(workers[idx], workers[idx].getName());
     psOut.println("testPrepareInParallel: PrepareWorker " + idx + " running");
     threads[idx].start();
   }

   psOut.println("testPrepareInParallel: all threads started");

   for (int idx = 0; idx < iMaxThreads; idx++) {
     threads[idx].join();
   }
   for (int idx = 0; idx < iMaxThreads; idx++) {
     graphs[idx].close();
     oracles[idx].dispose();
   }
 }

 static class PrepareWorker implements Runnable
 {
   GraphOracleSem graph = null;
   int idx;
   int threads;
   File[] files = null;
   String lang = null;
   String tbs  = null;
   PrintStream psOut;

   public void run()
   {
     long lStartTime = System.currentTimeMillis();
     for (int idxFile = idx; idxFile < files.length; idxFile += threads) {
       File file = files[idxFile];
       try {
         FileInputStream fis = new FileInputStream(file);
         graph.getBulkUpdateHandler().prepareBulk(
             fis,
             "http://base.com/",
             lang,
             tbs,
             null,                   // flags
             new MyListener(psOut),  // listener
             null                    // table name
             );
         fis.close();
       }
       catch (Exception e) {
         psOut.println("PrepareWorker: thread ["+getName()+"] error "+ e.getMessage());
       }
       psOut.println("PrepareWorker: thread ["+getName()+"] done to "
           + idxFile + ", file = " + file.toString()
           + " in (ms) " + (System.currentTimeMillis() - lStartTime));
     }
   }

   public PrepareWorker(GraphOracleSem graph,
                        File[] files,
                        int idx,
                        int threads,
                        String lang,
                        String tbs,
                        PrintStream psOut)
   {
     this.graph   = graph;
     this.files   = files;
     this.psOut   = psOut;
     this.idx     = idx;
     this.threads = threads;
     this.files   = files;
     this.lang    = lang;
     this.tbs     = tbs ;
   }

   public String getName()
   {
     return "PrepareWorker" + idx;
   }
 } 
 
 static class MyListener implements StatusListener 
 {
   PrintStream m_ps = null;
   public MyListener(PrintStream ps) { m_ps = ps; }
   long lLastBatch = 0;

   public void statusChanged(long count)
   {
     if (count - lLastBatch >= 10000) {
       m_ps.println("process to " + Long.toString(count));
       lLastBatch = count;
     }
   }

   public int illegalStmtEncountered(Node graphNode, Triple triple, long count)
   {
     m_ps.println("hit illegal statement with object " + triple.getObject().toString());
     return 0; // skip it
   }
 }

7.11.2 Handling Illegal Syntax During Data Loading

You can skip illegal triples and quads when using prepareBulk. This feature is useful if the source RDF data may contain syntax errors. In Example 7-12, a customized implementation of the StatusListener interface (defined in package oracle.spatial.rdf.client.jena) is passed as a parameter to prepareBulk. In this example, the illegalStmtEncountered method prints the object field of the illegal triple, and returns 0 so that prepareBulk can skip that illegal triple and move on.

Example 7-12 Skipping Triples with Illegal Syntax

....
 
Oracle oracle = new Oracle(jdbcUrl, user, password);
GraphOracleSem graph = new GraphOracleSem(oracle, modelName);
PrintStream psOut = System.err;
 
graph.getBulkUpdateHandler().prepareBulk(
  new FileInputStream(rdfDataFilename),
  "http://base.com/",    // base 
  lang,                  // data format, can be "N-TRIPLES" "RDF/XML" ...
  tbs,                   // tablespace name
  null,                  // flags
  new MyListener(psOut), // call back to show progress and also process illegal triples/quads
  null,                  // tableName, if null use default names
  false                  // truncate existing staging tables
  );
 
 graph.close();
 oracle.dispose();
 .... 
 
 // A customized StatusListener interface implementation
  public class MyListener implements StatusListener
 {
   PrintStream m_ps = null;
   public MyListener(PrintStream ps) { m_ps = ps; }
 
   public void statusChanged(long count)
   {
     // m_ps.println("process to " + Long.toString(count));
   }
 
  public int illegalStmtEncountered(Node graphNode, Triple triple, long count)
  {
    m_ps.println("hit illegal statement with object " + triple.getObject().toString());
    return 0; // skip it
  }
 }

7.12 Automatic Variable Renaming

Previously, variable names used in SPARQL queries were passed directly on to Oracle Database as a part of a SQL statement. If the variable names included a SQL or PL/SQL reserved keyword, the query failed to execute. For example, the following SPARQL query used to fail because the word date as a special meaning to the Oracle Database SQL processing engine.

select ?date { :event  :happenedOn ?date }

Currently, this query does not fail, because a "smart scan" is performed and automatic replacement is done on certain reserved variable names (or variable names that are very long) before the query is sent to Oracle database for execution. The replacement is based on a list of reserved keywords that are stored in the following file embedded in sdordfclient.jar:

oracle/spatial/rdf/client/jena/oracle_sem_reserved_keywords.lst

This file contains over 100 entries, and you can edit the file to add entries if necessary.

The following are examples of SPARQL queries that use SQL or PL/SQL reserved keywords as variables, and that will succeed because of automatic variable renaming:

7.13 JavaScript Object Notation (JSON) Format Support

JavaScript Object Notation (JSON) format is supported for SPARQL query responses. JSON data format is simple, compact, and well suited for JavaScript programs.

For example, assume the following Java code snippet, which calls the ResultSetFormatter.outputAsJSON method:

Oracle oracle = new Oracle(jdbcUrl, user, password);
 
GraphOracleSem graph = new GraphOracleSem(oracle, modelName);
ModelOracleSem model = new ModelOracleSem(graph);
 
graph.add(new Triple(
                   Node.createURI("http://ds1"),
                   Node.createURI("http://dp1"),
                   Node.createURI("http://do1")
                   )
         );
 
graph.add(new Triple(
                   Node.createURI("http://ds2"),
                   Node.createURI("http://dp2"),
                   Node.createURI("http://do2")
                   )
         );
graph.commitTransaction();
 
Query q = QueryFactory.create("select ?s ?p ?o where {?s ?p ?o}",
                              Syntax.syntaxARQ);
QueryExecution qexec = QueryExecutionFactory.create(q, model);
 
ResultSet results = qexec.execSelect();
ResultSetFormatter.outputAsJSON(System.out, results);

The JSON output is as follows:

{
  "head": {
    "vars": [ "s" , "p" , "o" ]
  } ,
  "results": {
    "bindings": [
      {
        "s": { "type": "uri" , "value": "http://ds1" } ,
        "p": { "type": "uri" , "value": "http://dp1" } ,
        "o": { "type": "uri" , "value": "http://do1" }
      } ,
      {
        "s": { "type": "uri" , "value": "http://ds2" } ,
        "p": { "type": "uri" , "value": "http://dp2" } ,
        "o": { "type": "uri" , "value": "http://do2" }
      }
    ]
  }
}

The preceding example can be changed as follows to query a remote SPARQL endpoint instead of directly against an Oracle database. (If the remote SPARQL endpoint is outside a firewall, then the HTTP Proxy probably needs to be set.)

Query q = QueryFactory.create("select ?s ?p ?o where {?s ?p ?o}",
                              Syntax.syntaxARQ);
QueryExecution qe = QueryExecutionFactory.sparqlService(sparqlURL, q);
 
ResultSet results = qexec.execSelect();
ResultSetFormatter.outputAsJSON(System.out, results);

To extend the first example in this section to named graphs, the following code snippet adds two quads to the same Oracle model, executes a named graph-based SPARQL query, and serializes the query output into JSON format:

DatasetGraphOracleSem dsgos = DatasetGraphOracleSem.createFrom(graph);
graph.close();
 
dsgos.add(new Quad(Node.createURI("http://g1"),
                   Node.createURI("http://s1"),
                   Node.createURI("http://p1"),
                   Node.createURI("http://o1")
                   )
         );
dsgos.add(new Quad(Node.createURI("http://g2"),
                   Node.createURI("http://s2"),
                   Node.createURI("http://p2"),
                   Node.createURI("http://o2")
                   )
         );
 
Query q1 = QueryFactory.create(
  "select ?g ?s ?p ?o where { GRAPH ?g {?s ?p ?o} }");
 
QueryExecution qexec1 = QueryExecutionFactory.create(q1,
    DataSourceImpl.wrap(dsgos));
 
ResultSet results1 = qexec1.execSelect();
ResultSetFormatter.outputAsJSON(System.out, results1);
 
dsgos.close();
oracle.dispose();

The JSON output is as follows:

{
  "head": {
    "vars": [ "g" , "s" , "p" , "o" ]
  } ,
  "results": {
    "bindings": [
      {
        "g": { "type": "uri" , "value": "http://g1" } ,
        "s": { "type": "uri" , "value": "http://s1" } ,
        "p": { "type": "uri" , "value": "http://p1" } ,
        "o": { "type": "uri" , "value": "http://o1" }
      } ,
      {
        "g": { "type": "uri" , "value": "http://g2" } ,
        "s": { "type": "uri" , "value": "http://s2" } ,
        "p": { "type": "uri" , "value": "http://p2" } ,
        "o": { "type": "uri" , "value": "http://o2" }
      }
    ]
  }
}

You can also get a JSON response through HTTP against a Joseki-based SPARQL endpoint, as in the following example. Normally, when executing a SPARQL query against a SPARQL Web service endpoint, the Accept request-head field is set to be application/sparql-results+xml. For JSON output format, replace the Accept request-head field with application/sparql-results+json.

http://hostname:7001/joseki/oracle?query=<URL_ENCODED_SPARQL_QUERY>&output=json

7.14 Other Recommendations and Guidelines

This section contains various recommendations and other information related to SPARQL queries.

7.14.1 BOUND or !BOUND Instead of EXISTS or NOT EXISTS

For better performance, use BOUND or !BOUND instead of EXISTS or NOT EXISTS.

7.14.2 SPARQL 1.1 SELECT Expressions

You can use SPARQL 1.1 SELECT expressions without any significant performance overhead, even if the function is not currently supported within Oracle Database. Examples include the following:

-- Quyery using MD5 and SHA1 functions
PREFIX foaf: <http://xmlns.com/foaf/0.1/>
PREFIX xsd:  <http://www.w3.org/2001/XMLSchema#>
PREFIX eg:   <http://biometrics.example/ns#>
SELECT ?name (md5(?name) as ?name_in_md5) (sha1(?email) as ?sha1) 
WHERE 
{ 
  ?x foaf:name  ?name ; eg:email ?email .
}

-- Quyery using CONCAT function
PREFIX foaf:   <http://xmlns.com/foaf/0.1/>
SELECT ( CONCAT(?G, " ", ?S) AS ?name )
WHERE  
{ 
  ?P foaf:givenName ?G ; foaf:surname ?S 
}

7.14.3 Syntax Involving Bnodes (Blank Nodes)

Syntax involving bnodes can be used freely in query patterns. For example, the following bnode-related syntax is supported at the parser level, so each is equivalent to its full triple-query-pattern-based version.

:x :q [ :p "v" ] .
 
(1 ?x 3 4) :p "w" .
 
(1 [:p :q] ( 2 ) ) .

7.14.4 Limit in the SERVICE Clause

When writing a SPARQL 1.1 federated query, you can set a limit on returned rows in the subquery inside the SERVICE clause. This can effectively constrain the amount of data to be transported between the local repository and the remote SPARQL endpoint.

For example, the following query specifies limit 100 in the subquery in the SERVICE clause:

PREFIX : <http://example.com/>
SELECT ?s ?o 
 WHERE 
     { 
       ?s :name "CA"  
       SERVICE <http://REMOTE_SPARQL_ENDPOINT_HERE>
          {
             select ?s  ?o 
                 {?s :info ?o} 
              limit 100 
           } 
     }

7.14.5 OracleGraphWrapperForOntModel Class for Better Performance

The Jena OntModel class lets you create, modify, and analyze an ontology stored in a Jena model. However, the OntModel implementation is not optimized for semantic data stored in a database. This results in suboptimal performance when using OntModel with an Oracle model. Therefore, the class OracleGraphWrapperForOntModel has been created to alleviate this performance issue.

The OracleGraphWrapperForOntModel class implements the Jena Graph interface and represents a graph backed by an Oracle RDF/OWL model that is meant for use with the Jena OntModel API. The OracleGraphWrapperForOntModel class uses two semantic stores in a hybrid approach for persisting changes and responding to queries. Both semantic stores contain the same data, but one resides in memory while the other resides in the Oracle database.

When queried through OntModel, the OracleGraphWrapperForOntModel graph runs the queries against the in-memory store to improve performance. However, the OracleGraphWrapperForOntModel class persists changes made through OntModel, such as adding or removing classes, by applying changes to both stores.

Due to its hybrid approach, an OracleGraphWrapperForOntModel graph requires that sufficient memory be allocated to the JVM to store a copy of the ontology in memory. In internal experiments, it was found that an ontology with approximately 3 million triples requires 6 or more GB of physical memory.

Example 7-13 shows how to use the OntModel APIs with an existing ontology stored in an Oracle model.

Example 7-13 Using OntModel with Ontology Stored in Oracle Database

// Set up connection to Oracle semantic store and the Oracle model
// containing the ontology
Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
GraphOracleSem oracleGraph = new GraphOracleSem(oracle, szModelName);
 
// Create a new hybrid graph using the oracle graph to persist
// changes.  This method will copy all the data from the oracle graph
// into an in-memory graph, which may significantly increase JVM memory
// usage.
Graph hybridGraph = OracleGraphWrapperForOntModel.getInstance(oracleGraph);
 
// Build a model around the hybrid graph and wrap the model with Jena's
// OntModel
Model model = ModelFactory.createModelForGraph(hybridGraph);
OntModel ontModel = ModelFactory.createOntologyModel(ontModelSpec, model);
 
// Perform operations on the ontology
OntClass personClass = ontModel.createClass("<http://someuri/person>");
ontModel.createIndividual(personClass);
 
// Close resources (will also close oracleGraph)!
hybridGraph.close();
ontModel.close();

Note that any OntModel object created using OracleGraphWrapperForOntModel will not reflect changes made to the underlying Oracle model by another process, through a separate OntModel, or through a separate Oracle graph referencing the same underlying model. All changes to an ontology should go through a single OntModel object and its underlying OracleGraphWrapperForOntModel graph until the model or graph have been closed.

If the default in-memory semantic store used by OracleGraphWrapperForOntModel is not sufficient for an ontology and system, the class provides an interface for specifying a custom graph to use as the in-memory store. Example 7-14 shows how to create an OracleGraphWrapperForOntModel that uses a custom in-memory graph to answer queries from OntModel.

Example 7-14 Using a Custom In-Memory Graph

// Set up connection to Oracle semantic store and the Oracle model
// containing the ontology
Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
GraphOracleSem oracleGraph = new GraphOracleSem(oracle, szModelName);
 
// Create a custom in-memory graph to use instead of the default
// Jena in-memory graph for quickly answering OntModel queries.
// Note that this graph does not *need* to be in-memory, but in-memory
// is preferred.
GraphBase queryGraph = new CustomInMemoryGraphImpl();
 
// Create a new hybrid graph using the oracle graph to persist
// changes and the custom in-memory graph to answer queries. 
// Also set the degree of parallelism to use when copying data from
// the oracle graph to the querying graph.
int degreeOfParallelism = 4;
Graph hybridGraph = OracleGraphWrapperForOntModel.getInstance(oracleGraph, queryGraph, degreeOfParallelism);
 
// Build a model and wrap the model with Jena's OntModel
Model model = ModelFactory.createModelForGraph(hybridGraph);
OntModel ontModel = ModelFactory.createOntologyModel(ontModelSpec, model);
 
// Perform operations on the ontology
// ...
 
// Close resources (will close oracleGraph and queryGraph)!
hybridGraph.close();
ontModel.close();

7.15 Example Queries Using the Jena Adapter

This section includes example queries using the Jena Adapter. Each example is self-contained: it typically creates a model, creates triples, performs a query that may involve inference, displays the result, and drops the model.

This section includes queries that do the following:

To run a query, you must do the following:

  1. Include the code in a Java source file. The examples used in this section are supplied in files in the examples directory of the Jena Adapter download.

  2. Compile the Java source file. For example:

    > javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test.java
    

    Note:

    The javac and java commands must each be on a single command line.

  3. Run the compiled file. For example:

    > java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
    

7.15.1 Test.java: Query Family Relationships

Example 7-15 specifies that John is the father of Mary, and it selects and displays the subject and object in each fatherOf relationship

Example 7-15 Query Family Relationships

import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.query.*;
public class Test {
 
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
          
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    Model model = ModelOracleSem.createOracleSemModel(
      oracle, szModelName);
 
    model.getGraph().add(Triple.create(
          Node.createURI("http://example.com/John"),
          Node.createURI("http://example.com/fatherOf"),
          Node.createURI("http://example.com/Mary")));
    Query query = QueryFactory.create(
        "select ?f ?k WHERE {?f <http://example.com/fatherOf> ?k .}");
    QueryExecution qexec = QueryExecutionFactory.create(query, model);
    ResultSet results = qexec.execSelect();
    ResultSetFormatter.out(System.out, results, query);
    model.close();
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-15, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
 
---------------------------------------------------------
| f                         | k                         |
=========================================================
| <http://example.com/John> | <http://example.com/Mary> |
---------------------------------------------------------

7.15.2 Test6.java: Load OWL Ontology and Perform OWLPrime inference

Example 7-16 loads an OWL ontology and performs OWLPrime inference. Note that the OWL ontology is in RDF/XML format, and after it is loaded into Oracle it will be serialized out in N-TRIPLE form. The example also queries for the number of asserted and inferred triples.

The ontology in this example can be retrieved from several locations, including http://owl.cs.manchester.ac.uk/2008/iswc-tones/ontologies/univ-bench.owl, and it describes roles, resources, and relationships in a university environment.

Example 7-16 Load OWL Ontology and Perform OWLPrime inference

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import oracle.spatial.rdf.client.jena.*;
public class Test6 {
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    Model model = ModelOracleSem.createOracleSemModel(oracle, szModelName);
        
    // load UNIV ontology
    InputStream in = FileManager.get().open("./univ-bench.owl" );
    model.read(in, null);
    OutputStream os = new FileOutputStream("./univ-bench.nt");
    model.write(os, "N-TRIPLE");
    os.close();
 
    String queryString =
      " SELECT ?subject ?prop ?object WHERE { ?subject ?prop ?object } ";
 
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
 
    try {
      int iTriplesCount = 0;
      ResultSet results = qexec.execSelect() ;
      for ( ; results.hasNext() ; ) {
        QuerySolution soln = results.nextSolution() ;
        iTriplesCount++;
      }
      System.out.println("Asserted  triples count: " + iTriplesCount);
    } 
    finally { 
      qexec.close() ; 
    }
    
    Attachment attachment = Attachment.createInstance(
        new String[] {}, "OWLPRIME",
        InferenceMaintenanceMode.NO_UPDATE, QueryOptions.DEFAULT);
 
    GraphOracleSem graph = new GraphOracleSem(oracle, szModelName, attachment);
    graph.analyze();
    graph.performInference();
 
    query = QueryFactory.create(queryString) ;
    qexec = QueryExecutionFactory.create(query,new ModelOracleSem(graph)) ;
 
    try {
      int iTriplesCount = 0;
      ResultSet results = qexec.execSelect() ;
      for ( ; results.hasNext() ; ) {
        QuerySolution soln = results.nextSolution() ;
        iTriplesCount++;
      }
      System.out.println("Asserted + Infered triples count: " + iTriplesCount);
    } 
    finally { 
      qexec.close() ; 
    }
    model.close();    
 
    OracleUtils.dropSemanticModel(oracle, szModelName);    
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-16, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test6.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test6 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
Asserted  triples count: 294
Asserted + Infered triples count: 341

Note that this output reflects an older version of the LUBM ontology. The latest version of the ontology has more triples.

7.15.3 Test7.java: Bulk Load OWL Ontology and Perform OWLPrime inference

Example 7-17 loads the same OWL ontology as in Section 7.15.2, but stored in a local file using Bulk Loader. Ontologies can also be loaded using an incremental and batch loader; these two methods are also listed in the example for completeness.

Example 7-17 Bulk Load OWL Ontology and Perform OWLPrime inference

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.rdf.model.*;
import com.hp.hpl.jena.util.*;
import oracle.spatial.rdf.client.jena.*;
 
public class Test7 
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    // in memory Jena Model
    Model model = ModelFactory.createDefaultModel();
    InputStream is = FileManager.get().open("./univ-bench.owl");
    model.read(is, "", "RDF/XML");
    is.close();
 
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem modelDest = ModelOracleSem.createOracleSemModel(oracle, 
szModelName);
 
    GraphOracleSem g = modelDest.getGraph();
    g.dropApplicationTableIndex();
 
    int method = 2; // try bulk loader
    String tbs = "SYSAUX"; // can be customized
    if (method == 0) {
      System.out.println("start incremental");
      modelDest.add(model);
      System.out.println("end size " + modelDest.size());
    }
    else if (method == 1) {
      System.out.println("start batch load");
      g.getBulkUpdateHandler().addInBatch(
          GraphUtil.findAll(model.getGraph()), tbs);
      System.out.println("end size " + modelDest.size());
    }
    else {
      System.out.println("start bulk load");
      g.getBulkUpdateHandler().addInBulk(
          GraphUtil.findAll(model.getGraph()), tbs);
      System.out.println("end size " + modelDest.size());
    }
    g.rebuildApplicationTableIndex();
 
    long lCount = g.getCount(Triple.ANY);
    System.out.println("Asserted  triples count: " + lCount);
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-17, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test7.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test7 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
start bulk load
end size 293
Asserted  triples count: 293

Note that this output reflects an older version of the LUBM ontology. The latest version of the ontology has more triples.

7.15.4 Test8.java: SPARQL OPTIONAL Query

Example 7-18 shows a SPARQL OPTIONAL query. It inserts triples that postulate the following:

  • John is a parent of Mary.

  • John is a parent of Jack.

  • Mary is a parent of Jill.

It then finds parent-child relationships, optionally including any grandchild (gkid) relationships.

Example 7-18 SPARQL OPTIONAL Query

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
 
public class Test8 
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, 
szModelName);
    GraphOracleSem g = model.getGraph();
 
    g.add(Triple.create(
          Node.createURI("u:John"), Node.createURI("u:parentOf"), 
Node.createURI("u:Mary")));
    g.add(Triple.create(
          Node.createURI("u:John"), Node.createURI("u:parentOf"), 
Node.createURI("u:Jack")));
    g.add(Triple.create(
          Node.createURI("u:Mary"), Node.createURI("u:parentOf"), 
Node.createURI("u:Jill")));
        
    String queryString =
  " SELECT ?s ?o ?gkid " +
  " WHERE { ?s <u:parentOf> ?o . OPTIONAL {?o <u:parentOf> ?gkid }} ";
 
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
 
    try {
      int iMatchCount = 0;
      ResultSet results = qexec.execSelect() ;
      ResultSetFormatter.out(System.out, results, query);
    } 
    finally { 
      qexec.close() ; 
    }
    model.close();    
 
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-18, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test8.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test8 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
----------------------------------
| s        | o        | gkid     |
==================================
| <u:John> | <u:Mary> | <u:Jill> |
| <u:Mary> | <u:Jill> |          |
| <u:John> | <u:Jack> |          |
---------------------------------- 

7.15.5 Test9.java: SPARQL Query with LIMIT and OFFSET

Example 7-19 shows a SPARQL query with LIMIT and OFFSET. It inserts triples that postulate the following:

  • John is a parent of Mary.

  • John is a parent of Jack.

  • Mary is a parent of Jill.

It then finds one parent-child relationship (LIMIT 1), skipping the first two parent-child relationships encountered (OFFSET 2), and optionally includes any grandchild (gkid) relationships for the one found.

Example 7-19 SPARQL Query with LIMIT and OFFSET

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
public class Test9 
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, 
szModelName);
    GraphOracleSem g = model.getGraph();
 
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"),
                    Node.createURI("u:Mary")));
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"),
                    Node.createURI("u:Jack")));
    g.add(Triple.create(Node.createURI("u:Mary"), Node.createURI("u:parentOf"),              
                    Node.createURI("u:Jill")));
        
    String queryString =
      " SELECT ?s ?o ?gkid " +
      " WHERE { ?s <u:parentOf> ?o . OPTIONAL {?o <u:parentOf> ?gkid }} " +
      " LIMIT 1 OFFSET 2";
 
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
 
    int iMatchCount = 0;
    ResultSet results = qexec.execSelect() ;
    ResultSetFormatter.out(System.out, results, query);
    qexec.close() ; 
    model.close();    
 
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-19, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test9.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test9 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
------------------------------
| s        | o        | gkid |
==============================
| <u:John> | <u:Jack> |      |
------------------------------ 

7.15.6 Test10.java: SPARQL Query with TIMEOUT and DOP

Example 7-20 shows the SPARQL query from Section 7.15.5 with additional features, including a timeout setting (TIMEOUT=1, in seconds) and parallel execution setting (DOP=4).

Example 7-20 SPARQL Query with TIMEOUT and DOP

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
 
public class Test10 {
  public static void main(String[] args) throws Exception  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, szModelName);
    GraphOracleSem g = model.getGraph();
 
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
                            Node.createURI("u:Mary")));
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
                        Node.createURI("u:Jack")));
    g.add(Triple.create(Node.createURI("u:Mary"), Node.createURI("u:parentOf"), 
                        Node.createURI("u:Jill")));
    String queryString =
        " PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#dop=4,timeout=1> " 
      + " SELECT ?s ?o ?gkid WHERE { ?s <u:parentOf> ?o . " 
      + " OPTIONAL {?o <u:parentOf> ?gkid }} "
      + " LIMIT 1 OFFSET 2";
 
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
 
    int iMatchCount = 0;
    ResultSet results = qexec.execSelect() ;
    ResultSetFormatter.out(System.out, results, query);
    qexec.close() ; 
    model.close();    
 
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-20, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test10.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test10 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
------------------------------
| s        | o        | gkid |
==============================
| <u:Mary> | <u:Jill> |      |
------------------------------

7.15.7 Test11.java: Query Involving Named Graphs

Example 7-21 shows a query involving named graphs. It involves a default graph that has information about named graph URIs and their publishers. The query finds graph names, their publishers, and within each named graph finds the mailbox value using the foaf:mbox predicate.

Example 7-21 Named Graph Based Query

import java.io.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.sparql.core.*;
import com.hp.hpl.jena.query.*;
import oracle.spatial.rdf.client.jena.*;
 
public class Test11
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    GraphOracleSem graph = new GraphOracleSem(oracle, szModelName);
    DatasetGraphOracleSem dataset = DatasetGraphOracleSem.createFrom(graph);
    
    // don't need the GraphOracleSem anymore, release resources
    graph.close();
    
    // add data to the default graph
    dataset.add(new Quad(
          Quad.defaultGraphIRI, // specifies default graph
          Node.createURI("http://example.org/bob"),
          Node.createURI("http://purl.org/dc/elements/1.1/publisher"),
          Node.createLiteral("Bob Hacker")));
    dataset.add(new Quad(
          Quad.defaultGraphIRI, // specifies default graph
          Node.createURI("http://example.org/alice"),
          Node.createURI("http://purl.org/dc/elements/1.1/publisher"),
          Node.createLiteral("alice Hacker")));
    
    // add data to the bob named graph
    dataset.add(new Quad(
          Node.createURI("http://example.org/bob"), // graph name
          Node.createURI("urn:bob"),
          Node.createURI("http://xmlns.com/foaf/0.1/name"),
          Node.createLiteral("Bob")));
    dataset.add(new Quad(
          Node.createURI("http://example.org/bob"), // graph name
          Node.createURI("urn:bob"),
          Node.createURI("http://xmlns.com/foaf/0.1/mbox"),
          Node.createURI("mailto:bob@example")));
    
    // add data to the alice named graph
    dataset.add(new Quad(
          Node.createURI("http://example.org/alice"), // graph name
          Node.createURI("urn:alice"),
          Node.createURI("http://xmlns.com/foaf/0.1/name"),
          Node.createLiteral("Alice")));
    dataset.add(new Quad(
          Node.createURI("http://example.org/alice"), // graph name
          Node.createURI("urn:alice"),
          Node.createURI("http://xmlns.com/foaf/0.1/mbox"),
          Node.createURI("mailto:alice@example")));
    
    DataSource ds = DatasetFactory.create(dataset);
    
    String queryString =  
          " PREFIX foaf: <http://xmlns.com/foaf/0.1/> "
        + " PREFIX dc: <http://purl.org/dc/elements/1.1/> "
        + " SELECT ?who ?graph ?mbox "
        + " FROM NAMED <http://example.org/alice> "
        + " FROM NAMED <http://example.org/bob> "
        + " WHERE "
        + " { " 
        + "    ?graph dc:publisher ?who . "
        + "    GRAPH ?graph { ?x foaf:mbox ?mbox } "
        + " } ";
    
    Query query = QueryFactory.create(queryString);
    QueryExecution qexec = QueryExecutionFactory.create(query, ds);
    
    ResultSet results = qexec.execSelect();
    ResultSetFormatter.out(System.out, results, query);
    
    qexec.close();
    dataset.close();
    
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-21, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test11.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test11 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1

7.15.8 Test12.java: SPARQL ASK Query

Example 7-22 shows a SPARQL ASK query. It inserts a triple that postulates that John is a parent of Mary. It then finds whether John is a parent of Mary.

Example 7-22 SPARQL ASK Query

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
public class Test12
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, 
          szModelName);
    GraphOracleSem g = model.getGraph();
 
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
                        Node.createURI("u:Mary")));
    String queryString = " ASK { <u:John> <u:parentOf> <u:Mary> } ";
 
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
    boolean b = qexec.execAsk();
    System.out.println("ask result = " + ((b)?"TRUE":"FALSE"));
    qexec.close() ; 
    
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-22, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test12.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test12 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
ask result = TRUE

7.15.9 Test13.java: SPARQL DESCRIBE Query

Example 7-23 shows a SPARQL DESCRIBE query. It inserts triples that postulate the following:

  • John is a parent of Mary.

  • John is a parent of Jack.

  • Amy is a parent of Jack.

It then finds all relationships that involve any parents of Jack.

Example 7-23 SPARQL DESCRIBE Query

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
 
public class Test13
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, szModelName);
    GraphOracleSem g = model.getGraph();
 
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
                    Node.createURI("u:Mary")));
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
 Node.createURI("u:Jack")));
    g.add(Triple.create(Node.createURI("u:Amy"), Node.createURI("u:parentOf"), 
 Node.createURI("u:Jack")));
    String queryString = " DESCRIBE ?x WHERE {?x <u:parentOf> <u:Jack>}";
 
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
    Model m = qexec.execDescribe();
    System.out.println("describe result = " + m.toString());
 
    qexec.close() ; 
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-23, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test13.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test13 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
describe result = <ModelCom   {u:Amy @u:parentOf u:Jack; u:John @u:parentOf u:Jack; u:John @u:parentOf u:Mary} | >

7.15.10 Test14.java: SPARQL CONSTRUCT Query

Example 7-24 shows a SPARQL CONSTRUCT query. It inserts triples that postulate the following:

  • John is a parent of Mary.

  • John is a parent of Jack.

  • Amy is a parent of Jack.

  • Each parent loves all of his or her children.

It then constructs an RDF graph with information about who loves whom.

Example 7-24 SPARQL CONSTRUCT Query

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
 
public class Test14
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, szModelName);
    GraphOracleSem g = model.getGraph();
 
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
 Node.createURI("u:Mary")));
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
 Node.createURI("u:Jack")));
    g.add(Triple.create(Node.createURI("u:Amy"), Node.createURI("u:parentOf"), 
 Node.createURI("u:Jack")));
    String queryString = " CONSTRUCT { ?s <u:loves> ?o } WHERE {?s <u:parentOf> ?o}";
 
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
    Model m = qexec.execConstruct();
    System.out.println("Construct result = " + m.toString());
 
    qexec.close() ; 
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-24, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test14.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test14 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
Construct result = <ModelCom   {u:Amy @u:loves u:Jack; u:John @u:loves u:Jack; u:John @u:loves u:Mary} | >

7.15.11 Test15.java: Query Multiple Models and Specify "Allow Duplicates"

Example 7-25 queries multiple models and uses the "allow duplicates" option. It inserts triples that postulate the following:

  • John is a parent of Jack (in Model 1).

  • Mary is a parent of Jack (in Model 2).

  • Each parent loves all of his or her children.

It then finds out who loves whom. It searches both models and allows for the possibility of duplicate triples in the models (although there are no duplicates in this example).

Example 7-25 Query Multiple Models and Specify "Allow Duplicates"

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
 
public class Test15
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName1 = args[3];
    String szModelName2 = args[4];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model1 = ModelOracleSem.createOracleSemModel(oracle, szModelName1);
    model1.getGraph().add(Triple.create(Node.createURI("u:John"), 
                     Node.createURI("u:parentOf"), Node.createURI("u:Jack")));
    model1.close();
 
    ModelOracleSem model2 = ModelOracleSem.createOracleSemModel(oracle, szModelName2);
    model2.getGraph().add(Triple.create(Node.createURI("u:Mary"), 
                     Node.createURI("u:parentOf"), Node.createURI("u:Jack")));
    model2.close();
 
    String[] modelNamesList = {szModelName2};
    String[] rulebasesList  = {};
    Attachment attachment = Attachment.createInstance(modelNamesList, rulebasesList, 
              InferenceMaintenanceMode.NO_UPDATE,
              QueryOptions.ALLOW_QUERY_VALID_AND_DUP);
 
    GraphOracleSem graph = new GraphOracleSem(oracle, szModelName1, attachment);
    ModelOracleSem model = new ModelOracleSem(graph);
 
    String queryString = " CONSTRUCT { ?s <u:loves> ?o } WHERE {?s <u:parentOf> ?o}";
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
    Model m = qexec.execConstruct();
    System.out.println("Construct result = " + m.toString());
 
    qexec.close() ; 
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName1);
    OracleUtils.dropSemanticModel(oracle, szModelName2);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-25, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test15.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test15 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1 M2
Construct result = <ModelCom   {u:Mary @u:loves u:Jack; u:John @u:loves u:Jack} | >

7.15.12 Test16.java: SPARQL Update

Example 7-26 inserts two triples into a model.

Example 7-26 SPARQL Update

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import com.hp.hpl.jena.util.iterator.*;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.update.*;
 
public class Test16
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, szModelName);
    GraphOracleSem g = model.getGraph();
    String insertString =  
      " PREFIX dc: <http://purl.org/dc/elements/1.1/> "         + 
      " INSERT DATA "                                           +
      " { <http://example/book3> dc:title    \"A new book\" ; " +
      "                         dc:creator  \"A.N.Other\" . "   + 
      " }   ";
 
    UpdateAction.parseExecute(insertString,  model);
    ExtendedIterator ei = GraphUtil.findAll(g);
    while (ei.hasNext()) {
      System.out.println("Triple " + ei.next().toString());
    }
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-26, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test16.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test16 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
Triple http://example/book3 @dc:title "A new book"
Triple http://example/book3 @dc:creator "A.N.Other"

7.15.13 Test17.java: SPARQL Query with ARQ Built-In Functions

Example 7-27 inserts data about two books, and it displays the book titles in all uppercase characters and the length of each title string.

Example 7-27 SPARQL Query with ARQ Built-In Functions

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import com.hp.hpl.jena.util.iterator.*;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.update.*;
 
public class Test17 {
  public static void main(String[] args) throws Exception  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, szModelName);
    GraphOracleSem g = model.getGraph();
    String insertString =  
      " PREFIX dc: <http://purl.org/dc/elements/1.1/> "         + 
      " INSERT DATA "                                           +
      " { <http://example/book3> dc:title    \"A new book\" ; " +
      "                         dc:creator  \"A.N.Other\" . "   + 
      "   <http://example/book4> dc:title    \"Semantic Web Rocks\" ; " +
      "                         dc:creator  \"TB\" . "   + 
      " }   ";
 
    UpdateAction.parseExecute(insertString,  model);
    String queryString = "PREFIX  dc:   <http://purl.org/dc/elements/1.1/> " +
      " PREFIX  fn: <http://www.w3.org/2005/xpath-functions#> " + 
      " SELECT ?subject (fn:upper-case(?object) as ?object1)  " + 
      "                 (fn:string-length(?object) as ?strlen) " + 
      " WHERE { ?subject dc:title ?object } " 
      ;
    Query query = QueryFactory.create(queryString, Syntax.syntaxARQ);
    QueryExecution qexec = QueryExecutionFactory.create(query, model);
    ResultSet results = qexec.execSelect();
    ResultSetFormatter.out(System.out, results, query);
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-27, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test17.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test17 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
----------------------------------------------------------
| subject                | object1              | strlen |
==========================================================
| <http://example/book3> | "A NEW BOOK"         | 10     |
| <http://example/book4> | "SEMANTIC WEB ROCKS" | 18     |
----------------------------------------------------------

7.15.14 Test18.java: SELECT Cast Query

Example 7-28 "converts" two Fahrenheit temperatures (18.1 and 32.0) to Celsius temperatures.

Example 7-28 SELECT Cast Query

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import com.hp.hpl.jena.util.iterator.*;
import oracle.spatial.rdf.client.jena.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.update.*;
 
public class Test18 {
  public static void main(String[] args) throws Exception  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
    
    Oracle oracle = new Oracle(szJdbcURL, szUser, szPasswd);
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, 
szModelName);
    GraphOracleSem g = model.getGraph();
    String insertString =  
      " PREFIX xsd: <http://www.w3.org/2001/XMLSchema#> " +
      " INSERT DATA "                                     +
      " { <u:Object1> <u:temp>    \"18.1\"^^xsd:float ; " +
      "               <u:name>    \"Foo... \" . "         + 
      "   <u:Object2> <u:temp>    \"32.0\"^^xsd:float ; " +
      "               <u:name>    \"Bar... \" . "         + 
      " }   ";
 
    UpdateAction.parseExecute(insertString,  model);
    String queryString = 
      " PREFIX  fn: <http://www.w3.org/2005/xpath-functions#> " + 
      " SELECT ?subject ((?temp - 32.0)*5/9 as ?celsius_temp) " +
      "WHERE { ?subject <u:temp> ?temp } " 
      ;
    Query query = QueryFactory.create(queryString, Syntax.syntaxARQ);
    QueryExecution qexec = QueryExecutionFactory.create(query, model);
    ResultSet results = qexec.execSelect();
    ResultSetFormatter.out(System.out, results, query);
 
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-28, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test18.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test18 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
------------------------------------------------------------------------
| subject     | celsius_temp                                           |
========================================================================
| <u:Object1> | "-7.7222223"^^<http://www.w3.org/2001/XMLSchema#float> |
| <u:Object2> | "0.0"^^<http://www.w3.org/2001/XMLSchema#float>        |
------------------------------------------------------------------------

7.15.15 Test19.java: Instantiate Oracle Database Using OracleConnection

Example 7-29 shows a different way to instantiate an Oracle object using a given OracleConnection object. (In a J2EE Web application, users can normally get an OracleConnection object from a J2EE data source.)

Example 7-29 Instantiate Oracle Database Using OracleConnection

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import com.hp.hpl.jena.util.iterator.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.update.*;
import oracle.spatial.rdf.client.jena.*;
import oracle.jdbc.pool.*;
import oracle.jdbc.*;
        
public class Test19 {
  public static void main(String[] args) throws Exception {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
 
    OracleDataSource ds = new OracleDataSource();
    ds.setURL(szJdbcURL);
    ds.setUser(szUser);
    ds.setPassword(szPasswd);
    OracleConnection conn = (OracleConnection) ds.getConnection();
    Oracle oracle = new Oracle(conn);
 
    ModelOracleSem model = ModelOracleSem.createOracleSemModel(oracle, 
szModelName);
    GraphOracleSem g = model.getGraph();
 
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
                        Node.createURI("u:Mary")));
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"),   
                        Node.createURI("u:Jack")));
    g.add(Triple.create(Node.createURI("u:Mary"), Node.createURI("u:parentOf"), 
         Node.createURI("u:Jill")));
    String queryString =
       " SELECT ?s ?o  WHERE { ?s <u:parentOf> ?o .} ";
    Query query = QueryFactory.create(queryString) ;
    QueryExecution qexec = QueryExecutionFactory.create(query, model) ;
 
    ResultSet results = qexec.execSelect() ;
    ResultSetFormatter.out(System.out, results, query);
    qexec.close() ; 
    model.close();    
    OracleUtils.dropSemanticModel(oracle, szModelName);
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-29, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test19.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test19 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
-----------------------
| s        | o        |
=======================
| <u:John> | <u:Mary> |
| <u:John> | <u:Jack> |
| <u:Mary> | <u:Jill> |
-----------------------

7.15.16 Test20.java: Oracle Database Connection Pooling

Example 7-30 uses Oracle Database connection pooling.

Example 7-30 Oracle Database Connection Pooling

import java.io.*;
import com.hp.hpl.jena.query.*;
import com.hp.hpl.jena.rdf.model.Model;
import com.hp.hpl.jena.util.FileManager;
import com.hp.hpl.jena.util.iterator.*;
import com.hp.hpl.jena.graph.*;
import com.hp.hpl.jena.update.*;
import oracle.spatial.rdf.client.jena.*;
import oracle.jdbc.pool.*;
import oracle.jdbc.*;
 
public class Test20
{
  public static void main(String[] args) throws Exception
  {
    String szJdbcURL = args[0];
    String szUser    = args[1];
    String szPasswd  = args[2];
    String szModelName = args[3];
 
    // test with connection properties (taken from some example)
    java.util.Properties prop = new java.util.Properties();
    prop.setProperty("MinLimit", "2");     // the cache size is 2 at least 
    prop.setProperty("MaxLimit", "10");
    prop.setProperty("InitialLimit", "2"); // create 2 connections at startup
    prop.setProperty("InactivityTimeout", "1800");    //  seconds
    prop.setProperty("AbandonedConnectionTimeout", "900");  //  seconds
    prop.setProperty("MaxStatementsLimit", "10");
    prop.setProperty("PropertyCheckInterval", "60"); // seconds
 
    System.out.println("Creating OraclePool");
    OraclePool op = new OraclePool(szJdbcURL, szUser, szPasswd, prop, 
               "OracleSemConnPool");
    System.out.println("Done creating OraclePool");
 
    // grab an Oracle and do something with it
    System.out.println("Getting an Oracle from OraclePool");
    Oracle oracle = op.getOracle();
    System.out.println("Done");
    System.out.println("Is logical connection:" +
        oracle.getConnection().isLogicalConnection());
    GraphOracleSem g = new GraphOracleSem(oracle, szModelName);
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
                        Node.createURI("u:Mary")));
    g.close();
    // return the Oracle back to the pool
    oracle.dispose();
    
    // grab another Oracle and do something else 
    System.out.println("Getting an Oracle from OraclePool");
    oracle = op.getOracle();
    System.out.println("Done");
    System.out.println("Is logical connection:" +
        oracle.getConnection().isLogicalConnection());
    g = new GraphOracleSem(oracle, szModelName);
    g.add(Triple.create(Node.createURI("u:John"), Node.createURI("u:parentOf"), 
                        Node.createURI("u:Jack")));
    g.close();
    
    OracleUtils.dropSemanticModel(oracle, szModelName); 
    
    // return the Oracle back to the pool
    oracle.dispose();
  }
}

The following are the commands to compile and run Example 7-30, as well as the expected output of the java command.

javac -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:/slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar Test20.java
java -classpath ./:./jena-2.6.4.jar:./sdordfclient.jar:./ojdbc6.jar:./slf4j-api-1.5.8.jar:./slf4j-log4j12-1.5.8.jar:./arq-2.8.8.jar:./xercesImpl-2.7.1.jar:./iri-0.8.jar:./icu4j-3.4.4.jar:./log4j-1.2.14.jar  Test20 jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
Creating OraclePool
Done creating OraclePool
Getting an Oracle from OraclePool
Done
Is logical connection:true
Getting an Oracle from OraclePool
Done
Is logical connection:true

7.16 SPARQL Gateway and Semantic Data

SPARQL Gateway is a J2EE web application that is included with the Jena Adapter for Oracle Database. It is designed to make semantic data (RDF/OWL/SKOS) easily available to applications that operate on relational and XML data, including Oracle Business Intelligence Enterprise Edition (OBIEE) 11g.

This section includes the following major topics:

7.16.1 SPARQL Gateway Features and Benefits Overview

SPQRQL Gateway handles several challenges in exposing semantic data to a non-semantic application:

  • RDF syntax, SPARQL query syntax and SPARQL protocol must be understood.

  • The SPARQL query response syntax must be understood.

  • A transformation must convert a SPARQL query response to something that the application can consume.

To address these challenges, SPARQL Gateway manages SPARQL queries and XSLT operations, executes SPARQL queries against any arbitrary standard-compliant SPARQL endpoints, and performs necessary XSL transformations before passing the response back to applications. Applications can then consume semantic data as if it is coming from an existing data source.

Different triple stores or quad stores often have different capabilities. For example, the SPARQL endpoint supported by Oracle Database, with the Jena Adapter and Joseki, allows parallel execution, query timeout, dynamic sampling, result cache, and other features, in addition to the core function of parsing and answering a given standard-compliant SPARQL query. However, these features may not be available from another given semantic data store.

With the Oracle Semantic Technologies SPARQL Gateway, you get certain highly desirable capabilities, such as the ability to set a timeout on a long running query and the ability to get partial results from a complex query in a given amount of time. Waiting indefinitely for a query to finish is a challenge for end users, as is an application with a response time constraint. SPARQL Gateway provides both timeout and best effort query functions on top of a SPARQL endpoint. This effectively removes some uncertainty from consuming semantic data through SPARQL query executions. (See Section 7.16.3.2, "Specifying a Timeout Value" and Section 7.16.3.3, "Specifying Best Effort Query Execution".)

7.16.2 Installing and Configuring SPARQL Gateway

To install and configure SPARQL Gateway, follow these major steps, which are explained in subsections that follow:

  1. Download the Jena Adapter .zip File (if Not Already Done)

  2. Deploy SPARQL Gateway in WebLogic Server

  3. Modify Proxy Settings, if Necessary

  4. Configure the OracleSGDS Data Source, if Necessary

  5. Add and Configure the SparqlGatewayAdminGroup Group, if Desired

7.16.2.1 Download the Jena Adapter .zip File (if Not Already Done)

If you have not already done so, download the Jena Adapter (jena_adaptor_for_release11.2.0.3.zip) from the Oracle Database Semantic Technologies page and unzip it into a temporary directory, as explained in Section 7.1.

Note that the SPARQL Gateway Java class implementations are embedded in sdordfclient.jar (see Section 7.16.5, "Using the SPARQL Gateway Java API").

7.16.2.2 Deploy SPARQL Gateway in WebLogic Server

Deploy SPARQL Gateway in Oracle WebLogic Server, as follows:

  1. Create a temporary directory, for example, /tmp/sg.

  2. From the directory into which you unzipped the SPARQL Gateway file that you downloaded, copy the sparqlgateway directory and everything under it (including subdirectories) into the temporary directory you created in step 1. For example:

    cd /tmp
    cp -rf  sparqlgateway/*  /tmp/sg
    

    (This sample directory structure is assumed in some later steps.)

  3. If you do not already have the ARQ 2.8.8 package, download it from http://sourceforge.net/projects/jena/files/ARQ/ARQ-2.8.8/arq-2.8.8.zip/download.

  4. Copy the following libraries from the ARQ 2.8.8 package to /tmp/sg/WEB-INF/lib:

    arq-2.8.8.jar
    icu4j-3.4.4.jar
    iri-0.8.jar
    jena-2.6.4.jar
    junit-4.5.jar
    log4j-1.2.14.jar
    lucene-core-2.3.1.jar
    slf4j-api-1.5.8.jar
    slf4j-log4j12-1.5.8.jar
    stax-api-1.0.1.jar
    wstx-asl-3.2.9.jar
    xercesImpl-2.7.1.jar
    
  5. Copy the following libraries to /tmp/sg/WEB-INF/lib:

    ojdbc6.jar  (from $ORACLE_HOME/jdbc/lib/ojdbc6.jar )
    sdordfclient.jar  (from jena_adaptor_for_release11.2.0.3.zip under the jar/ directory)
    sdordf.jar  from $ORACLE_HOME/md/jlib/sdordf.jar )
    
  6. Customize the /tmp/sg/WEB-INF/web.xml file as needed. Be sure to specify appropriate values for the sparql_gateway_repository_filedir and sparql_gateway_repository_url parameters.

  7. Add XSLT files or SPARQL query files to the /tmp/sg directory, if necessary.

    The following files are provided by Oracle in that directory: default.xslt, noop.xslt, and qb1.sparql. The default.xslt file is intended mainly for transforming SPARQL query responses (XML) to a format acceptable to Oracle Business Intelligence Enterprise Edition (OBIEE).

    (These files are described in Section 7.16.3.1, "Storing SPARQL Queries and XSL Transformations"; using SPARQL Gateway with OBIEE is explained in Section 7.16.7, "Using SPARQL Gateway as an XML Data Source to OBIEE".)

  8. Go to the autodeploy directory of WebLogic Server, create directory sparqlgateway.war, and copy files, as follows. (For information about auto-deploying applications in development domains, see: http://download.oracle.com/docs/cd/E11035_01/wls100/deployment/autodeploy.html)

    cd <domain_name>/autodeploy
    mkdir sparqlgateway.war  
    cp -rf  /tmp/sg/*  <domain_name>/autodeploy/sparqgateway.war
      
    

    In this example, <domain_name> is the name of a WebLogic Server domain.

  9. If you want to build a .war file from the /tmp/sg directory, enter the following commands:

    cd /tmp/sg
    jar cvf /tmp/sparqlgateway.war *
    
  10. Start or restart WebLogic Server.

  11. Verify your deployment by using your Web browser to connect to a URL in the following format (assume that the Web application is deployed at port 7001):

    http://<hostname>:7001/sparqlgateway
    

7.16.2.3 Modify Proxy Settings, if Necessary

If your SPARQL Gateway is behind a firewall and you want SPARQL Gateway to communicate with SPARQL endpoints on the Internet as well as those inside the firewall, you probably need to use the following JVM settings:

-Dhttp.proxyHost=<your_proxy_host>
-Dhttp.proxyPort=<your_proxy_port>
-Dhttp.nonProxyHosts=127.0.0.1|<hostname_1_for_sparql_endpoint_inside_firewall>|<hostname_2_for_sparql_endpoint_inside_firewall>|...|<hostname_n_for_sparql_endpoint_inside_firewall>

You can specify these settings in the startWebLogic.sh script.

7.16.2.4 Configure the OracleSGDS Data Source, if Necessary

If an Oracle database is used for storage of and access to SPARQL queries and XSL transformations for SPARQL Gateway, then you must configure a data source named OracleSGDS.

To create this data source, follow the instructions in Section 7.2.1, "Creating the Required Data Source Using WebLogic Server"; however, specify OracleSGDS as the data source name instead of OracleSemDS.

If the OracleSGDS data source is configured and available, SPARQL Gateway servlet will automatically create all the necessary tables and indexes upon initialization.

7.16.2.5 Add and Configure the SparqlGatewayAdminGroup Group, if Desired

The following JSP files in SPARQL Gateway can help you to view, edit, and update SPARQL queries and XSL transformations that are stored in an Oracle database:

http://<host>:7001/sparqlgateway/admin/sparql.jsp
http://<host>:7001/sparqlgateway/admin/xslt.jsp

These files are protected by HTTP Basic Authentication. In WEB-INF/weblogic.xml, a principal named SparqlGatewayAdminGroup is defined.

To be able to log in to either of these JSP pages, you must use the WebLogic Server to add a group named SparqlGatewayAdminGroup, and create a new user or assign an existing user to this group.

7.16.3 Using SPARQL Gateway with Semantic Data

The primary interface for an application to interact with SPARQL Gateway is through a URL with the following format:

http://host:port/sparqlgateway/sg?<SPARQL_ENDPOINT>&<SPARQL_QUERY>&<XSLT>

In the preceding format:

  • <SPARQL_ENDPOINT> specifies the ee parameter, which contains a URL encoded form of a SPARQL endpoint.

    For example, ee=http%3A%2F%2Fsparql.org%2Fbooks is the URL encoded string for SPARQL endpoint http://sparql.org/books. It means that SPARQL queries are to be executed against endpoint http://sparql.org/books.

  • <SPARQL_QUERY> specifies either the SPARQL query, or the location of the SPARQL query.

    If it is feasible for an application to accept a very long URL, you can encode the whole SPARQL query and set eq=<encoded_SPARQL_query> in the URL If it is not feasible for an application to accept a very long URL, you can store the SPARQL queries and make them available to SPARQL Gateway using one of the approaches described in Section 7.16.3.1.

  • <XSLT> specifies either the XSL transformation, or the location of the XSL transformation.

    If it is feasible for an application to accept a very long URL, you can encode the whole XSL transformation and set ex=<encoded_XSLT> in the URL If it is not feasible for an application to accept a very long URL, you can store the XSL transformations and make them available to SPARQL Gateway using one of the approaches described in Section 7.16.3.1.

Related topics:

7.16.3.1 Storing SPARQL Queries and XSL Transformations

If it is not feasible for an application to accept a very long URL, you can specify the location of the SPARQL query and the XSL transformation in the <SPARQL_QUERY> and <XSLT> portions of the URL format described in Section 7.16.3, using any of the following approaches:

  • Store the SPARQL queries and XSL transformations in the SPARQL Gateway Web application itself.

    To do this, unpack the sparqlgateway.war file, and store the SPARQL queries and XSL transformations in the top-level directory; then pack the sparqlgateway.war file and redeploy it.

    The sparqlgateway.war file includes the following example files: qb1.sparql (SPARQL query) and default.xslt (XSL transformation).


    Tip:

    Use the file extension .sparql for SPARQL query files, and the file extension .xslt for XSL transformation files.

    The syntax for specifying these files (using the provided example file names) is wq=qb1.sparql for a SPARQL query file and wx=default.xslt for an XSL transformation file.

    If you want to customize the default XSL transformations, see the examples in Section 7.16.4, "Customizing the Default XSLT File".

    If you specify wx=noop.xslt, XSL transformation is not performed and the SPARQL response is returned "as is" to the client.

  • Store the SPARQL queries and XSL transformations in a file system directory, and make sure that the directory is accessible for the deployed SPARQL Gateway Web application.

    By default, the directory is set to /tmp, as shown in the following <init-param> setting:

    <init-param>
       <param-name>sparql_gateway_repository_filedir</param-name>
       <param-value>/tmp/</param-value>
    </init-param>
    

    It is recommended that you customize this directory before deploying the SPARQL Gateway. To change the directory setting, edit the text in between the <param-value> and </param-value> tags.

    The follwoing examples specify a SPARQL query file and an XSL transflrmation file that are in the directory specified in the <init-param> element for sparql_gateway_repository_filedir:

    fq=qb1.sparql
    fx=myxslt1.xslt
    
  • Make the SPARQL queries and XSL transformations accessible from a website.

    By default, the website directory is set to http://127.0.0.1/queries/, as shown in the following <init-param> setting:

    <init-param>
       <param-name>sparql_gateway_repository_url</param-name>
       <param-value>http://127.0.0.1/queries/</param-value>
    </init-param>
    

    Customize this directory before deploying the SPARQL Gateway. To change the website setting, edit the text in between the <param-value> and </param-value> tags.

    The follwoing example specifies a SPARQL query file and an XSL transflrmation file that are in the URL specified in the <init-param> element for sparql_gateway_repository_url.

    uq=qb1.sparql
    ux=myxslt1.xslt
    

    Internally, SPARQL Gateway computes the appropriate complete URL, fetches the content, starts query execution, and applies the XSL transformation to the query response XML.

  • Store the SPARQL queries and XSL transformations in an Oracle database.

    This approach requires that the J2EE data source OracleSGDS be defined. After SPARQL Gateway retrieves a database connection from the OracleSGDS data source, a SPARQL query is read from the database table ORACLE_ORARDF_SG_QUERY using the integer ID provided.

    The syntax for fetching a SPARQL query from an Oracle database is dq=<integer-id>, and the syntax for fetching an XSL transformation from an Oracle database is dx=<integer-id>.

    Upon servlet initialization, the following tables are created automatically if they do not already exist (you do not need to create them manually):

    • ORACLE_ORARDF_SG_QUERY with a primary key of QID (integer type)

    • ORACLE_ORARDF_SG_XSLT with a primary key of XID (integer type)

7.16.3.2 Specifying a Timeout Value

When you submit a potentially long-running query using the URL format described in Section 7.16.3, you can limit the execution time by specifying a timeout value in milliseconds. For example, the following shows the URL format and a timeout specification that the SPARQL query execution started from SPARQL Gateway is to be ended after 1000 milliseconds (1 second):

http://host:port/sparqlgateway/sg?<SPARQL_ENDPOINT>&<SPARQL_QUERY>&<XSLT>&t=1000

If a query does not finish when timeout occurs, then an empty SPARQL response is constructed by SPARQL Gateway.

Note that even if SPARQL Gateway times out a query execution at the HTTP connection level, the query may still be running on the server side. The actual behavior will be vendor-dependent.

7.16.3.3 Specifying Best Effort Query Execution


Note:

You can specify best effort query execution only if you also specify a timeout value (described in Section 7.16.3.2).

When you submit a potentially long-running query using the URL format described in Section 7.16.3, if you specify a timeout value, you can also specify a "best effort" limitation on the query. For example, the following shows the URL format with a timeout specification of 1000 milliseconds (1 second) and a best effort specification (&b=t):

http://host:port/sparqlgateway/sg?<SPARQL_ENDPOINT>&<SPARQL_QUERY>&<XSLT>&t=1000&b=t

The web.xml file includes two parameter settings that affect the behavior of the best effort option: sparql_gateway_besteffort_maxrounds and sparql_gateway_besteffort_maxthreads. The following show the default definitions:

<init-param>
  <param-name>sparql_gateway_besteffort_maxrounds</param-name>
  <param-value>10</param-value>
</init-param>
 
<init-param>
  <param-name>sparql_gateway_besteffort_maxthreads</param-name>
  <param-value>3</param-value>
</init-param>

When a SPARQL SELECT query is executed in best effort style, a series of queries will be executed with an increasing LIMIT value setting in the SPARQL query body. (The core idea is based on the observation that a SPARQL query runs faster with a smaller LIMIT setting.) SPARQL Gateway starts query execution with a "LIMIT 1" setting. Ideally, this query can finish before the timeout is due. Assume that is the case, the next query will have its LIMIT setting is increased, and subsequent queries have higher limits. The maximum number of query executions is controlled by the sparql_gateway_besteffort_maxrounds parameter.

If it is possible to run the series of queries in parallel, the sparql_gateway_besteffort_maxthreads parameter controls the degree of parallelism.

7.16.3.4 Specifying a Content Type Other Than text/xml

By default, SPARQL Gateway assumes that XSL transformations generate XML, and so the default content type set for HTTP response is text/xml. However, if your application requires a response format other than XML, you can specify the format in an additional URL parameter (with syntax &rt=), using the following format:

http://host:port/sparqlgateway/sg?<SPARQL_ENDPOINT>&<SPARQL_QUERY>&<XSLT>&rt=<content_type>

Note that <content_type> must be URL encoded.

7.16.4 Customizing the Default XSLT File

You can customize the default XSL transformation file (the one referenced using wx=default.xslt). This section presents some examples of customizations.

The following example implements this namespace prefix replacement logic: if a variable binding returns a URI that starts with http://purl.org/goodrelations/v1#, that portion is replaced by gr:; and if a variable binding returns a URI that starts with http://www.w3.org/2000/01/rdf-schema#, that portion is replaced by rdfs:.

<xsl:when test="starts-with(text(),'http://purl.org/goodrelations/v1#')">
   <xsl:value-of select="concat('gr:',substring-after(text(),'http://purl.org/goodrelations/v1#'))"/>
</xsl:when>
...
<xsl:when test="starts-with(text(),'http://www.w3.org/2000/01/rdf-schema#')">
   <xsl:value-of select="concat('rdfs:',substring-after(text(),'http://www.w3.org/2000/01/rdf-schema#'))"/>
</xsl:when>

The following example implements logic to trim a leading http://localhost/ or a leading http://127.0.0.1/.

<xsl:when test="starts-with(text(),'http://localhost/')">
  <xsl:value-of select="substring-after(text(),'http://localhost/')"/>
</xsl:when>
<xsl:when test="starts-with(text(),'http://127.0.0.1/')">
  <xsl:value-of select="substring-after(text(),'http://127.0.0.1/')"/>
</xsl:when>

7.16.5 Using the SPARQL Gateway Java API

In addition to a Web interface, the SPARQL Gateway administration service provides a convenient Java application programming interface (API) for managing SPARQL queries and their associated XSL transformations. The Java API is included in the Jena Adapter library, sdordfclient.jar.

Java API reference information is available in the javadoc_sparqlgateway.zip file that is included in the SPARQL Gateway .zip file (described in Section 7.16.2.1).

The main entry point for this API is the oracle.spatial.rdf.client.jena.SGDBHandler class (SPARQL Gateway Database Handler), which provides the following static methods for managing queries and transformations:

These methods manipulate and retrieve entries in the SPARQL Gateway associated tables that are stored in an Oracle Database instance. To use these methods, the necessary associated tables must already exist. If the tables do not exist, deploy the SPARQL Gateway on a Web server and access a URL in the following format:

http://<host>:<port>/sparqlgateway/sg?

where <host> is the host name of the Web server and <port> is the listening port of the Web server. Accessing this URL will automatically create the necessary tables if they do not already exist.

Any changes made through the Java API affect the SPARQL Gateway Web service in the same way as changes made through the administration Web interface. This provides the flexibility to manage queries and transformations using the interface you find most convenient.

Note that the insert methods provided by the Java API will not replace existing queries or transformations stored in the tables. Attempting to replace an existing query or transformation will fail. To replace a query or transformation, you must remove the existing entry in the table using one of the delete methods, and then insert the new query or transformation using one of the insert methods.

The following examples demonstrate how to perform common management tasks using the Java API. The examples assume a connection has already been established to the underlying Oracle Database instance backing the SPARQL Gateway.

Example 7-31 adds a query and an XSL transformation to the database backing the SPARQL Gateway. After the query and transformation are added, other programs can use the query and transformation through the gateway by specifying the appropriate query ID (qid) and XSL transformation ID (xid) in the request URL.


Note:

Although Example 7-31 inserts both a query and transformation, the query and transformation are not necessarily related and do not need to be used together when accessing SPARQL Gateway. Any query in the database can be used with any transformation in the database when submitting a request to SPARQL Gateway.

Example 7-31 Storing a SPARQL Query and an XSL Transformation

String query = "PREFIX ... SELECT ..."; // full SPARQL query text
String xslt  = "<?xml ...> ...";        // full XSLT transformation text
 
String queryDesc = "Conference attendee information"; // description of SPARQL query
String xsltDesc = "BIEE table widget transformation"; // description of XSLT transformation
 
int queryId = queryIdCounter++; // assign a unique ID to this query
int xsltId  = xsltIdCounter++;  // assign a unique ID to this transformation
 
// Inserting a query or transformation will fail if the table already contains
// an entry with the same ID.  Setting this boolean to true will ignore these
// exceptions (but the table will remain unchanged). Here we specify that we
// want an exception thrown if we encounter a duplicate ID.
boolean ignoreDupException = false;
 
// add the query
try {
  // Delete query if one already exists with this ID (this will not throw an
  // error if no such entry exists)
  SGDBHandler.deleteSparqlQuery( connection, queryId );
  SGDBHandler.insertSparqlQuery( connection, queryId, query, queryDesc, ignoreDupException );
} catch( SQLException sqle ) {
  // Handle exception
} catch( QueryException qe ) {
  // Handle query syntax exception
}
 
// add the XSLT
try {
  // Delete xslt if one already exists with this ID (this will not throw an
  // error if no such entry exists)
  SGDBHandler.deleteXslt( connection, xsltId );
  SGDBHandler.insertXslt( connection, xsltId, xslt, xsltDesc, ignoreDupException );
}  catch( SQLException sqle ) {
  // Handle database exception
} catch( TransformerConfigurationException tce ) {
  // Handle XSLT syntax exception
}

Example 7-32 retrieves an existing query from the database, modifies it, then stores the updated version of the query back in the database. These steps simulate editing a query and saving the changes. (Note that if the query does not exist, an exception is thrown.)

Example 7-32 Modifying a Query

StringBuilder query;
StringBuilder description;
 
// Populate these with the query text and description from the database
query = new StringBuilder( );
description = new StringBuilder( );
 
// Get the query from the database
try {
  SGDBHandler.getSparqlQuery( connection, queryId, query, description );
} catch( SQLException sqle ) {
  // Handle exception
  // NOTE: exception is thrown if query with specified ID does not exist
}
 
// The query and description should be populated now
 
// Modify the query 
String updatedQuery = query.toString( ).replaceAll("invite", "attendee");
 
// Insert the query back into the database
boolean ignoreDup = false;
try {
  // First must delete the old query
  SGDBHandler.deleteSparqlQuery( connection, queryId );
  // Now we can add
  SGDBHandler.insertSparqlQuery( connection, queryId, updatedQuery, description.toString( ), ignoreDup );
} catch( SQLException sqle ) {
  // Handle exception
} catch( QueryException qe ) {
  // Handle query syntax exception
}

Example 7-33 retrieves an existing XSL transformation and prints it to standard output. (Note that if the transformation does not exist, an exception is thrown.)

Example 7-33 Retrieving and Printing an XSL Transformation

StringBuilder xslt;
StringBuilder description;
 
// Populate these with the XSLT text and description from the database
xslt = new StringBuilder( );
description = new StringBuilder( );
 
try {
  SGDBHandler.getXslt( connection, xsltId, xslt, description );
} catch( SQLException sqle ) {
  // Handle exception
  // NOTE: exception is thrown if transformation with specified ID does not exist
}
 
// Print it to standard output
System.out.printf( "XSLT description: %s\n", description.toString( ) );
System.out.printf( "XSLT body:\n%s\n", xslt.toString( ) );

7.16.6 Using the SPARQL Gateway Graphical Web Interface

SPARQL Gateway provides several browser-based interfaces to help you test queries, navigate semantic data, and manage SPQARQL query and XSLT files:

7.16.6.1 Main Page (index.html)

http://<host>:<port>/sparqlgateway/index.html provides a simple interface for executing SPARQL queries and then applying the transformations in the default.xslt file to the response. Figure 7-2 shows this interface for executing a query.

Figure 7-2 Graphical Interface Main Page (index.html)

Description of Figure 7-2 follows
Description of "Figure 7-2 Graphical Interface Main Page (index.html)"

Enter or select a SPARQL Endpoint, specify the SPARQL SELECT Query Body, and press Submit Query.

For example, if you specify http://dbpedia.org/sparql as the SPARQL endpoint and use the SPARQL query body from Figure 7-2, the response will be similar to Figure 7-3. Note that the default transformations (in default.xslt) have been applied to the XML output in this figure.

Figure 7-3 SPARQL Query Main Page Response

Description of Figure 7-3 follows
Description of "Figure 7-3 SPARQL Query Main Page Response"

7.16.6.2 Navigation and Browsing Page (browse.jsp)

http://<host>:<port>/sparqlgateway/browse.jsp provides navigation and browsing capabilities for semantic data. It works against any standard compliant SPARQL endpoint. Figure 7-4 shows this interface for executing a query.

Figure 7-4 Graphical Interface Navigation and Browsing Page (browse.jsp)

Description of Figure 7-4 follows
Description of "Figure 7-4 Graphical Interface Navigation and Browsing Page (browse.jsp)"

Enter or select a SPARQL Endpoint, specify the SPARQL SELECT Query Body, optionally specify a Timeout (ms) value in milliseconds and the Best Effort option, and press Submit Query.

The SPARQL response is parsed and then presented in table form, as shown in Figure 7-5.

Figure 7-5 Browsing and Navigation Page: Response

Description of Figure 7-5 follows
Description of "Figure 7-5 Browsing and Navigation Page: Response"

In Figure 7-5, note that URIs are clickable to allow navigation, and that when users move the cursor over a URI, tool tips are shown for the URIs which have been shortened for readability (as in http://purl.org.dc/elements/1.1/title being displayed as the tool tip for dc:title in the figure).

If you click the URI http://example.org/book/book5 in the output shown in Figure 7-5, a new SPARQL query is automatically generated and executed. This generated SPARQL query has three query patterns that use this particular URI as subject, predicate, and object, as shown in Figure 7-6. Such a query can give you a good idea about how this URI is used and how it is related to other resources in the data set.

Figure 7-6 Query and Response from Clicking URI Link

Description of Figure 7-6 follows
Description of "Figure 7-6 Query and Response from Clicking URI Link"

When there are many matches of a query, the results are organized in pages and you can click on any page. The page size by default is 50 results. To display more (or fewer) than 50 rows per page in a response with the Browsing and Navigation Page (browse.jsp), you can specify the &resultsPerPage parameter in the URL. For example, to allow 100 rows per page, include the following in the URL:

&resultsPerPage=100

7.16.6.3 XSLT Management Page (xslt.jsp)

http://<host>:<port>/sparqlgateway/admin/xslt.jsp provides a simple XSLT management interface. You can enter an XSLT ID (integer) and click Get XSLT to retrieve both the Description and XSLT Body. You can modify the XSLT Body text and then save the changes by clicking Save XSLT. Note that there is a previewer to help you navigate among available XSLT definitions.

Figure 7-7 shows the XSLT Management Page.

Figure 7-7 XSLT Management Page

Description of Figure 7-7 follows
Description of "Figure 7-7 XSLT Management Page"

7.16.6.4 SPARQL Management Page (sparql.jsp)

http://<host>:<port>/sparqlgateway/admin/xslt.jsp provides a simple SPARQL management interface. You can enter a SPARQL ID (integer) and click Get SPARQL to retrieve both the Description and SPARQL Body. You can modify the SPARQL Body text and then save the changes by clicking Save SPARQL. Note that there is a previewer to help you navigate among available SPARQL queries.

Figure 7-8 shows the SPARQL Management Page.

Figure 7-8 SPARQL Management Page

Description of Figure 7-8 follows
Description of "Figure 7-8 SPARQL Management Page"

7.16.7 Using SPARQL Gateway as an XML Data Source to OBIEE

This section explains how to create an XML Data source for Oracle Business Intelligence Enterprise Edition (OBIEE), by integrating OBIEE with RDF using SPARQL Gateway as a bridge. (The specific steps and illustrations reflect the Oracle BI Administration Tool Version 11.1.1.3.0.100806.0408.000.)

  1. Start the Oracle BI Administration Tool.

  2. Click File, then Import Metadata. The first page of the Import Metadata wizard is displayed, as shown in Figure 7-9.

    Figure 7-9 Import Metadata - Select Data Source

    Description of Figure 7-9 follows
    Description of "Figure 7-9 Import Metadata - Select Data Source"

    Connection Type: Select XML.

    URL: URL for an application to interact with SPARQL Gateway, as explained in Section 7.16.3. You can also include the timeout and best effort options.

    Ignore the User Name and Password fields.

  3. Click Next. The second page of the Import Metadata wizard is displayed, as shown in Figure 7-10.

    Figure 7-10 Import Metadata - Select Metadata Types

    Description of Figure 7-10 follows
    Description of "Figure 7-10 Import Metadata - Select Metadata Types"

    Select the desired metadata types to be imported. Be sure that Tables is included in the selected types.

  4. Click Next. The third page of the Import Metadata wizard is displayed, as shown in Figure 7-11.

    Figure 7-11 Import Metadata - Select Metadata Objects

    Description of Figure 7-11 follows
    Description of "Figure 7-11 Import Metadata - Select Metadata Objects"

    In the Data Source View, expand the node that has the table icon, select the column names (mapped from projected variables defined in the SPARQL SELECT statement), and click the right-arrow (>) button to move the selected columns to the Repository View.

  5. Click Finish.

  6. Complete the remaining steps for the usual BI Business Model work and Mapping and Presentation definition work, which are not specific to SPARQL Gateway or RDF data.

PK ]PK.AOEBPS/sem_glossary.htmV' Glossary

Glossary

The following are selected terms relevant to the Oracle Database implementation of semantic technologies support. This is not a comprehensive RDF and OWL glossary.

apply pattern

Part of a data access constraint defines additional graph patterns to be applied on the resources that match the match pattern before they can be used to construct the query results. See also: match pattern

basic graph pattern (BGP)

A set of triple patterns. From the W3C SPARQL Query Language for RDF Recommendation: "SPARQL graph pattern matching is defined in terms of combining the results from matching basic graph patterns. A sequence of triple patterns interrupted by a filter comprises a single basic graph pattern. Any graph pattern terminates a basic graph pattern."

clique

A graph in which every node of it is connected to, bidirectionally, every other node in the same graph.

Cytoscape

An open source bioinformatics software platform for visualizing molecular interaction networks and integrating these interactions with gene expression profiles and other state data. (See http://www.cytoscape.org/.) An RDF viewer (available for download) is provided as a Cytoscape plug-in.

entailment

An object containing precomputed triples that can be inferred from applying a specified set of rulebases to a specified set of models. See also: rulebase

extractor policy

A named dictionary entity that determines the characteristics of a semantic index that is created using the policy. Each extractor policy refers, directly or indirectly, to an instance of an extractor type.

graph pattern

A combination of triples constructed by combining triple patterns in various ways, including conjunction of triple patterns into groups, optionally using filter conditions, and then combining such groups via connectors similar to disjunctions, outer-joins, and so on. SPARQL querying is based around graph pattern matching.

inferencing

The ability to make logical deductions based on rules. Inferencing enables you to construct queries that perform semantic matching based on meaningful relationships among pieces of data, as opposed to just syntactic matching based on string or other values. Inferencing involves the use of rules, either supplied by Oracle or user-defined, placed in rulebases.

information extractor

An application that processes unstructured documents and extract meaningful information from them, often using natural-language processing engines with the aid of ontologies.

Jena Adapter for Oracle Database

An Oracle-supplied adapter (available for download) for Jena, which is a Java framework for building Semantic Web applications.

match pattern

Part of a constraint that determines the type of access restriction it enforces and binds one or more variables to the corresponding data instances accessed in the user query. See also: apply pattern

model

A user-created semantic structure that has a model name, and refers to triples stored in a specified table column. Examples in this manual are the Articles and Family models.

ontology

A shared conceptualization of knowledge in a particular domain. It consists of a collection of classes, properties, and optionally instances. Classes are typically related by class hierarchy (subclass/ superclass relationship). Similarly, the properties can be related by property hierarchy (subproperty/ superproperty relationship). Properties can be symmetric or transitive, or both. Properties can also have domain, ranges, and cardinality constraints specified for them.

OWLPrime

An Oracle-defined subset of OWL capabilities; refers to the elements of the OWL standard supported by the Oracle Database semantic technologies native inferencing engine.

reasoning

See inferencing

rule

An object that can be applied to draw inferences from semantic data.

rulebase

An object that can contain rules. See also: rule

rules index

See: entailment

semantic index

An index of type MDSYS.SEMCONTEXT, created on textual documents stored in a column of a table, and used with information extractors to locate and extract meaningful information from unstructured documents. See also: information extractor

Sesame Adapter for Oracle Database

An Oracle-supplied adapter (available for download) that integrates the popular Sesame Java APIs with Oracle Semantic Technologies support.

Simple Knowledge Organization System (SKOS)

A data model that is especially useful for representing thesauri, classification schemes, taxonomies, and other types of controlled vocabulary. SKOS is based on standard semantic web technologies including RDF and OWL, which makes it easy to define the formal semantics for those knowledge organization systems and to share the semantics across applications.

triple pattern

Similar to an RDF triple, but allows use of a variable in place of any of the three components (subject, predicate, or object). Triple patterns are basic elements in graph patterns used in SPARQL queries. A triple pattern used in a query against an RDF graph is said to match if, substitution of RDF terms for the variables present in the triple pattern, creates a triple that is present in the RDF graph. See also: graph pattern

PKARo['V'PK.AOEBPS/sem_enable.htmGi Enabling, Downgrading, or Removing Semantic Technologies Support

A Enabling, Downgrading, or Removing Semantic Technologies Support

This appendix describes the required steps that you must perform before you can use any types, synonyms, or PL/SQL packages related to Oracle semantic technologies support in the current Oracle Database release. You must run one or more scripts, and you must ensure that Spatial is installed and the Partitioning option is enabled. These requirements are explained in Section A.1.

This appendix also describes the steps if, after enabling semantic technologies support, you need to do any of the following:

A.1 Enabling Semantic Technologies Support

Before you can use any types, synonyms, or PL/SQL packages related to Oracle semantic technologies support in the current Oracle Database release, you must either install the capabilities in a new Oracle Database installation or upgrade the capabilities from a previous release, following the steps in whichever of the following sections applies to your situation:

If you upgraded to Release 11.2.0.2.0 and your Semantic Technologies installation is invalid, see Section A.1.4, "Release 11.2.0.2: Required Actions if Semantic Technologies Installation is Invalid"

In addition, you must ensure that Spatial is installed and the Partitioning option is enabled, as explained in Section A.1.5.

A.1.1 Enabling Semantic Technologies Support in a New Database Installation

To enable semantic technologies support in a new installation of Oracle Database Release 11.2 (or for an upgrade from Release 10.2 if no RDF network exists, as explained in Section A.1.3), follow these steps:

  1. Connect to the database as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted).

  2. Start SQL*Plus, and enter the following statement:

    • Linux: @$ORACLE_HOME/md/admin/catsem.sql

    • Windows: @%ORACLE_HOME%\md\admin\catsem.sql

If a semantic network already exists, or if any semantic technologies types, tables, or PL/SQL packages already exist, the catsem.sql script exits with an error.

If the script completes successfully, a row with the following column values is inserted into the MDSYS.RDF_PARAMETER table:

  • NAMESPACE: MDSYS

  • ATTRIBUTE: SEM_VERSION

  • VALUE: (string starting with 112)

  • DESCRIPTION: VALID

If you use Workspace Manager with RDF data, you must also run the appropriate script after you run catsem.sql, as explained in Section 6.1, "Enabling Workspace Manager Support for RDF Data".

A.1.2 Upgrading Semantic Technologies Support from Release 11.1

If you are upgrading from Oracle Database Release 11.1 that includes the semantic technologies support for Release 11.1, the semantic technologies support is automatically upgraded to Release 11.2 when the database is upgraded. If you use Workspace Manager with RDF data, you must also run the appropriate script after the upgrade of semantic technologies support, as explained in Section 6.1, "Enabling Workspace Manager Support for RDF Data".

However, you will also need to migrate RDF data if you have an existing Release 11.1 RDF network containing triples that include typed literal values of type xsd:float, xsd:double, xsd:boolean, or xsd:time.

To check if you need to migrate RDF data, connect to the database as a user with DBA privileges and query the MDSYS.RDF_PARAMETER table, as follows:

SELECT namespace, attribute, value FROM mdsys.rdf_parameter
  WHERE namespace='MDSYS' 
  AND attribute IN ('FLOAT_DOUBLE_DECIMAL',
                    'XSD_TIME', 'XSD_BOOLEAN', 
                    'DATA_CONVERSION_CHECK');

If the FLOAT_DOUBLE_DECIMAL, XSD_TIME, or XSD_BOOLEAN attributes have the string value INVALID or if the DATA_CONVERSION_CHECK attribute has the string value FAILED_UNABLE_TO_LOCK_APPLICATION_TABLES, FAILED_INSUFFICIENT_WORKSPACE_PRIVILEGES, or FAILED_OLS_POLICIES_ARE_ENABLED, you need to migrate RDF data.

However, if the FLOAT_DOUBLE_DECIMAL, XSD_TIME, and XSD_BOOLEAN attributes do not exist or have the string value VALID and if the DATA_CONVERSION_CHECK attribute does not exist, you do not need to migrate RDF data. However, if your semantic network may have any empty RDF literals, see Section A.1.2.1, "Handling of Empty RDF Literals"; and if you choose to migrate existing empty literals to the new format, follow the steps in this section.

To migrate RDF data, follow these steps:

  1. Connect to the database as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted), and enter: SET CURRENT_SCHEMA=MDSYS

  2. Ensure that the user MDSYS has the following privileges:

    • INSERT privilege on all application tables in the semantic network

    • ALTER ANY INDEX privilege (optional: only necessary if Semantic Indexing for Documents is being used)

    • ACCESS privilege for any workspace in which version-enabled application tables have been modified (optional: only necessary if Workspace Manager is being used for RDF data)

  3. Ensure that any OLS policies for RDF data are temporarily disabled (optional: only necessary if OLS for RDF Data is being used). OLS policies can be re-enabled after running convert_old_rdf_data.

  4. Start SQL*Plus. If you want to migrate the RDF data without converting existing empty literals to the new format (see Section A.1.2.1), enter the following statement:

    EXECUTE sdo_rdf_internal.convert_old_rdf_data;
    

    If you want to migrate the RDF data and also convert existing empty literals to the new format, call convert_old_rdf_data with the flags parameter set to 'CONVERT_ORARDF_NULL'. In addition, you can use an optional tablespace_name parameter to specify the tablespace to use when creating intermediate tables during data migration. For example, the following statement migrates old semantic data, converts existing "orardf:null " values to "", and uses the MY_TBS tablespace for any intermediate tables:

    EXECUTE sdo_rdf_internal.convert_old_rdf_data(
      flags=>'CONVERT_ORARDF_NULL', 
      tablespace_name=>'MY_TBS');
    

    The sdo_rdf_internal.convert_old_rdf_data procedure may take a significant amount of time to run if the semantic network contains many triples that are using (or affected by use of) xsd:float, xsd:double, xsd:time, or xsd:boolean typed literals.

  5. Enter the following statement:

    • Linux: @$ORACLE_HOME/md/admin/semrelod.sql

    • Windows: @%ORACLE_HOME%\md\admin\semrelod.sql


Note:

You may encounter the ORA-00904 (invalid identifier) error when executing a SEM_MATCH query if the sdo_rdf_internal.convert_old_rdf_data procedure and the semrelod.sql script were not run after the upgrade to Release 11.2.

A.1.2.1 Handling of Empty RDF Literals

The way empty-valued RDF literals are handled has changed in Release 11.2. Before this release, the values of empty-valued literals were converted to "orardf:null ". In Release 11.2, such values are stored without modification (that is, as ""). However, whether you migrate existing "orardf:null " values to "" is optional.

To check if "orardf:null " values exist in your semantic network, connect to the database as a user with DBA privileges and query the MDSYS.RDF_PARAMETER table, as follows:

SELECT namespace, attribute, value FROM mdsys.rdf_parameter
  WHERE namespace='MDSYS' 
  AND attribute = 'NULL_LITERAL';

If the NULL_LITERAL attribute has the value EXISTS, then "orardf:null " values are present in your semantic network.

A.1.3 Upgrading Semantic Technologies Support from Release 10.2

If you are upgrading from Oracle Database Release 10.2 that includes the semantic technologies support for Release 10.2, the actions required depend on whether there is an existing RDF network.

If there is not an existing RDF network, skip the rest of this section and follow the instructions in Section A.1.1, "Enabling Semantic Technologies Support in a New Database Installation".

If there is an existing RDF network, perform the following steps after the Oracle Database upgrade:

  1. Connect to the database (Release 11.2) as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted).

  2. Start SQL*Plus, and enter the following statement:

    • Linux: @$ORACLE_HOME/md/admin/catsem10i.sql

    • Windows: @%ORACLE_HOME%\md\admin\catsem10i.sql

    The catsem10i.sql script enables Release 11.2 semantic technologies support and migrates RDF data from Release 10.2 to Release 11.2 format. The script may take a long time to run if the existing Release 10.2 RDF network contains a large amount of RDF data.

A.1.4 Release 11.2.0.2: Required Actions if Semantic Technologies Installation is Invalid

Further action may be required if your Semantic Technologies installation is invalid after upgrading to Release 11.2.0.2.0. (This is not an issue with upgrades to Release 11.2.0.3 and later.) Action is required if either or both of the following are true after upgrading to Release 11.2.0.2.0:

  • A row exists with the following values in the MDSYS.RDF_PARAMETER table:

    MDSYS   SEM_VERSION   11202   INVALID
    
  • The STATUS column value is INVALID in any row returned by the following query:

    SELECT object_name, status FROM all_objects WHERE object_name = 'SEMCONTEXTINDEXMETHODS' or object_name = 'PRLLIDX_CREATE';
    

In such a case, enter the following sequence of commands as SYSDBA to complete the upgrade of your installation:

SQL> @ $ORACLE_HOME/md/admin/sdordfctx.sql
SQL> @ $ORACLE_HOME/md/admin/semrelod.sql
SQL> EXECUTE sys.validate_sdo;

For the preceding commands:

  • When running sdordfctx.sql, you can ignore any ORA-01921, ORA-02303, ORA-00955, and ORA-29809 errors.

  • Executing sys.validate_sdo is needed only if SDO is invalid, which it could be if Semantic Technologies is invalid; however, there is no harm in executing sys.validate_sdo in any case.

After running the preceding commands, check whether your Semantic Technologies installation is valid by checking if a row exists with the following values in the MDSYS.RDF_PARAMETER table:

MDSYS   SEM_VERSION   11202   VALID

A.1.5 Spatial and Partitioning Requirements

Oracle Spatial must be installed before you can use any of the RDF and OWL capabilities. Oracle Locator is not sufficient. For information about Spatial and Locator, see Oracle Spatial Developer's Guide.

The Partitioning option must be enabled before you can use any of the RDF and OWL capabilities. For licensing information about the Partitioning option, see Oracle Database Licensing Information. For usage information about partitioning, see Oracle Database VLDB and Partitioning Guide.

A.2 Downgrading Semantic Technologies Support to a Previous Release

This section explains how to downgrade the semantic technologies support, in conjunction with an Oracle Database downgrade to Release 11.1 or 10.2.

Downgrading is strongly discouraged, except for rare cases where it is necessary. If you downgrade to a previous release, you will not benefit from bug fixes and enhancements that have been made in intervening releases.

If you have enabled Workspace Manager support for RDF data, you must remove that support before you downgrade semantic technologies support in the database, as explained in Section 6.1.1.

A.2.1 Downgrading to Release 11.1 Semantic Technologies Support

If you need to downgrade to Oracle Database Release 11.1, the semantic technologies component will be downgraded automatically when you downgrade the database. However, any RDF or OWL data that is specific to Release 11.2 (that is, Release 11.2 RDF/OWL persistent structures that are not supported in previous versions) must be dropped before you perform the downgrade, so that the database is compatible with Release 11.1.

Any virtual models will not be preserved during a downgrade, because they are only available through a Release 11.1.0.7.0 patch (number 7600122) that was made available on My Oracle Support (MetaLink) in November, 2008. To use virtual models, you must reapply this patch after the downgrade, and then re-create the virtual models. For convenience, you can use the semvmrecreate.sql script to generate a script that you can later use (after the downgrade) to automatically re-create any virtual models that were present in the Release 11.2 semantic network. To do so, follow these steps before you downgrade the database :

  1. Connect to the database (Release 11.2) as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted).

  2. Start SQL*Plus, and enter a statement in the following format:

    • Linux: @$ORACLE_HOME/md/admin/semvmrecreate.sql /my_dir_path/my_vm_recreate.sql

    • Windows: @%ORACLE_HOME%\md\admin\semvmrecreate.sql \my_dir_path\my_vm_recreate.sql

    where my_dir_path is the path to the location in which to create the my_vm_recreate.sql script file. (You can use another file name for the script, if you want.)

To check if any Release 11.2 RDF data is incompatible with Release 11.1, perform the following steps:

  1. Connect to the database (Release 11.2) as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted).

  2. Start SQL*Plus, and enter the following statements:

    SET SERVEROUT ON
    EXECUTE SDO_SEM_DOWNGRADE.CHECK_111_COMPATIBLE;
    

If any RDF data is incompatible with Release 11.1, the procedure generates an error and displays a list of the incompatible data. In this case, you must run the semrelod111.sql script after the database downgrade, as follows:

  1. In the Release 11.2 database, save a copy of the semrelod111.sql file, which is in the following location:

    • Linux: $ORACLE_HOME/md/admin/semrelod111.sql

    • Windows: %ORACLE_HOME%\md\admin\semrelod111.sql

  2. If you have any virtual models that you want to be able to re-create after the download, run the semvmrecreate.sql script (if you have not already done so), as explained earlier in this section.

  3. Remove any Release 11.2 release-specific RDF or OWL data (including virtual models) if you have not already done so, as explained earlier in this section.

  4. Perform the database downgrade.

  5. In the Release 11.1 database, copy the saved copy of the semrelod111.sql file to the following location:

    • Linux: $ORACLE_HOME/md/admin/

    • Windows: %ORACLE_HOME%\md\admin\

  6. Connect to the Release 11.1 database as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted).

  7. Start SQL*Plus, and enter the following statement:

    • Linux: @$ORACLE_HOME/md/admin/semrelod111.sql

    • Windows: @%ORACLE_HOME%\md\admin\semrelod111.sql

    After the semrelod111.sql script completes successfully, Oracle semantic technologies support for Release 11.1 is enabled and ready to use, and all Release 11.1-compatible data is preserved.

  8. If you want to enable support for virtual models, apply the Release 11.1.0.7 interim patch number 7600122.

  9. If you previously generated a my_vm_recreate.sql script to re-create your virtual models after the downgrade, go to the my_dir_path location, connect to the database as the SYS user with SYSDBA privileges, start SQL*Plus, and run the my_vm_recreate.sql script.

A.2.2 Downgrading to Release 10.2 Semantic Technologies Support

If you need to downgrade to Oracle Database Release 10.2, follow the instructions in Section A.2.1 to downgrade to Release 11.1, and then follow the instructions in the rest of this section (taken from the Release 11.1 manual) to downgrade from Release 11.1 to Release 10.2.

If you need to downgrade to Oracle Database Release 10.2, and if you used Oracle Database Release 11 RDF or OWL features and want to preserve existing semantic data and rulebases, you must execute a statement to prepare for the downgrade, perform the downgrade, and execute another statement to restore the semantic data.

However, the following considerations apply:

  • Entailed graph data will not be preserved, because the same information can be regenerated using the Oracle Database Release 10.2 RDF inference API.

  • No rulebases or rules indexes related to OWL are preserved, because Oracle Database Release 10.2 did not support the OWL vocabulary.

Perform the following steps:

  1. Before the database downgrade, connect to the Release 11 database as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted).

  2. Start SQL*Plus and enter the following statement:

    EXECUTE SDO_SEM_DOWNGRADE.PREPARE_DOWNGRADE_FROM_11;
    

    When this statement executes successfully, all existing semantic data and rulebases are saved. You will restore the semantic data after the database downgrade.

  3. Perform the database downgrade.

  4. Download the following file from the Semantic Technologies page of the Oracle Technology Network site: sdosemdgu.plb

  5. If (and only if) your Oracle Database Release 10.2 release number is 10.2.0.1, click the Software link, and download and install the RDF-specific patch. (This patch is needed because Release 10.2.0.1 did not have the batch loading feature, which is used to restore the semantic data.)

  6. Connect to the Release 10.2 database as the SYS user with SYSDBA privileges.

  7. Start SQL*Plus and enter a statement in the following statement:

    EXECUTE SDO_SEM_DOWNGRADE_UTL.PREPARE_DOWNGRADE_TO_102('<tablespace-name>');
    

    Where <tablespace-name> is the name of the tablespace in which the RDF network will be created.

    When this statement executes successfully, all semantic data that had been saved before the downgrade is restored and ready to use.

A.3 Removing Semantic Technologies Support

This section explains how to remove the semantic technologies support from the database. Removing this support is strongly discouraged, unless you have a strong reason for doing it. After you remove this support, no applications or database users will be able to use any types, synonyms, or PL/SQL packages related to Oracle semantic technologies support.

If you have enabled Workspace Manager support for RDF data, you must remove that support before you remove semantic technologies support from the database, as explained in Section 6.1.1.

To remove the semantic technologies support from the database, perform the following steps:

  1. Connect to the database (Release 11.2) as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted).

  2. Start SQL*Plus, and enter the following statement:

    • Linux: @$ORACLE_HOME/md/admin/semremov.sql

    • Windows: @%ORACLE_HOME%\md\admin\semremov.sql

The semremov.sql script drops the semantic network and removes any semantic technologies types, tables, and PL/SQL packages:

PKfLiGiPK.AOEBPS/fine_grained_acc.htm Fine-Grained Access Control for RDF Data

5 Fine-Grained Access Control for RDF Data

The default control of access to the Oracle Database semantic data store is at the model level: the owner of a model may grant select, delete, and insert privileges on the model to other users by granting appropriate privileges on the view named RDFM_<model_name>. However, for applications with stringent security requirements, you can enforce a fine-grained access control mechanism by using either the Virtual Private Database feature or the Oracle Label Security option of Oracle Database:

Some factors to consider in choosing whether use VPD or OLS with RDF data include the following:

The application programming interface (API) for implementing VPD or OLS with semantic data is provided in the SEM_RDFSA PL/SQL package. Chapter 13 provides reference information about the programs in the SEM_RDFSA package.

VPD and OLS support for RDF data is included in the semantic technologies support for Release 11.2. (For information about enabling, downgrading, or removing semantic technologies support, see Appendix A.)

This chapter contains the following major sections:

5.1 Virtual Private Database (VPD) for RDF Data

The Virtual Private Database (VPD) feature is a row-level security mechanism that restricts access to specific rows in a relational table or view using a security policy and an application context. The security policy includes a policy function that dynamically generates predicates that are enforced for each row returned for the user query. The security predicates returned by the policy function associated with a table are typically expressed using the columns in the table and are thus dependent on the table metadata. Effectively, the security predicates ensure that the rows returned for a user query satisfy additional conditions that are applied on the contents of the row.

When the relational data is mapped to RDF, the data stored in a specific relational table represent triples describing instances of a specific RDF class. In this representation, the columns in the relational table map to RDF properties that are used to describe a resource. This mapping may be further extended to the application of VPD policies.

A VPD policy applied to RDF data restricts users' access to instances of a specific RDF class or property by applying predicates, in the form of graph patterns and filter conditions, on the instance data. For example, a VPD policy may be defined to restrict access to instances of a Contract RDF class only to the users belonging to a specific department. Furthermore, access to the hasContractValue property for a resource identified as instance of the Contract RDF class may be restricted only to the manager of the contract. VPD support for RDF data allows security conditions or data access constraints to be associated with RDF classes and properties, so that access to corresponding instance data is restricted.

A data access constraint associated with an RDF class or property specifies a graph query pattern that must be enforced for all corresponding data instances that are returned as the query result. For example, a SPARQL query pattern to find the due dates for all instances of a Contract class, {?contract :hasDueDate ?due}, may activate a data access constraint that ensures that the information returned pertains to contracts belonging to a specific department. This is achieved by logically rewriting the user's graph query pattern to include additional graph patterns, as shown in the following example:

{ ?contract   :hasDueDate  ?due . 
  ?contract   :drivenBy    dept:Dept1 }

Furthermore, the values bound into the rewritten graph query pattern may make use of session context to enforce dynamic access restrictions. In the following example, the sys_context function in the object position of the triple pattern binds the appropriate department value based on the session context:

{ ?contract   :hasDueDate   ?due . 
  ?contract   :drivenBy
             "sys_context('sa$appctx','user_dept'}"^^orardf:instruction }

In a relational data model, the metadata, in the form of table definition, always exists along with the data (the rows stored in the table); thus, the VPD policies defined using the metadata are well formed and the security conditions are generated using a procedural logic in the policy function.

However, the RDF data model allows data with no accompanying metadata, and therefore the class information for instance data may not always be available for a given RDF graph. For example, in an RDF graph a resource known to be a contract might not accompany a triple that asserts that the resource is an instance of Contract class. Usually such triples can be inferred using available domain and range specifications for the properties describing the resource.

Similarly, a VPD policy relies on the properties' domain and range specifications for deriving the class information for the instance data and for enforcing appropriate data access constraints. However, to avoid runtime dependencies on the user data, a VPD policy maintains the minimal metadata required to derive the class information in its dictionary, separate from the asserted and inferred triples. This also ensures that the metadata maintained by a VPD policy is complete even when some necessary information is missing from the asserted triples and that a VPD policy, with its data access constraints and the metadata, is self-contained and portable with no external dependencies.

A VPD policy with specific data access constraints and RDF metadata specifications can be used to enforce access restrictions for the data stored in an RDF model. Each SPARQL query issued on the model is analyzed to deduce the class information for the resources accessed in the query, and appropriate data access constraints are applied. To facilitate the compile-time analysis and derivation of class information for instance data, a graph query pattern with an unbound predicate is restricted when a VPD policy is in effect. For example, a graph pattern of the following form, anywhere in a SPARQL query pattern, raises an exception when any underlying model has a VPD policy:

{ <contract:projectHLS>  ?pred  ?obj }

VPD policies are only enforced for SEM_MATCH queries expressed in SPARQL syntax. All other forms of data access (such as classic syntax for graph pattern or direct query on the model view) are not permitted.

5.1.1 VPD Policy for RDF Data

A VPD policy for RDF data is a named dictionary entity that can be used to enforce access restrictions for the data stored in one or more RDF models. A VPD policy defined for RDF data has unique characteristics, and it cannot be reused to enforce security policies for relational data. An RDF-VPD policy defined in the database includes the following:

  • The RDF Schema statements or metadata necessary for deriving class information for the data referenced in a SPARQL user query

  • The data access constraints that enforce access restrictions for the instance data

  • Application context that allows conditional inclusions of groups of data access constraints based on the runtime environment

An RDF-VPD policy is defined, owned, and maintained by a user with a security administrator role in an organization. This user must have at least EXECUTE privileges on the SYS.DBMS_RLS package. The owner of an RDF-VPD policy can maintain the metadata associated with the policy, define new data access constraints, and apply the policy to one or more RDF models.

A SPARQL query issued on an RDF model with a VPD policy is analyzed, and zero or more data access constraints defined in the policy are enforced such that the data instances that are returned as the query result also satisfy these constraints. The exact data access constraints enforced for a user query vary, based on the resources referenced in the query and the application context. For example, a policy that restricts a manager's access to the hasContractValue property may be relaxed for a user with the Vice President role.

Based on the role of the user, as captured in the application context, specific constraints to be applied are determined at runtime. To facilitate this dynamic inclusion of subsets of constraints defined in a VPD policy, the data access constraints are arranged into named groups that can be activated and deactivated based on the application context. During query analysis, only the constraints defined in the active groups are considered for enforcement.

The constraint groups within a VPD policy are managed using an application context and its package implementation. Each VPD policy can specify the namespace for a context created with the CREATE CONTEXT command. Each attribute associated with the context is treated as the name of a constraint group that can be activated by initializing its value to 1. For example, setting the value for MANAGER attribute of the context associated with a VPD policy to 1 will activate the constraints associated with MANAGER group for the user session. The logic that initializes specific constraint groups based on the user context is typically embedded in the package associated with the context type. The following example shows an excerpt from a sample implementation for one such package:

CREATE CONTEXT contracts_constr_ctx using sec_admin.contracts_ctx_pack;
 
begin
  -- create the VPD policy with a context -- 
  sem_rdfsa.create_vpd_policy(policy_name    => 'CONTRACTS_POLICY',
                              policy_context => 'contracts_constr_ctx');
end;
/
 
create or replace package sec_admin.contracts_ctx_pack as
  procedure init_constr_groups;
end;
/
 
create or replace package body sec_admin.contracts_ctx_pack as
  procedure init_contr_groups is 
    hrdata EmpRole%rowtype; 
  begin
    -- specific users with FULL access to the data associated with 
    -- the policy -- 
    if (sys_context('userenv', 'session_user') = 'RDF_ADMIN') then 
      dbms_session.set_context('contracts_constr_ctx',
                                sem_rdfsa.VPD_FULL_ACCESS, 1); 
      return;
    end if; 
 
    SELECT * into hrdata FROM EmpRole WHERE guid = 
                          sys_context('userenv','session_user');
 
    if (hrdata.emprole = 'VP') then 
      -- if the user logged in has VP role, activate the constraint
      -- group named VP and keep all other groups inactive. 
      dbms_session.set_context('contracts_constr_ctx','VP', '1'); 
    elsif (hrdata.emprole = 'MANAGER') then 
      dbms_session.set_context('contracts_constr_ctx', 'MANAGER', '1'); 
    elsif ...
      ...  
    else 
      raise_application_error(-20010, 'unknown user role'); 
    end if;
 
  exception when others then 
    -- enforce constraints for all groups --
    dbms_session.clear_all_context('contracts_constr_ctx');
  end init_contr_groups; 
end;
/

By default, when a namespace is not associated with an RDF-VPD policy or when a specific constraint group is not activated in a session, all the constraints defined in the policy are active and they are enforced for each user query. However, when a specific constraint group is activated by setting the corresponding namespace-attribute value to 1, only the constraints belonging to the group and any other constraints that are not associated with any group are enforced. For a given session, one or more constraint groups may be activated, in which case all the applicable constraints are enforced conjunctively.

At the time of creation, the data access constraints defined in a RDF-VPD policy may specify the name of a constraint group (explained in Section 5.1.3, "Data Access Constraints"). Within a database session, appropriate groups of constraints are activated based on the session context set by the context package. For all subsequent SPARQL queries in the database session, the constraints belonging to the active groups are consulted for enforcing appropriate security policies.

Maintenance operations on an RDF model with a VPD policy require unconditional access to data in the model. These operations include creation of an entailment using at least one VPD protected model, and load or data manipulation operations. You can grant unconditional access to the data stored in an RDF model by initializing a reserved attribute for the namespace associated with the VPD policy. The reserved attribute is defined by the package constant sem_rdfsa.VPD_FULL_ACCESS, and the context package implementation shown in the preceding example grants FULL access to the RDF_ADMIN user.

DML operations on the application table are not validated for VPD constraint violations, so only a user with FULL access to the corresponding model can add or modify existing triples.

You can use the SEM_MATCH operator to query an RDF model with a VPD policy in a standard SQL query, and to perform a multi-model query on a combination of VPD-enabled models and models with no VPD policy. However, when more than one model in a multi-model query is VPD-enabled, they must all be associated with the same VPD policy. A VPD policy associated with an RDF model is automatically extended to any data inferred from the model. When multiple RDF models are specified during inference, all VPD-enabled models within the set should use the same VPD policy.

5.1.2 RDF Metadata for Enforcing VPD Policies

The types of RDF metadata used to enforce VPD policies include the following:

  • Domain and range information for the properties used in the graph

  • Subclass relationships in the graph

  • Subproperty relationships in the graph

  • Equivalent properties in the graph

The RDF metadata associated with a VPD policy is specified as one or more RDF Schema statements using one of the following property URIs:

  • http://www.w3.org/2000/01/rdf-schema#domain

  • http://www.w3.org/2000/01/rdf-schema#range

  • http://www.w3.org/2000/01/rdf-schema#subClassOf

  • http://www.w3.org/2000/01/rdf-schema#subPropertyOf

  • http://www.w3.org/2002/07/owl#equivalentProperty

For example, the following RDF Schema statement associated with contracts_policy asserts that the domain of the hasContractValue property is a Contract class. Note that range specification for the predicates can be skipped if they are not relevant or if they are of literal type

begin
  sem_rdfsa.maint_vpd_metadata(
        policy_name => 'contracts_policy',
        t_subject   => '<http://www.myorg.com/pred/hasContractValue>',
        t_predicate => '<http://www.w3.org/2000/01/rdf-schema#domain>',
        t_object    => '<http://www.myorg.com/classes/Contract>');
end;
/

An RDF-VPD policy maintains its metadata separate from the asserted and inferred triples. You can derive this metadata programmatically from the RDF models and the corresponding entailments. For example, if the domain and range information for the properties and subclass and subproperty relationships are already established in the asserted or inferred triples, you can use a SQL query on the underlying model views to populate the metadata for an RDF-VPD policy.

The domain and range information for the properties aid the query analysis in determining the RDF class type for the terms and unbound variables referenced in the query. This information is further used to enforce appropriate data access constraints on the data accessed by the query. The metadata relating to the subclass property is used to ensure that a data access constraint defined for a specific class in a class hierarchy is automatically enforced for all its subclasses. Similarly, the subproperty specification in a VPD policy is used to enforce any constraints associated with a property to all its subproperties.

The RDF Schema statements associated with a VPD policy are not used to infer additional statements, and the security administrator should ensure that the metadata captured in a VPD policy is complete by cross checking it with inferred data. For example, a subproperty schema statement does not automatically infer the domain and range information for the property based on the domain and range specified for the super-property.

Certain owl and rdfs properties in the asserted triples, when left unchecked, may be used to infer data that may be used to circumvent the VPD policies. For example, when the new property is defined as a super-property of a property that has a specific data access constraint, the inferred data may duplicate all instances of the subproperty using the super-property. Unless the VPD policy explicitly defines access constraints for the super-property, the inferred data may be used to circumvent the access restrictions.

The ability to infer new data is only granted to users with FULL access, and such users should ensure that the metadata associated with the VPD policy is complete in light of newly inferred data. Specifically, the metadata associated with the VPD policy should be maintained if some new rdfs:subClassOf, rdfs:superClassOf, rdfs:subPropertyOf, rdfs:superPropertyOf, or owl:equivalentProperty assertions are generated during inference. Also, any new properties introduced by the rulebases used for inference may need domain and range specifications, as well as data access constraints, if they are associated with some sensitive information.

In a VPD policy, a property can be declared to be equivalent to another property so that the domain and range information, as well as any constraints defined for the original property, are automatically duplicated for the equivalent property. However, within a VPD policy, additional metadata or data access constraints cannot be directly assigned to the property declared to be an equivalent of another property.

5.1.3 Data Access Constraints

The data access constraints associated with a VPD policy fall into two general categories, based on the types of access restrictions that they enforce:

  • Those that restrict access to instances of specific RDF classes

  • Those that restrict to assertions using specific RDF properties

The access restrictions are enforced conditionally, based on the application context and the characteristics of the resources being accessed in a SPARQL query. Data access constraints restrict access to instances of an RDF class or property using some properties associated with the resource. For example, access to a resource that is a member of the Contract class may be restricted only to the users who work on the contract, identified using the hasMember property associated with the resource. Similarly, access to the hasContractValue property for a resource may be restricted to a user identified as the manager of the contract using hasManager property associated with the same resource.

Each data access constraint is expressed using two graph patterns identified as a match pattern and an apply pattern. The match pattern of a constraint determines the type of access restriction it enforces and binds one or more variables to the corresponding data instances accessed in the user query. For example, the following match pattern is defined for instances of the Contract class, and it binds a variable to all such instances accessed through a SPARQL query:

{ ?contract  rdf:type  <http://www.myorg.com/classes/Contract> }

Similarly, a match pattern for a constraint involving an RDF property matches the instances of the property accessed in a SPARQL query, and binds two variables to the resources in the subject and object position of such instances. For example, the match pattern for a constraint on the hasContractValue property is defined as follows:

{ ?contract  <http://www.myorg.com/pred/hasContractValue>  ?cvalue }

The apply pattern of a data access constraint defines additional graph patterns to be applied on the resources that match the match pattern before they can be used to construct the query results. One or more variables defined in the match pattern of a data access constraint are used in the corresponding apply pattern to enforce the access restrictions on the identified resources. For example, the following match pattern and apply pattern combination ensures that the hasContractValue of a contract can be accessed only if Andy is the manager of the contract being accessed.:

Match:  { ?contract  pred:hasContractValue  ?cvalue  }
Apply:  { ?contract  pred:hasManager        emp:Andy }

A data access constraint with its match and apply patterns expressed in SPARQL syntax can be added to a VPD policy to enforce access restrictions on the data stored in RDF models that are associated with the VPD policy. The following example, which adds a constraint to the VPD policy, assumes that the VPD policy is defined with appropriate namespace map for the pred and emp namespace prefixes. (To associate a namespace map with a VPD policy, use the SEM_RDFSA.CREATE_VPD_POLICY procedure.)

begin
  sem_rdfsa.add_vpd_constraint(
          policy_name   => 'contracts_policy',
          constr_name   => 'andy_constraint_1',
          match_pattern => '{?contract  pred:hasContractValue ?cvalue }',
          apply_pattern => '{?contract  pred:hasManager       emp:Andy }', 
          constr_group  => 'andy');
end;
/

The ability to arrange data access constraints into groups could ensure that the previous constraint is applied only for the sessions associated with Andy. However, to avoid proliferation of structurally similar constraints for each user, you can define a common constraint that uses the application context in the object position of the apply graph patterns, as shown in the following example:

begin
  sem_rdfsa.add_vpd_constraint(
          policy_name   => 'contracts_policy',
          constr_name   => 'manager_constraint_1',
          match_pattern => '{?contract  pred:hasContractValue ?cvalue }',
          apply_pattern => '{?contract  pred:hasManager     
             "sys_context(''sa$appctx'',''app_user_uri''}"^^orardf:instruction }',
          constr_group  => 'manager');
end;
/

In the preceding example. the data access constraint, defined within the manager constraint group, can be activated for all sessions involving users with a manager role. In this case, the secure application context can be programmed to initialize the attribute app_user_uri of the sa$appctx namespace with the URI for the user logged in. For example, when user Andy logs into the application, the app_user_uri attribute can be initialized to <http://www.myorg.com/employee/Andy>, in which case the constraint will ensure that user Andy can view the value for a contract only if user Andy manages the contract. Generally, the sys_context function can be used in the object position of any graph pattern to allow dynamic URIs or literal values to be bound at the time of query execution. Note that if the context is not initialized properly, the preceding constraint will fail for all data instances and effectively restrict the user from accessing any data.

A SPARQL query issued on an RDF model with a VPD policy is analyzed using the match patterns of all the active data access constraints that are defined in the policy. In the next example, the SPARQL query refers to the hasContractValue property, thereby enforcing the constraint if the group is active. Logically, the enforcement of a constraint is equivalent to rewriting the original SPARQL graph pattern to include the apply patterns for all the relevant constraints, using appropriate variables and terms from the user query. With the previous access restriction on the hasContractValue property, the following SPARQL graph pattern passed to a SEM_MATCH operator is logically rewritten as shown in the following example:

Query:     
{ ?contr  pred:drivenBy         ?dept . 
  ?contr  pred:hasContractValue ?val }
 
Rewritten query:     
{ ?contr  pred:drivenBy         ?dept . 
  ?contr  pred:hasContractValue ?val .
  ?contr  pred:hasManager
                "sys_context('sa$appctx','app_user_uri'}"^^orardf:instruction }

When the match pattern of a data access constraint on an RDF property matches the pattern being accessed in a user query, the equivalent VPD-enforced query appends the corresponding apply patterns to the SPARQL query using the variables and terms appearing in the matched pattern. When a SPARQL query has nested graph patterns, the data access constraints are applied to appropriate basic query graph pattern block. In the following example, the hasContractValue property is referenced in the OPTIONAL graph pattern, and therefore the corresponding apply pattern is enforced just for this block of the graph pattern.

Query:     
{ ?contr  pred:drivenBy         ?dept . 
   OPTIONAL { ?contr  pred:hasContractValue ?val } } 
 
Rewritten query:     
{ ?contr  pred:drivenBy         ?dept . 
   OPTIONAL { ?contr  pred:hasContractValue ?val .
              ?contr  pred:hasManager
                "sys_context('sa$appctx','app_user_uri'}"^^orardf:instruction }

The apply pattern for a data access constraint can be any valid basic graph pattern with multiple triple patterns and a FILTER clause. For example, the access constraint on the hasContractValue property for a user with VP role may stipulate that the user can access the property only if he or she is the Vice President of the department driving the contract. The match and apply patterns for such constraint can be defined as follows:

Match:  { ?contract  pred:hasContractValue  ?cvalue  }
Apply:  { ?contract  pred:drivenBy          ?dept . 
          ?dept      pred:hasVP
               "sys_context('sa$appctx','app_user_uri'}"^^orardf:instruction }

A match pattern defined for a data access constraint associated with an RDF class identifies all variables and terms that are known to be instances of the class. The RDF metadata defined in the VPD policy is used to determine the type for each variable and the term in a SPARQL query, and the appropriate access constraints are applied on these variables and terms. For example, the following VPD constraint ensures that a resource that is a member of the Contract class can only be accessed by a user who has a hasMember relationship with the resource:

Match:  { ?contract  rdf:type  <http://www.myorg.com/classes/Contract> }
Apply:  { ?contract  pred:hasMember           
               "sys_context('sa$appctx','app_user_uri'}"^^orardf:instruction }

The class information for a variable or term appearing in a SPARQL query is derived using the domain and range information for the properties appearing in the query. In the SPARQL query in the next example, if the VPD policy has an RDF Schema statement that asserts that the domain of the drivenBy property is the Contract class, the variable ?contr is known to hold instances of the Contract class. Therefore, with the previously defined access restriction for the Contract class, the user query is rewritten to include an appropriate apply pattern, as shown in the following example:

Query:     
{ ?contr  pred:drivenBy    ?dept . 
  ?contr  pred:hasDueDate  ?due } 
 
Rewritten query: 
{ ?contr  pred:drivenBy    ?dept . 
  ?contr  pred:hasDueDate  ?due  . 
  ?contr  pred:hasMember           
               "sys_context('sa$appctx','app_user_uri'}"^^orardf:instruction }

When a basic graph pattern in a SPARQL query matches multiple data access constraints, the corresponding apply patterns are combined to form a conjunctive graph pattern, which is subsequently enforced for the specific graph pattern by logically rewriting the SPARQL query. While considering the data access constraints to be enforced for a given SPARQL query, the class and property hierarchy associated with the VPD policy is consulted to automatically enforce all applicable constraints.

  • A variable or term identified as an instance of a specific RDF class enforces constraints associated with the class and all its superclasses.

  • A constraint associated with a property is enforced when the user query references the property or any property defined as its subproperty or an equivalent property.

You can use the sys_context function in a data access constraint to enforce context-dependent access restrictions with structurally similar graph patterns. You can dynamically activate and deactivate constraint groups, based on the application context, to enforce alternate access restrictions using structurally different graph patterns.

5.1.4 RDFVPD_POLICIES View

The MDSYS.RDFVPD_POLICIES view contains information about all VPD policies defined in the schema or the policies to which the user has FULL access. If the same policy is associated with multiple models, this view has one entry for each such association. This view exists only after the semantic network and a VPD policy have been created.

The MDSYS.RDFVPD_POLICIES view contains the columns shown in Table 5-1.

Table 5-1 MDSYS.RDFVPD_POLICIES View Columns

Column NameData TypeDescription

POLICY_OWNER

VARCHAR2(32)

Owner of the VPD policy.

POLICY_NAME

VARCHAR2(32)

Name of the VPD policy.

NAMESPACE_MAP

RDF_ALIASES

Mapping for namespace entries that are used in the VPD constraint definitions.

CONTEXT_NAME

VARCHAR2(32)

Name of the context used to manage constraint groups.


5.1.5 RDFVPD_MODELS View

The MDSYS.RDFVPD_MODELS view contains information about RDF models and their associated VPD policies. This view exists only after the semantic network and a VPD policy have been created.

The MDSYS.RDFVPD_MODELS view contains the columns shown in Table 5-2.

Table 5-2 MDSYS.RDFVPD_MODELS View Columns

Column NameData TypeDescription

MODEL_NAME

VARCHAR2(25)

Name of the model.

POLICY_OWNER

VARCHAR2(32)

Owner of the VPD policy.

POLICY_NAME

VARCHAR2(32)

Name of the VPD policy.

OPERATION_TYPE

VARCHAR2(9)

Type of operation for which the VPD policy is enforced: QUERY or DML.


5.1.6 RDFVPD_POLICY_CONSTRAINTS View

The MDSYS.RDFVPD_POLICY_CONSTRAINTS view contains information about the constraints defined in the VPD policy that are accessible to the current user. This view exists only after the semantic network and a VPD policy have been created.

The MDSYS.RDFVPD_POLICY_CONSTRAINTS view contains the columns shown in Table 5-3.

Table 5-3 MDSYS.RDFVPD_POLICY_CONSTRAINTS View Columns

Column NameData TypeDescription

POLICY_OWNER

VARCHAR2(32)

Owner of the VPD policy.

POLICY_NAME

VARCHAR2(32)

Name of the VPD policy.

CONSTRAINT_NAME

VARCHAR2(32)

Name of the constraint.

MATCH_PATTERN

VARCHAR2(1000)

Match pattern for the constraint.

APPLY_PATTERN

VARCHAR2(4000)

Apply pattern for the constraint.

CONSTRAINT_GROUP

VARCHAR2(32)

Name of the constraint group to which the constraint belongs. (Not case-sensitive.).


5.1.7 RDFVPD_PREDICATE_MDATA View

The MDSYS.RDFVPD_PREDICATE_MDATA view contains information about the predicate metadata associated with a VPD policy. This view exists only after the semantic network and a VPD policy have been created.

The MDSYS.RDFVPD_PREDICATE_MDATA view contains the columns shown in Table 5-4.

Table 5-4 MDSYS.RDFVPD_PREDICATE_MDATA View Columns

Column NameData TypeDescription

POLICY_OWNER

VARCHAR2(32)

Owner of the VPD policy.

POLICY_NAME

VARCHAR2(32)

Name of the VPD policy.

PREDICATE

VARCHAR2(4000)

URI for the predicate for which the domain and range information is defined.

HASDOMAIN

VARCHAR2(4000)

URI representing the domain of the predicate.

HASRANGE

VARCHAR2(4000)

URI representing the range of the predicate.


5.1.8 RDFVPD_RESOURCE_REL View

The MDSYS.RDFVPD_RESOURCE_REL view contains information about the subclass, subproperty, and equivalence property relationships that are defined between resources in a VPD policy. This view exists only after the semantic network and a VPD policy have been created.

The MDSYS.RDFVPD_RESOURCE_REL view contains the columns shown in Table 5-5.

Table 5-5 MDSYS.RDFVPD_RESOURCE_REL View Columns

Column NameData TypeDescription

POLICY_OWNER

VARCHAR2(32)

Owner of the VPD policy.

POLICY_NAME

VARCHAR2(32)

Name of the VPD policy.

SUBJECT_RESOURCE

VARCHAR2(4000)

Subject resource.

OBJECT_RESOURCE

VARCHAR2(4000)

Object resource.

RELATIONSHIP_TYPE

VARCHAR2(4000)

Relationship that exists between the subject resource and the object resource.


5.2 Oracle Label Security (OLS) for RDF Data

Oracle Label Security (OLS) for RDF data provides two options for securing semantic data:

To specify an option, use the SEM_RDFSA.APPLY_OLS_POLICY procedure with the appropriate rdfsa_options parameter value.

To switch from one option to the other, remove the existing policy by using the SEM_RDFSA.REMOVE_OLS_POLICY procedure, and then apply the new policy by using the SEM_RDFSA.APPLY_OLS_POLICY procedure with the appropriate rdfsa_options parameter value.

5.2.1 Triple-Level Security

The triple-level security option provides a thin layer of RDF-specific capabilities on top of the Oracle Database native support for label security. This option provides better performance and is easier to use than the resource-level security (described in Section 5.2.2), especially for performing inference while using OLS. The main difference is that with triple-level security there is no need to assign labels, explicitly or implicitly, to individual triple resources (subjects, properties, objects).


Note:

To use triple-level security, you must first install Patch 9819833 SEMANTIC TECHNOLOGIES 11G R2 FIX BUNDLE 2 (available from My Oracle Support).

To use triple-level security, specify SEM_RDFSA.TRIPLE_LEVEL_ONLY as the rdfsa_options parameter value when you execute the SEM_RDFSA.APPLY_OLS_POLICY procedure. For example:

EXECUTE sem_rdfsa.apply_ols_policy('defense', SEM_RDFSA.TRIPLE_LEVEL_ONLY);

Do not specify any of the other available parameters for the SEM_RDFSA.APPLY_OLS_POLICY procedure.

When you use triple-level security, OLS is applied to each semantic model in the network. That is, label security is applied to the relevant internal tables and to all the application tables; there is no need to manually apply policies to the application tables of existing semantic models. However, if you need to create additional models after applying the OLS policy, you must use the SEM_OLS.APPLY_POLICY_TO_APP_TAB procedure to apply OLS to the application table before creating the model. Similarly, if you have dropped a semantic model and you no longer need to protect the application table, you can use the SEM_OLS.REMOVE_POLICY_FROM_APP_TAB procedure. (These procedures are described in Chapter 10.)

With triple-level security, duplicate triples with different labels can be inserted in the semantic model. (Such duplicates are not allowed with resource-level security.) For example, assume that you have a triple with a very sensitive label, such as:

(<urn:X>,<urn:P>,<urn:Y>, "TOPSECRET")

This does not prevent a low-privileged (UNCLASSIFIED) user from inserting the triple (<urn:X>,<urn:P>,<urn:Y>, "UNCLASSIFIED"). Because SPARQL and SEM_MATCH do not return label information, a query will return both rows (assuming the user has appropriate privileges), and it will not be easy to distinguish between the TOPSECRET and UNCLASSIFIED triples.

To filter out such low-security triples when querying the semantic models, you can one or more the following options with SEM_MATCH:

  • POLICY_NAME specifies the OLS policy name.

  • MIN_LABEL specifies the minimum label for triples that are included in the query

In other words, every triple that contains a label that is strictly dominated by MIN_LABEL is not included in the query. For example, to filter out the "UNCLASSIFIED" triple, you could use the following query (assuming the OLS policy name is DEFENSE and that the query user has read privileges over UNCLASSIFIED and TOPSECRET triples):

SELECT s,p,y FROM table(sem_match('{?s ?p ?y}' , 
  sem_models(TEST'), null, null, null, null, 
  'MIN_LABEL=TOPSECRET POLICY_NAME=DEFENSE'));

Note that the filtering in the preceding example occurs in addition to the security checks performed by the native OLS software.

After a triple has been inserted, you can view and update the label information can be done through the CTXT1 column in the application table for the semantic model (assuming that you have the WRITEUP and WRITEDOWN privileges to modify the labels).

There are no restrictions on who can perform inference or bulk loading with triple-level security; all of the inferred or bulk loaded triples are inserted with the user's session row label. Note that you can change the session labels by using the SA_UTL package. (For more information, see Oracle Label Security Administrator's Guide.)

5.2.2 Resource-Level Security

The resource-level security option enables you to assign one or more security labels that define a security level for table rows. Conceptually, a table in a relational data model can be mapped to an equivalent RDF graph. Specifically, a row in a relational table can be mapped to a set of triples, each asserting some facts about a specific Subject. In this scenario, the subject represents the primary key for the row and each non-key column-value combination from the row is mapped to a predicate-object value combination for the corresponding triples.

A row in a relational data model is identified by its key, and OLS, as a row-level access control mechanism, effectively restricts access to the values associated with the key. With this conceptual mapping between relational and RDF data models, restricting access to a row in a relational table is equivalent to restricting access to a subgraph involving a specific subject. In a model that supports sensitivity labels for each triple, this is enforced by applying the same label to all the triples involving the given subject. However, you can also achieve greater flexibility by allowing the individual triples to have different labels, while maintaining a minimum bound for all the labels.

OLS support for RDF data employs a multilevel approach in which sensitivity labels associated with the triple components (subject, predicate, object) collectively form a minimum bound for the sensitivity label for the triple. With this approach, a data sensitivity label associated with an RDF resource (used as subject, predicate, or object) restricts unauthorized users from accessing any triples involving the resource and from creating new triples with the resource. For example, projectHLS as a subject may have a minimum sensitivity label, which ensures that all triples describing this subject have a sensitivity label that at least covers the label for projectHLS. Additionally, hasContractValue as a predicate may have a higher sensitivity label; and when this predicate is used with projectHLS to form a triple, that triple minimally has a label that covers both the subject and the predicate labels, as in the following example:

Triple 1: <http://www.myorg.com/contract/projectHLS> :ownedBy
                               <http://www.myorg.com/department/Dept1>
Triple 2: <http://www.myorg.com/contract/projectHLS> :hasContractValue
                               "100000"^^xsd:integer

Sensitivity labels are associated with the RDF resources (URIs) based on the position in which they appear in a triple. For example, the same RDF resource may appear in different positions (subject, predicate, or object) in different triples. Three unique labels can be assigned to each resource, so that the appropriate label is used to determine the label for a triple based on the position of the resource in the triple. You can choose the specific resource positions to be secured in a database instance when you apply an OLS policy to the RDF repository. You can secure subjects, objects, predicates, or any combination, as explained in separate sections to follow. The following example applies an OLS policy named defense to the RDF repository and allows sensitivity labels to be associated with RDF subjects and predicates.

begin
  sem_rdfsa.apply_ols_policy(
        policy_name   => 'defense',
        rdfsa_options => sem_rdfsa.SECURE_SUBJECT+
                         sem_rdfsa.SECURE_PREDICATE); 
end;
/

The same RDF resource can appear in both the subject and object positions (and sometime even as the predicate), and such a resource can have distinct sensitivity labels based on its position. A triple using the resource at a specific position should have a label that covers the label corresponding to the resource's position. In such cases, the triple can be asserted or accessed only by the users with labels that cover the triple and the resource labels.

In a specific RDF repository, security based on data classification techniques can be turned on for subjects, predicates, objects, or a combination of these. This ensures that all the triples added to the repository automatically conform to the label relationships described above.

5.2.2.1 Securing RDF Subjects

An RDF resource (typically a URI) appears in the subject position of a triple when an assertion is made about the resource. In this case, a sensitivity label associated with the resource has following characteristics:

  • The label represents the minimum sensitivity label for any triple using the resource as a subject. In other words, the sensitivity label for the triple should dominate or cover the label for the subject.

  • The label for a newly added triple is initialized to the user initial row label or is generated using the label function, if one is specified. Such operations are successful only if the triple's label dominates the label associated with the triple's subject.

  • Only a user with an access label that dominates the subject's label and the triple's label can read the triple.

By default, the sensitivity label for a subject is derived from the user environment when an RDF resource is used in the subject position of a triple for the first time. The default sensitivity label in this case is set to the user's initial row label (the default that is assigned to all rows inserted by the user).

However, you can categorize an RDF resource as a subject and assign a sensitivity label to it even before it is used in a triple. The following example assigns a sensitivity label named SECRET:HLS:US to the projectHLS resource, thereby restricting the users who are able to define new triples about this resource and who are able to access existing triples with this resource as the subject:

begin
  sem_rdfsa.set_resource_label(
         model_name   => 'contracts',
         resource_uri => '<http://www.myorg.com/contract/projectHLS>',
         label_string => 'SECRET:HLS:US',
         resource_pos => 'S');
end;

5.2.2.2 Securing RDF Predicates

An RDF predicate defines the relationship between a subject and an object. You can use sensitivity labels associated with RDF predicates to restrict access to specific types of relationships with all subjects.

RDF predicates are analogous to columns in a relational table, and the ability to restrict access to specific predicates is equivalent to securing relational data at the column level. As in the case of securing the subject, the predicate's sensitivity label creates a minimum bound for any triples using this predicate. It is also the minimum authorization that a user must posses to define a triple with the predicate or to access a triple with the predicate.

The following example assigns the label HSECRET:FIN (in this scenario, a label that is Highly Secret and that also belongs to the Finance department) to triples with the hasContractValue predicate, to ensure that only a user with such clearance can define the triple or access it:

begin
  sem_rdfsa.set_predicate_label( 
         model_name   => 'contracts',
         predicate    => '<http://www.myorg.com/pred/hasContractValue>',
         label_string => 'HSECRET:FIN');
end;  
/

You can secure predicates in combination with subjects. In such cases, the triples using a subject and a predicate are ensured to have a sensitivity label that at least covers the labels for both the subject and the predicate. Extending the preceding example, if projectHLS as a subject is secured with label SECRET:HLS:US and if hasContractValue as a predicate is secured with label HSECRET:MFIN:, a triple assigning a monetary value for projectHLS should at least have a label HSECRET:HLS,FIN:US. Effectively, a user's label must dominate this triple's label to be able to define or access the triple.

5.2.2.3 Securing RDF Objects

An RDF triple can have an URI or a literal in its object position. The URI in object position of a triple represents some resource. You can secure a resource in the object position by associating a sensitivity label to it, to restrict the ability to use the resource as an object in triples.

Typically, a resource (URI or non-literal) appearing in the object position of a triple may itself be described using additional RDF statements. Effectively, an RDF resource in the object position could appear in the subject position in some other triples. When the RDF resources are secured at the object position without explicit sensitivity labels, the label associated with the same resource in the subject position is used as the default label for the object.

5.2.2.4 Generating Labels for Inferred Triples

RDF data model allows for specification of declarative rules, enabling it to infer the presence of RDF statements that are not explicitly added to the repository. The following shows some simple declarative rules associated with the logic that projects can be owned by departments and departments have Vice Presidents, and in such cases the project leader is by default the Vice President of the department that owns the project.

RuleID -> projectLedBy
Antecedent Expression -> (?proj :ownedBy ?dept) (?dept :hasVP ?person)
Consequent Expression -> (?proj :isLedBy ?person)

An RDF rule uses some explicitly asserted triples as well as previously inferred triples as antecedents, and infers one or more consequent triples. Traditionally, the inference process is executed as an offline operation to pregenerate all the inferred triples and to make them available for subsequent query operations.

When the underlying RDF graph is secured using OLS, any additional data inferred from the graph should also be secured to avoid exposing the data to unauthorized users. Additionally, the inference process should run with higher privileges, specifically with full access to data, in order to ensure completeness.

OLS support for RDF data offers techniques to generate sensitivity labels for inferred triples based on labels associated with one or more RDF artifacts. It provides label generation techniques that you can invoke at the time of inference. Additionally, it provides an extensibility framework, which allows an extensible implementation to receive a set of possible labels for a specific triple and determine the most appropriate sensitivity label for the triple based on some application-specific logic. The techniques that you can use for generating the labels for inferred triples include the following (each technique, except for Use Antecedent Labels, is associated with a SEM_RDFSA package constant):

  • Use Rule Label (SEM_RDFSA.LABELGEN_RULE): An inferred triple is directly generated by a specific rule, and it may be indirectly dependent on other rules through its antecedents. Each rule may have a sensitivity label, which is used as the sensitivity label for all the triples directly inferred by the rule.

  • Use Subject Label (SEM_RDFSA.LABELGEN_SUBJECT): Derives the label for the inferred triple by considering any sensitivity labels associated with the subject in the new triple. Each inferred triple has a subject, which could in turn be a subject, predicate, or object in any of the triple's antecedents. When such RDF resources are secured, the subject in the newly inferred triple may have one or more labels associated with it. With the Use Subject Label technique, the label for the inferred triple is set to the unique label associated with the RDF resource. When more than one label exists for the resource, you can implement the extensible logic to determine the most relevant label for the new triple.

  • Use Predicate Label (SEM_RDFSA.LABELGEN_PREDICATE): Derives the label for the inferred triple by considering any sensitivity labels associated with the predicate in the new triple. Each inferred triple has a predicate, which could in turn be a subject, predicate, or object in any of the triple's antecedents. When such RDF resources are secured, the predicate in the newly inferred triple may have one or more labels associated with it. With the Use Predicate Label technique, the label for the inferred triple is set to the unique label associated with the RDF resource. When more than one label exists for the resource, you can implement the extensible logic to determine the most relevant label for the new triple.

  • Use Object Label (SEM_RDFSA.LABELGEN_OBJECT): Derives the label for the inferred triple by considering any sensitivity labels associated with the object in the new triple. Each inferred triple has an object, which could in turn be a subject, predicate, or object in any of the triple's antecedents. When such RDF resources are secured, the object in the newly inferred triple may have one or more labels associated with it. With the Use Object Label technique, the label for the inferred triple is set to the unique label associated with the RDF resource. When more than one label exists for the resource, you can implement the extensible logic to determine the most relevant label for the new triple.

  • Use Dominating Label (SEM_RDFSA.LABELGEN_DOMINATING): Each inferred triple minimally has four direct components: subject, predicate, object, and the rule that produced the triple. With the Use Dominating Label technique, at the time of inference the label generator computes the most dominating of the sensitivity labels associated with each of the component and assigns it as the sensitivity label for the inferred triple. Exception labels are assigned when a clear dominating relationship cannot be established between various labels.

  • Use Antecedent Labels: In addition to the four direct components for each inferred triple (subject, predicate, object, and the rule that produced the triple), a triple may have one or more antecedent triples, which are instrumental in deducing the new triple. With the Use Antecedent Labels technique, the labels for all the antecedent triples are considered, and conflict resolution criteria are implemented to determine the most appropriate label for the new triple. Since an inferred triple may be dependent on other inferred triples, a strict order is followed while generating the labels for all the inferred triples.

    The Use Antecedent Labels technique requires that you use a custom label generator. For information about creating and using a custom label generator, see Section 5.2.2.5.

The following example creates an entailment (rules index) for the contracts data using a specific rule base. This operation can only be performed by a user with FULL access privilege with the OLS policy applied to the RDF repository. In this case, the labels generated for the inferred triples are based on the labels associated with their predicates, as indicated by the use of the SEM_RDFSA.LABELGEN_PREDICATE package constant in the label_gen parameter.

begin
  sem_rdfsa.create_entailment(
         index_name_in   => 'contracts_inf',
         models_in       => SDO_RDF_Models('contracts'),
         rulebases_in    => SDO_RDF_Rulebases('contracts_rb'),
         options         => 'USER_RULES=T',
         label_gen       => sem_rdfsa.LABELGEN_PREDICATE);
end;

When the predefined or extensible label generation implementation cannot compute a unique label to be applied to an inferred triple, an exception label is set for the triple. Such triples are not accessible by any user other than the user with full access to RDF data (also the user initiating the inference process). The triples with exception labels are clearly marked, so that a privileged user can access them and apply meaningful labels manually. After the sensitivity labels are applied to inferred triples, only users with compatible labels can access these triples. The following example updates the sensitivity label for triples for which an exception label was set:

update mdsys.rdfi_contracts_inf 
     set ctxt1 = char_to_label('defense', 'SECRET:HLS:US')
     where ctxt1 = -1;

Inferred triples accessed through generated labels might not be same as conceptual triples inferred directly from the user accessible triples and rules. The labels generated using system-defined or custom implementations cannot be guaranteed to be precise. See the information about Fine-Grained Access Control (VPD and OLS) Considerations in the Usage Notes for the SEM_APIS.CREATE_ENTAILMENT procedure in Chapter 9 for details.

5.2.2.5 Using Labels Based on Application Logic

The MDSYS.RDFSA_LABELGEN type is used to apply appropriate label generator logic at the time of index creation; however, you can also extend this type to implement a custom label generator and generate labels based on application logic. The label generator is specified using the label_gen parameter with the SEM_APIS.CREATE_ENTAILMENT procedure. To use a system-defined label generator, specify a SEM_RDFSA package constant, as explained in Section 5.2.2.4; to use a custom label generator, you must implement a custom label generator type and specify an instance of that type instead of a SEM_RDFSA package constant.

To create a custom label generator type, you must have the UNDER privilege on the RDFSA_LABELGEN type. In addition, to create an index for RDF data , you must should have the EXECUTE privilege on this type. The following example grants these privileges to a user named RDF_ADMIN:

GRANT under, execute ON mdsys.rdfsa_labelgen TO rdf_admin;

The custom label generator type must implement a constructor, which should set the dependent resources and specify the getNumericLabel method to return the label computed from the information passed in, as shown in the following example:

CREATE OR REPLACE TYPE CustomSPORALabel UNDER mdsys.rdfsa_labelgen  (
   constructor function CustomSPORALabel return self as result,
   overriding member function getNumericLabel (
                                    subject   rdfsa_resource,
                                    predicate rdfsa_resource,
                                    object    rdfsa_resource,
                                    rule      rdfsa_resource,
                                    anteced   rdfsa_resource)
        return number);

The label generator constructor uses a set of constants defined in the SEM_RDFSA package to indicate the list of resources on which the label generator relies. The dependent resources are identified as an inferred triple's subject, its predicate, its object, the rule that produced the triple, and its antecedents. A custom label generator can rely on any subset of these resources for generating the labels, and you can specify this in its constructor by using the constants defined in SEM_RDFSA package : USE_SUBJECT_LABEL, USE_PREDICATE_LABEL, USE_OBJECT_LABEL, USE_RULE_LABEL, USE_ANTCED_LABEL. The following example creates the type body and specifies the constructor:

Example 5-1 creates the type body, specifying the constructor function and the getNumericLabel member function. (Application-specific logic is not included in this example.)

Example 5-1 Creating a Custom Label Generator Type

CREATE OR REPLACE TYPE BODY CustomSPORALabel AS
 
   constructor function CustomSPORALabel return self as result as
   begin
     self.setDepResources(sem_rdfsa.USE_SUBJECT_LABEL+
                          sem_rdfsa.USE_PREDICATE_LABEL+
                          sem_rdfsa.USE_OBJECT_LABEL+
                          sem_rdfsa.USE_RULE_LABEL+
                          sem_rdfsa.USE_ANTECED_LABELS);
     return;
   end CustomSPORALabel;
   
   overriding member function getNumericLabel (
                                    subject   rdfsa_resource,
                                    predicate rdfsa_resource,
                                    object    rdfsa_resource,
                                    rule      rdfsa_resource,
                                    anteced   rdfsa_resource)
        return number as
     labellst mdsys.int_array := mdsys.int_array(); 
   begin
    -- Find dominating label of S P O R A –
    –- Application specific logic for computing the triple label –
    -- Copy over all labels to labellst --
    for li in 1 .. subject.getLabelCount() loop
      labellst.extend; 
      labellst(labellst.COUNT) = subject.getLabel(li); 
    end loop; 
    --- Copy over other labels as well --- 
    --- Find a dominating of all the labels. Generates –1 if no
    --- dominating label within the set
    return self.findDominatingOf(labellst); 
   end getNumericLabel;
  end CustomSPORALabel;  
  /

In Example 5-1, the sample label generator implementation uses all the resources contributing to the inferred triple for generating a sensitivity label for the triple. Thus, the constructor uses the setDepResources method defined in the superclass to set all its dependent components. The list of dependent resources set with this step determines the exact list of values passed to the label generating routine.

The getNumericLabel method is the label generation routine that has one argument for each resource that an inferred triple may depend on. Some arguments may be null values if the corresponding dependent resource is not set in the constructor implementation.

The label generator implementation can make use of a general-purpose static routine defined in the RDFSA_LABELGEN type to find a domination label for a given set of labels. A set of labels is passed in an instance of MDSYS.INT_ARRAY type, and the method finds a dominating label among them. If no such label exists, an exception label –1 is returned.

After you have implemented the custom label generator type, you can use the custom label generator for inferred data by assigning an instance of this type to the label_gen parameter in the SEM_APIS.CREATE_ENTAILMENT procedure, as shown in the following example:

begin
  sem_apis.create_entailment(
         index_name_in  => 'contracts_rdfsinf',
         models_in      => SDO_RDF_Models('contracts'),
         rulebases_in   => SDO_RDF_Rulebases('RDFS'),
         options        => '',
         label_gen      => CustomSPORALabel());
end;
/

5.2.2.6 RDFOLS_SECURE_RESOURCE View

The MDSYS.RDFOLS_SECURE_RESOURCE view contains information about resources secured with Oracle Label Security (OLS) policies and the sensitivity labels associated with these resources.

Select privileges on this view can be granted to appropriate users. To view the resources associated with a specific model, you must also have select privileges on the model (or the corresponding RDFM_model-name view).

The MDSYS.RDFOLS_SECURE_RESOURCE view contains the columns shown in Table 5-6.

Table 5-6 MDSYS.RDFOLS_SECURE_RESOURCE View Columns

Column NameData TypeDescription

MODEL_NAME

VARCHAR2(25)

Name of the model.

MODEL_ID

NUMBER

Internal identifier for the model.

RESOURCE_ID

NUMBER

Internal identifier for the resource; to be joined with MDSYS.RDF_VALUE$.VALUE_ID column for information about the resource.

RESOURCE_TYPE

VARCHAR2(16)

One of the following string values to indicate the resource type for which the label is assigned: SUBJECT, PREDICATE, OBJECT, GLOBAL.

CTXT1

NUMBER

Sensitivity label assigned to the resource.


PKwGMMPK.AOEBPS/sem_perf_ref.htm& SEM_PERF Package Subprograms

11 SEM_PERF Package Subprograms

The SEM_PERF package contains subprograms for examining and enhancing the performance of the Resource Description Framework (RDF) and Web Ontology Language (OWL) support in an Oracle database. To use the subprograms in this chapter, you must understand the conceptual and usage information in Chapter 1, "Oracle Database Semantic Technologies Overview" and Chapter 2, "OWL Concepts".

This chapter provides reference information about the subprograms, listed in alphabetical order.


SEM_PERF.GATHER_STATS

Format

SEM_PERF.GATHER_STATS();

     just_on_values_table IN BOOLEAN DEFAULT FALSE,

     degree IN NUMBER(38) DEFAULT NULL);

Description

Gathers statistics about RDF and OWL tables and their indexes.

Parameters

just_on_values_table

TRUE collects statistics only on the table containing the lexical values of triples; FALSE (the default) collects statistics on all major tables related to the storage of RDF and OWL data.

A value of TRUE reduces the execution time for the procedure; and it may be sufficient if you need only to collect statistics on the values table (for example, if you use other interfaces to collect any other statistics that you might need).

degree

Degree of parallelism. For more information about parallel execution, see Oracle Database VLDB and Partitioning Guide.

Usage Notes

To use this procedure, you must connect as a user with permission to execute it. By default, when Spatial is installed as part of Oracle Database, only the MDSYS user can execute this procedure; however execution permission on this procedure can be granted to users as needed.

This procedure collects statistical information that can help you to improve inferencing performance, as explained in Section 2.2.7. This procedure internally calls the DBMS_STATS.GATHER_TABLE_STATS procedure to collect statistics on RDF- and OWL-related tables and their indexes, and stores the statistics in the Oracle Database data dictionary. For information about using the DBMS_STATS package, see Oracle Database PL/SQL Packages and Types Reference.

Gathering statistics uses significant system resources, so execute this procedure when it cannot adversely affect essential applications and operations.

Examples

The following example gathers statistics about RDF and OWL related tables and their indexes.

EXECUTE SEM_PERF.GATHER_STATS;
PK{PK.AOEBPS/skos.htm?- Simple Knowledge Organization System (SKOS) Support

3 Simple Knowledge Organization System (SKOS) Support

You can perform inferencing based on a core subset of the Simple Knowledge Organization System (SKOS) data model, which is especially useful for representing thesauri, classification schemes, taxonomies, and other types of controlled vocabulary. SKOS is based on standard semantic web technologies including RDF and OWL, which makes it easy to define the formal semantics for those knowledge organization systems and to share the semantics across applications.

Support is provided for most, but not all, of the features of SKOS, the detailed specification of which is available at http://www.w3.org/TR/skos-reference/.

Around 40 SKOS-specific terms are included in the Oracle Database semantic technologies support, such as skos:broader, skos:relatedMatch, and skos:Concept. Over 100 SKOS axiomatic triples have been added, providing the basic coverage of SKOS semantics. However, support is not included for the integrity conditions described in the SKOS specification.

To perform SKOS-based inferencing, specify the system-defined SKOSCORE rulebase in the rulebases_in parameter in the call to the SEM_APIS.CREATE_ENTAILMENT procedure, as in the following example:

EXECUTE sem_apis.create_entailment('tstidx',sem_models('tst'), sem_rulebases('skoscore'));

Example 3-1 defines, in Turtle format, a simple electronics scheme and two relevant concepts, cameras and digital cameras. Its meaning is straightforward and its representation is in RDF. It can be managed by Oracle Database in the same way as other RDF and OWL data.

Example 3-1 SKOS Definition of an Electronics Scheme

ex1:electronicsScheme rdf:type skos:ConceptScheme;
 
ex1:cameras rdf:type skos:Concept;
   skos:prefLabel "cameras"@en;
   skos:inScheme ex1:electronicsScheme.
 
ex1:digitalCameras rdf:type skos:Concept;
   skos:prefLabel "digital cameras"@en;
   skos:inScheme ex1:electronicsScheme.
 
ex1:digitalCameras skos:broader ex1:cameras.

3.1 Supported and Unsupported SKOS Semantics

This section describes features of SKOS semantics that are and are not supported by Oracle Database.

3.1.1 Supported SKOS Semantics

All terms defined in SKOS and SKOS extension for labels are recognized. When the SKOSCORE rulebase is chosen for inference, the recognized terms include the following:

skos:altLabel
skos:broader
skos:broaderTransitive
skos:broadMatch
skos:changeNote
skos:closeMatch
skos:Collection
skos:Concept
skos:ConceptScheme
skos:definition
skos:editorialNote
skos:exactMatch
skos:example
skos:hasTopConcept
skos:hiddenLabel
skos:historyNote
skos:inScheme
skos:mappingRelation
skos:member
skos:memberList
skos:narrower
skos:narrowerTransitive
skos:narrowMatch
skos:notation
skos:note
skos:OrderedCollection
skos:prefLabel
skos:related
skos:relatedMatch
skos:scopeNote
skos:semanticRelation
skos:topConceptOf
skosxl:altLabel
skosxl:hiddenLabel
skosxl:Label
skosxl:labelRelation
skosxl:literalForm
skosxl:prefLabel

Most SKOS axioms and definitions are supported including the following: S1-S8, S10-S11, S15-S26, S28-S31, S33-S36, S38-S45, S47-S50, and S53-S54. (See the SKOS detailed specification for definitions.)

Most SKOS integrity conditions are supported, including S9, S13, S27, S37, and S46.

S52 is partially supported.

S55, S56, and S57 are not supported by default.

  • S55, the property chain (skosxl:prefLabel, skosxl:literalForm), is a subproperty of skos:prefLabel.

  • S56, the property chain (skosxl:altLabel, skosxl:literalForm), is a subproperty of skos:altLabel.

  • S57, the property chain (skosxl:hiddenLabel, skosxl:literalForm), is a subproperty of skos:hiddenLabel.chains.

However, S55, S56, and S57 can be implemented using the OWL 2 subproperty chain construct. For information about property chain handling, see Section 3.2.2.

3.1.2 Unsupported SKOS Semantics

The following features of SKOS semantics are not supported:

  • S12 and S51: The rdfs:range of the relevant predicates is the class of RDF plain literals. There is no check that the object values of these predicates are indeed plain literals; however, applications can perform such a check.

  • S14: A resource has no more than one value of skos:prefLabel per language tag. This integrity condition is even beyond OWL FULL semantics, and it is not enforced in the current release.

  • S32: The rdfs:range of skos:member is the union of classes skos:Concept and skos:Collection. This integrity condition is not enforced.

  • S55, S56, and S57 are not supported by default, but they can be implemented using the OWL 2 subproperty chain construct, as explained in Section 3.1.1.

3.2 Performing Inference on SKOS Models

To create an SKOS model, use the same procedure (SEM_APIS.CREATE_SEM_MODEL) as for creating a semantic model. You can load data into an SKOS model in the same way as for semantic models.

To infer new relationships for one or more SKOS models, use the SEM_APIS.CREATE_ENTAILMENT procedure with the system-defined rulebase SKOSCORE. For example:

EXECUTE sem_apis.create_entailment('tstidx',sem_models('tst'), sem_rulebases('skoscore'));

The inferred data will include many of the axioms defined in the SKOS detailed specification. Like other system-defined rulebases, SKOSCORE has no explicit rules; all the semantics supported are coded into the implementation.

3.2.1 Validating SKOS Models and Entailments

You can use the SEM_APIS.VALIDATE_ENTAILMENT and SEM_APIS.VALIDATE_MODEL procedures to validate the supported integrity conditions. The output will include any inconsistencies caused by the supported integrity conditions, such as OWL 2 propertyDisjointWith and S52.

Example 3-2 validates an SKOS entailment.

Example 3-2 Validating an SKOS Entailment

set serveroutput on
declare
  lva mdsys.rdf_longVarcharArray;
  idx int;
begin
  lva := sem_apis.validate_entailment(sdo_rdf_models('tstskos'), sem_rulebases('skoscore'));
  if (lva is null) then
    dbms_output.put_line('No conflicts');
  else
  for idx in 1..lva.count loop
    dbms_output.put_line('entry ' || idx || ' ' || lva(idx));
  end loop;
  end if;
end;
 /

3.2.2 Property Chain Handling

The SKOS S55, S56, and S57 semantics are not supported by default. However, you can add support for them by using the OWL 2 subproperty chain construct.

Example 3-3 inserts the necessary chain definition triples for S55 into an SKOS model. After the insertion, an invocation of SEM_APIS.CREATE_ENTAILMENT that specifies the SKOSCORE rulebase will include the semantics defined in S55.

Example 3-3 Property Chain Insertions to Implement S55

INSERT INTO tst VALUES(sdo_rdf_triple_s('tst','<http://www.w3.org/2004/02/skos/core#prefLabel>', '<http://www.w3.org/2002/07/owl#propertyChainAxiom>', '_:jA1'));
INSERT INTO tst VALUES(sdo_rdf_triple_s('tst','_:jA1', '<http://www.w3.org/1999/02/22-rdf-syntax-ns#first>', '<http://www.w3.org/2008/05/skos-xl#prefLabel>'));
INSERT INTO tst VALUES(sdo_rdf_triple_s('tst','_:jA1', '<http://www.w3.org/1999/02/22-rdf-syntax-ns#rest>', '_:jA2'));
INSERT INTO tst VALUES(sdo_rdf_triple_s('tst','_:jA2', '<http://www.w3.org/1999/02/22-rdf-syntax-ns#first>', '<http://www.w3.org/2008/05/skos-xl#literalForm>'));
INSERT INTO tst VALUES(sdo_rdf_triple_s('tst','_:jA2', '<http://www.w3.org/1999/02/22-rdf-syntax-ns#rest>', '<http://www.w3.org/1999/02/22-rdf-syntax-ns#nil>'));
PK }D-?-PK.AOEBPS/preface.htmL Preface

Preface

Oracle Database Semantic Technologies Developer's Guide provides usage and reference information about Oracle Database Enterprise Edition support for semantic technologies, including storage, inference, and query capabilities for data and ontologies based on Resource Description Framework (RDF), RDF Schema (RDFS), and Web Ontology Language (OWL). The Semantic Technologies feature is licensed with the Oracle Spatial option to Oracle Database Enterprise Edition, and it requires the Oracle Partitioning option to Oracle Database Enterprise Edition.


Note:

You must perform certain actions and meet prerequisites before you can use any types, synonyms, or PL/SQL packages related to Oracle semantic technologies support. These actions and prerequisites are explained in Section A.1.

Audience

This guide is intended for those who need to use semantic technology to store, manage, and query semantic data in the database.

You should be familiar with at least the main concepts and techniques for the Resource Description Framework (RDF) and the Web Ontology Language (OWL).

Documentation Accessibility

For information about Oracle's commitment to accessibility, visit the Oracle Accessibility Program website at http://www.oracle.com/pls/topic/lookup?ctx=acc&id=docacc.

Access to Oracle Support

Oracle customers have access to electronic support through My Oracle Support. For information, visit http://www.oracle.com/pls/topic/lookup?ctx=acc&id=info or visit http://www.oracle.com/pls/topic/lookup?ctx=acc&id=trs if you are hearing impaired.

Related Documents

For an excellent explanation of RDF concepts, see the World Wide Web Consortium (W3C) RDF Primer at http://www.w3.org/TR/rdf-primer/.

For information about OWL, see the OWL Web Ontology Language Reference at http://www.w3.org/TR/owl-ref/.

Conventions

The following text conventions are used in this document:

ConventionMeaning
boldfaceBoldface type indicates graphical user interface elements associated with an action, or terms defined in text or the glossary.
italicItalic type indicates book titles, emphasis, or placeholder variables for which you supply particular values.
monospaceMonospace type indicates commands within a paragraph, URLs, code in examples, text that appears on the screen, or text that you enter.

PKŠQLPK.AOEBPS/index.htm Index

Index

A  B  C  D  E  F  G  H  I  J  L  M  N  O  P  Q  R  S  T  U  V  W  X 

A

ADD_DATATYPE_INDEX procedure, 9
ADD_DEPENDENT_POLICY procedure, 12
ADD_SEM_INDEX procedure, 9
ADD_VPD_CONSTRAINT procedure, 13
Advanced Compression, 7.2.2.2
aliases
SEM_ALIASES and SEM_ALIAS data types, 1.6, 4.4
ALL_BGP_HASH
query option for SEM_MATCH, 1.6
ALL_BGP_NL
query option for SEM_MATCH, 1.6
ALL_LINK_HASH
query option for SEM_MATCH, 1.6
ALL_LINK_NL
query option for SEM_MATCH, 1.6
ALL_ORDERED
query option for SEM_MATCH, 1.6
ALLOW_DUP=T
query option for SEM_MATCH, 1.6
ALTER_DATATYPE_INDEX procedure, 9
ALTER_ENTAILMENT procedure, 9
ALTER_MODEL procedure, 9
ALTER_SEM_INDEX_ON_ENTAILMENT procedure, 9
ALTER_SEM_INDEX_ON_MODEL procedure, 9
ANALYZE_ENTAILMENT procedure, 9
ANALYZE_MODEL procedure, 9
apply pattern, 5.1.3
APPLY_OLS_POLICY procedure, 13
APPLY_POLICY_TO_APP_TAB procedure, 10
APPLY_VPD_POLICY procedure, 13

B

basic graph pattern (BGP), 1.6
batch (bulk) loading, 9, 9
batch loading semantic data, 1.7.2
best effort
specifying for SPARQL query, 7.16.3.3
BGP (basic graph pattern), 1.6
blank nodes, 1.3.4
bulk loading, 9, 9
bulk loading semantic data, 1.7.1
BULK_LOAD_FROM_STAGING_TABLE procedure, 9

C

Calais
configuring the Calais extractor type, 4.9
canonical forms, 1.3.2.1
catsem10i.sql script, A.1.3, A.2.1
catsem.sql script, A.1.1
change tracking
disabling, 9
enabling, 9
getting information, 9
CLEANUP_FAILED procedure, 9
client identifiers, 7.2.2.1
cliques (sameAs), 2.2.8
COMPOSE_RDF_TERM function, 9
connection pooling
support in Jena Adapter, 7.10.2
support in Sesame Adapter, 8.7.2
constructors for semantic data, 1.5
convert_old_rdf_data procedure, A.1.2
corpus-centric inference, 4.15
CREATE_ENTAILMENT procedure, 9
CREATE_POLICY procedure, 12
CREATE_RULEBASE procedure, 9
CREATE_SEM_MODEL procedure, 9
CREATE_SEM_NETWORK procedure, 9
CREATE_SOURCE_EXTERNAL_TABLE procedure, 9
CREATE_VIRTUAL_MODEL procedure, 9
CREATE_VPD_POLICY procedure, 13

D

data type indexes
adding, 9
altering, 9
dropping, 9
SEM_DTYPE_INDEX_INFO view, 1.9
using, 1.9
data types
for literals, 1.3.2.1
data types for semantic data, 1.5
default.xslt file
customizing, 7.16.4
DELETE_VPD_CONSTRAINT procedure, 13
demo files
semantic data, 1.11
DISABLE_CHANGE_TRACKING procedure, 9
DISABLE_INC_INFERENCE procedure, 9
DISABLE_OLS_POLICY procedure, 13
discussion forum
semantic technologies, 1.13
document-centric inference, 4.15
documents
semantic indexing for, 4
DOT files
outputting, 7.9.1
downgrading
semantic technologies support to Release 10.2, A.2.2
semantic technologies support to Release 11.1, A.2.1
downloads
semantic technologies, 1.13
DROP_DATATYPE_INDEX procedure, 9
DROP_ENTAILMENT procedure, 9
DROP_POLICY procedure, 12
DROP_RULEBASE procedure, 9
DROP_SEM_INDEX procedure, 9
DROP_SEM_MODEL procedure, 9
DROP_SEM_NETWORK procedure, 9
DROP_USER_INFERENCE_OBJS procedure, 9
DROP_VIRTUAL_MODEL procedure, 9
DROP_VPD_POLICY procedure, 13
duplicate triples
checking for, 1.3.2.1
removing from model, 9

E

ENABLE_CHANGE_TRACKING procedure, 9
ENABLE_INC_INFERENCE procedure, 9
ENABLE_OLS_POLICY procedure, 13
entailment
invalid status, 1.6
entailment rules, 1.3.6
entailments, 1.3.7
altering, 9
deleting if in failed state, 9
incomplete status, 1.6, 2.3.1
invalid status, 2.3.1
SEM_RULES_INDEX_DATASETS view, 1.3.7
SEM_RULES_INDEX_INFO view, 1.3.7
See entailments
examples
Java (on Oracle Technology Network), 1.11
PL/SQL, 1.11
exporting semantic data, 1.7
external documents
indexing, 4.8
external table
creating, 9
extractor policies, 4.2
RDFCTX_POLICIES view, 4.16.1, 4.16.2
extractors
information, 4.1
policies, 4.2

F

failed state
rulebase or entailment, 9
federated queries, 7.6.4.1
filter
attribute of SEM_MATCH, 1.6, 4.4
FINAL_VALUE_HASH
query option for SEM_MATCH, 1.6
FINAL_VALUE_NL
query option for SEM_MATCH, 1.6

G

GATE (General Architecture for Text Engineering)
sample Java implementation, 4.10
using, 4.10
GATHER_STATS procedure, 11
General Architecture for Text Engineering (GATE)
sample Java implementation, 4.10
using, 4.10
GET_CHANGE_TRACKING_INFO procedure, 9
GET_INC_INF_INFO procedure, 9
GET_MODEL_ID function, 9
GET_MODEL_NAME function, 9
GET_TRIPLE_ID function, 9
GETV$DATETIMETZVAL function, 9
GETV$DATETZVAL function, 9
GETV$NUMERICVAL function, 9
GETV$STRINGVAL function, 9
GETV$TIMETZVAL function, 9
GRAPH_MATCH_UNNAMED=T
query option for SEM_MATCH, 1.6
graphs
attribute of SEM_MATCH, 1.6
.gv files (DOT files)
outputting, 7.9.1

H

HINT0
query option for SEM_MATCH, 1.6

I

incremental inference, 2.2.9
disabling, 9
enabling, 9
incremental inferencing
getting information, 9
index_status
attribute of SEM_MATCH, 1.6, 2.3.1
INF_ONLY
query option for SEM_MATCH, 1.6
inferencing, 1.3.6
information extractors, 4.1
inverseOf keyword
using to force use of semantic index, 2.3.4
IS_TRIPLE function, 9

J

Java examples
GATE listener, 4.10
OTN semantic technologies page, 1.11
JavaScript Object Notation (JSON) format support, 7.13
Jena Adapter, 7
functions supported in SPARQL queries, 7.7
optimized handling of SPARQL queries, 7.5
optimized handling or property paths, 7.5.2
query examples, 7.15
RDFa support with prepareBulk, 7.11
SEM_MATCH and Jena Adapter queries compared, 7.4
setting up software environment, 7.1
setting up SPARQL service, 7.2
support for connection pooling, 7.10.2
support for semantic model PL/SQL interfaces, 7.10.3
support for virtual models, 7.10.1
Joseki
using with Jena Adapter, 7.2
JSON format support, 7.13

L

literals
data types for, 1.3.2.1
LOAD_INTO_STAGING_TABLE procedure, 9
loading semantic data, 1.7
bulk, 9, 9
LOOKUP_ENTAILMENT procedure, 9

M

MAINT_VPD_METADATA procedure, 13
MAINTAIN_TRIPLES procedure, 12
match pattern, 5.1.3
mdsys.SemContent index type, 4.3
MERGE_MODELS procedure, 9
metadata
semantic, 1.3.1
metadata tables and views for semantic data, 1.4
methods for semantic data, 1.5
model ID
getting, 9
model name
getting, 9
models, 1.2
altering, 9
creating, 9
deleting (dropping), 9
disabling support in the database, 9
enabling support in the database, 9
merging, 9
renaming, 9
SEM_MODELS data type, 1.6
SEMI_entailment-name view, 1.3.7
SEMM_model-name view, 1.3.1
swapping names, 9
virtual, 1.3.8

N

named graph based inference
global, 2.2.11
local, 2.2.11
named graphs
support for, 1.3.9
named_graphs
attribute of SEM_MATCH, 1.6
network indexes
refreshing information, 9
SEM_NETWORK_INDEX_INFO view, 1.8.1
NGGI (named graph based global inference), 2.2.11
NGLI (named graph based local inference), 2.2.11
N-Quad format, 1.3.9
N-QUADS data format, 1.3.9.1
N-Triple format, 1.3.9

O

OBIEE
using SPARQL Gateway as an XML data source, 7.16.7
objects, 1.3.3
OLTP compression, 7.2.2.2, 7.2.2.2
options
attribute of SEM_MATCH, 1.6
Oracle Advanced Compression
OLTP compression, 7.2.2.2
Oracle Business Intelligence Enterprise Edition (OBIEE)
using SPARQL Gateway as an XML data source, 7.16.7
Oracle Label Security (OLS), 5.2.2.6
applying policy, 10, 13
disabling policy, 13
enabling policy, 13
removing policy, 10, 13
resetting labels associated with a model, 13
resource-level security, 5.2.2
setting sensitivity label for a resource that may be used in the subject and/or object position of a triple, 13
setting sensitivity level for a predicate, 13
setting sensitivity level for a rule belonging to a rulebase, 13
setting sensitivity level for RDFS schema elements, 13
triple-level security, 5.2.1
using with RDF data, 5.2
Oracle Spatial
prerequisite software for RDF and OWL capabilities, A.1.5
orageo
area function, B
buffer function, B
centroid function, B
convexHull function, B
difference function, B
distance function, B
intersection function, B
length function, B
mbr function, B
nearestNeighbor function, B
relate function, B
union function, B
withinDistance function, B
xor function, B
OTN page
semantic technologies, 1.13
OWL
queries using the SEM_DISTANCE ancillary operator, 2.3.2
queries using the SEM_RELATED operator, 2.3
owl
SameAs
optimizing inference, 2.2.8
sameAs
SEMCL_entailment-name view, 2.2.8
OWL 2 RL support, 2.1.2
OWL2RL rulebase, 2.1.2

P

parallel inference, 2.2.10
Partitioning
must be enabled for RDF and OWL, A.1.5
PelletInfGraph class
support deprecated in Jena Adapter, 7.10.5
PLUS_RDFT
query option for SEM_MATCH, 1.6
properties, 1.3.5
property chain handling, 3.2.2
property paths
optimized handling by Jena Adapter, 7.5.2

Q

quality of search, 4.7
queries
using the SEM_DISTANCE ancillary operator, 2.3.2
using the SEM_MATCH table function, 1.6
using the SEM_RELATED operator, 2.3

R

RDF rulebase
subset of RDFS rulebase, 1.3.6
RDF$ET_TAB table, 1.7.1.2
RDF_VALUE$ table, 1.3.2
RDFa
support with prepareBulk (Jena Adapter), 7.11
RDFCTX_INDEX_EXCEPTIONS view, 4.16.3
RDFCTX_POLICIES view, 4.16.1, 4.16.2
RDFOLS_SECURE_RESOURCE view, 5.2.2.6, 5.2.2.6
RDFS entailment rules, 1.3.6
RDFS rulebase
implements RDFS entailment rules, 1.3.6
RDFVPD_MODELS view, 5.1.5, 5.1.5
RDFVPD_POLICIES view, 5.1.4, 5.1.4
RDFVPD_POLICY_CONSTRAINTS view, 5.1.6,  pߏ5.1.6
RDFVPD_PREDICATE_MDATA view, 5.1.7, 5.1.7
RDFVPD_RESOURCE_REL view, 5.1.8, 5.1.8
REFRESH_SEM_NETWORK_INDEX_INFO procedure, 9
REMOVE_DUPLICATES procedure, 9
REMOVE_OLS_POLICY procedure, 13
REMOVE_POLICY_FROM_APP_TAB procedure, 10
REMOVE_VPD_POLICY procedure, 13
removing semantic technologies support, A.3
RENAME_ENTAILMENT procedure, 9
RENAME_MODEL procedure, 9
RESET_MODEL_LABELS procedure, 13
resource-level security, 5.2.2
resultsPerPage parameter, 7.16.6.2
rulebases, 1.3.6
attribute of SEM_MATCH, 2.3.1
deleting if in failed state, 9
SEM_RULEBASE_INFO view, 1.3.6
SEM_RULEBASES data type, 1.6
SEMR_rulebase-name view, 1.3.6
rules, 1.3.6

S

sameAs
optimizing inference (OWL), 2.2.8
sameTerm built-in function, 1.6.7.3
sdo_rdf_internal.convert_old_rdf_data procedure, A.1.2
sdordfwm_rm.sql script, 6.1.1
sdordfwm.sql script, 6.1
search
quality of, 4.7
security considerations, 1.3.10
SEM_ALIAS data type, 1.6, 4.4
SEM_ALIASES data type, 1.6, 4.4
SEM_APIS package
ADD_DATATYPE_INDEX, 9
ADD_SEM_INDEX, 9
ALTER_DATATYPE_INDEX, 9
ALTER_ENTAILMENT, 9
ALTER_MODEL, 9
ALTER_SEM_INDEX_ON_ENTAILMENT semantic network indexes
altering on entailment, 9
ALTER_SEM_INDEX_ON_MODEL, 9
ANALYZE_ENTAILMENT, 9
ANALYZE_MODEL, 9
BULK_LOAD_FROM_STAGING_TABLE, 9
CLEANUP_FAILED, 9
COMPOSE_RDF_TERM, 9
CREATE_ENTAILMENT, 9
CREATE_RULEBASE, 9
CREATE_SEM_MODEL, 9
CREATE_SEM_NETWORK, 9
CREATE_SOURCE_EXTERNAL_TABLE, 9
CREATE_VIRTUAL_MODEL, 9
DISABLE_CHANGE_TRACKING, 9
DISABLE_INC_INFERENCE, 9
DROP_DATATYPE_INDEX, 9
DROP_ENTAILMENT, 9
DROP_RULEBASE, 9
DROP_SEM_INDEX, 9
DROP_SEM_MODEL, 9
DROP_SEM_NETWORK, 9
DROP_USER_INFERENCE_OBJS, 9
DROP_VIRTUAL_MODEL, 9
ENABLE_CHANGE_TRACKING, 9
ENABLE_INC_INFERENCE, 9
GET_CHANGE_TRACKING_INFO, 9
GET_INC_INF_INFO, 9
GET_MODEL_ID, 9
GET_MODEL_NAME, 9
GET_TRIPLE_ID, 9
GETV$DATETIMETZVAL, 9
GETV$DATETZVAL, 9
GETV$NUMERICVAL, 9
GETV$STRINGVAL, 9
GETV$TIMETZVAL, 9
LOAD_INTO_STAGING_TABLE, 9
LOOKUP_ENTAILMENT, 9
MERGE_MODELS, 9
reference information, 9, 11
REFRESH_SEM_NETWORK_INDEX_INFO, 9
REMOVE_DUPLICATES, 9
RENAME_ENTAILMENT, 9
RENAME_MODEL, 9
SWAP_NAMES, 9
TRIPLE, 9
VALIDATE_ENTAILMENT, 9
VALIDATE_MODEL, 9
VALUE_NAME_PREFIX, 9, 9
SEM_CONTAINS operator
syntax, 4.4
SEM_CONTAINS_COUNTancillary operator
syntax, 4.4.2
SEM_CONTAINS_SELECT ancillary operator
syntax, 4.4.1
using in queries, 4.6
SEM_DISTANCE ancillary operator, 2.3.2
SEM_DTYPE_INDEX_INFO view, 1.9
SEM_GRAPHS data type, 9
SEM_INDEXTYPE index type, 2.3.3
SEM_MATCH table function, 1.6
SEM_MODEL$ view, 1.3.1
virtual model entries, 1.3.8
SEM_MODELS data type, 1.6
SEM_NETWORK_INDEX_INFO view, 1.8.1
SEM_OLS package
APPLY_POLICY_TO_APP_TAB, 10
REMOVE_POLICY_FROM_APP_TAB, 10
SEM_PERF package
GATHER_STATS, 11
SEM_RDFCTX package
ADD_DEPENDENT_POLICY, 12
CREATE_POLICY, 12
DROP_POLICY, 12
MAINTAIN_TRIPLES, 12
reference information, 12
SET_DEFAULT_POLICY, 12
SET_EXTRACTOR_PARAM, 12
SEM_RDFSA package
ADD_VPD_CONSTRAINT, 13
APPLY_OLS_POLICY, 13
APPLY_VPD_POLICY, 13
CREATE_VPD_POLICY, 13
DELETE_VPD_CONSTRAINT, 13
DISABLE_OLS_POLICY, 13
DROP_VPD_POLICY, 13
ENABLE_OLS_POLICY, 13
MAINT_VPD_METADATA, 13
reference information, 10, 13
REMOVE_OLS_POLICY, 13
REMOVE_VPD_POLICY, 13
RESET_MODEL_LABELS, 13
SET_PREDICATE_LABEL, 13
SET_RDFS_LABEL, 13
SET_RESOURCE_LABEL, 13
SET_RULE_LABEL, 13
SEM_RELATED operator, 2.3
SEM_RULEBASE_INFO view, 1.3.6
SEM_RULEBASES data type, 1.6
SEM_RULES_INDEX_DATASETS view, 1.3.7
SEM_RULES_INDEX_INFO view, 1.3.7
SEM_VMODEL_DATASETS view, 1.3.8
SEM_VMODEL_INFO view, 1.3.8
semantic data
blank nodes, 1.3.4
constructors, 1.5
data types, 1.5
demo files, 1.11
examples (Java), 1.11
examples (PL/SQL), 1.11
in the database, 1.3
metadata, 1.3.1
metadata tables and views, 1.4
methods, 1.5
modeling, 1.2
objects, 1.3.3
properties, 1.3.5
queries using the SEM_MATCH table function, 1.6
security considerations, 1.3.10
statements, 1.3.2
steps for using, 1.10
subjects, 1.3.3
semantic index
creating (MDSYS.SEM_INDEXTYPE), 2.3.3
indexing documents, 4.3
using for documents, 4
semantic indexes
RDFCTX_INDEX_EXCEPTIONS view, 4.16.3
semantic network indexes
adding, 9
altering on model, 9
dropping, 9
using, 1.8
semantic technologies
overview, 1
See semantic technologies
semantic technologies support
downgrading to Release 10.2, A.2.2
downgrading to Release 11.1, A.2.1
enabling, A.1
removing, A.3
upgrading from Release 10.2, A.1.3
upgrading from Release 11.1, A.1.2
SEMCL_entailment-name view, 2.2.8
SemContent
mdsys.SemContent index type, 4.3
SEMI_entailment-name view, 1.3.7
SEMM_model-name view, 1.3.1
SEMR_rulebase-name view, 1.3.6
semrelod111.sql script, A.2.1
semrelod.sql script, A.1.2
semremov.sql script, A.3
Sesame Adapter, 8
optimized handling of SPARQL queries, 8.4
performance recommendations, 8.5
query examples, 8.10
SEM_MATCH and Sesame Adapter queries compared, 8.3
setting up software environment, 8.2.1
setting up SPARQL endpoint, 8.2.2
support for connection pooling, 8.7.2
support for semantic model PL/SQL interfaces, 8.7.3
support for virtual models, 8.7.1
SET_DEFAULT_POLICY procedure, 12
SET_EXTRACTOR_PARAM procedure, 12
SET_PREDICATE_LABEL procedure, 13
SET_RDFS_LABEL procedure, 13
SET_RESOURCE_LABEL procedure, 13
SET_RULE_LABEL procedure, 13
Simple Knowledge Organization System (SKOS)
property chain handling, 3.2.2
support for, 3
SKOS (Simple Knowledge Organization System)
property chain handling, 3.2.2
support for, 3
SNOMED CT (Systematized Nomenclature of Medicine - Clinical Terms)
support for, 9
SPARQL
configuring the service, 7.2.2, 8.2.2.2
optimized handling of queries, 7.5, 8.4
searching for documents using SPARQL query pattern, 4.5
setting up service for Jena Adapter, 7.2
setting up service for Sesame Adapter, 8.2.2
SPARQL Gateway, 7.16
customizing the default XSLT file, 7.16.4
features and benefits overview, 7.16.1
installing and configuring, 7.16.2
Java API, 7.16.5
specifying best effort for SPARQL query, 7.16.3.3
specifying content type other than text/xml, 7.16.3.4
specifying timeout value for SPARQL query, 7.16.3.2
using as an XML data source to OBIEE, 7.16.7
using with semantic data, 7.16.3
Spatial
prerequisite software for RDF and OWL capabilities, A.1.5
spatial support
orageo
area function, B
buffer function, B
centroid function, B
convexHull function, B
difference function, B
distance function, B
intersection function, B
length function, B
mbr function, B
nearestNeighbor function, B
relate function, B
union function, B
withinDistance function, B
xor function, B
staging table
loading data from, 9
loading data into, 9
staging table for bulk loading semantic data, 1.7.1
statements
RDF_VALUE$ table, 1.3.2
statistics
gathering for RDF and OWL, 11
STRICT_DEFAULT
query option for SEM_MATCH, 1.6
subjects, 1.3.3
subproperty chaining, 3.2.2
SWAP_NAMES procedure, 9
Systematized Nomenclature of Medicine - Clinical Terms (SNOMED CT)
support for, 9

T

timeout value
specifying for SPARQL query, 7.16.3.2
TriG data format, 1.3.9.1
triple-level security, 5.2.1
triples
constructors for inserting, 1.5.1
duplication checking, 1.3.2.1
IS_TRIPLE function, 9

U

uninstalling semantic technologies support, A.3
upgrading
semantic technologies support from Release 10.2, A.1.3
semantic technologies support from Release 11.1, A.1.2
URI prefix
using when values are not stored as URIs, 2.3.5
URIPREFIX keyword, 2.3.5

V

VALIDATE_ENTAILMENT procedure, 9
VALIDATE_MODEL procedure, 9
VALUE_NAME_PREFIX function, 9, 9
version-enabling an RDF model, 6.2, 6.4
virtual models, 1.3.8
SEM_MODEL$ view entries, 1.3.8
SEM_VMODEL_DATASETS view, 1.3.8
SEM_VMODEL_INFO view, 1.3.8
support in Jena Adapter, 7.10.1
support in Sesame Adapter, 8.7.1
Virtual Private Database (VPD), 5.1.4, 5.1.5, 5.1.6, 5.1.7, 5.1.8
virtual private database (VPD)
adding constraint, 13
applying policy, 13
creating policy, 13
deleting constraint, 13
dropping policy, 13
maintaining metadata, 13
removing policy, 13
using with RDF data, 5.1

W

Workspace Manager
enabling support for RDF data, 6.1
inferring from version-enabled RDF models, 6.3
merging and refreshing workspaces, 6.4
removing support for RDF data, 6.1.1
restrictions and considerations with RDF data, 6.5
restrictions with version-enabled models, 6.6
usage flow example, 6.6
version-enabling an RDF model, 6.2
workspaces
merging and refreshing, 6.4

X

XSLT file
customizing default for SPARQL Gateway, 7.16.4
PK—* PK.AOEBPS/sem_prtusage.htmr Conceptual and Usage Information

Part I

Conceptual and Usage Information

This document has the following parts:

Part I contains the following chapters:

PK aw r PK.AOEBPS/img/inferencing.gif GIF89aF"""***ؿݻrrrHHHPPPaaa DDDfff&&&777nnn???LLL]]]jjjwww...UUU{{{333YYYǪ;;;,F@pH,Ȥrl:ШtJZجvzxL.zn]uq~\iYThMQRJNGcBIXEOŤDˠC՗٥žKG\FZ ,@.s 4p{R@PZ{X$L@j`@?] !hzBŇ&a\eL4u̹瀟A -z4)DE(-T$(SUH N~p(BQj \Vc/`FcB[S*!AŠY0E`ѢR/(qwdzҶz @8qȕ3wNdA -nPȁJߡAu-1@+GK2{Y|pן8`D\ CDJU $m@f$30{U@\E E* )O5ޘ#;c{)D ?UiHF*jE6HѠ5 C 0X Bcf]+fsyg{z矁Z:&3O@@]H5* e~ lԚKÞ.l<;-nCml{FЈ .--baKQ/ zBr0Y n' g셬1Ok,23!Q;vƞ 8+/*2C elҨj0E3}Ҷ6l2V_?y]v kc>wn{zi2|#NbEm8gM\J_9ՊnF:Բ:g<4Ԟ{},n އp¯|7Ѓ.>FW/[1t'ᯡ=w>l[>/*}#_?y$h%}#*\SUA TF0~HПDx'E@vp !My0qdAn"^-*$=OdB7&v 9  8 4a4Xh@)J94"b)pitD x @`<1I tP?jHȇ0`vyG%c$2iNv")xh3(}HJD|[6\d+l0ܮВh0cHB gfb~\,(!$0`Q%<a%TЂS=bX 4B})E!V'@4Ljb! 2Hd5 e Qj`iJ`tyK_M1i}ػVcA 8J`-!;9Ϥb 0SZWJ@>%eXPgP`.Jx`0)QxSݠPX7`Z3KyF pOr7(p*: OJepYl@a`; "h: < D1`8"u  :B~Bf%˹նF]R@m2d`  !(.Nux)eEwԦA.0+)$j ~;ivBx7Ke{,u .1,ZKbBLSHf[\%]@#xB " xz)V n #`"*;16@N hsd{_9|@TB& Z_T83\5Oa̦&4gRP ~5%kV_ lSmk5KyVt0efv d[;J|起 !ܾ6:p[d.Krms{Q_·kw/&sm!`C|O&&Õ-XrOV1/nn r!(ry 9k Eˡ l|r*bDm{c^śş<cϘFԲqp#{=7xW.$+y@fĺ]}o+ V7T;r1X6&,lfs&!\1UwXV$'ɥ;sF{+jO|C@ 8 ɧݕ@ zFZT CӖSԢ-F;ѐ(+6^ `ſ[)x7N{ʥ3CR@?wdU͗,@UT5]U!tUBGN`.OuR0WYVn%fW"0WqL! `w:#}2TG XڷX6XY.#p#@r,l8 lZQÇZZg)E%` ւD8[g U{\UUy~Z; U`܅ו][zzX}@8OvL_UgOO_`ŵ||eƆG;Å-5c$a!vafv1a!$fbOW~Ejfh4azf9&;vCc@&d_Udf+v(eBPeWFLY%e`E F4Njp>fv%gBg:g\hO*xwȈ=V7iBPi.piiiA&wqFԌxcvrE9'7@B 29.UF.70C qBH BV\ٕ^`b9d QLh3j3sn4;PKg PK.A!OEBPS/img/spgateway_browse_1a.jpgJFIFLEAD Technologies Inc. V1.01  }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyzw!1AQaq"2B #3Rbr $4%&'()*56789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz ?o@_&kɪ=v\J ɺmOV:͆Lrnہny ?VP㌎/KyX_jЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxofGxcǻ%eECKDs Ӹj}9xTc(AWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo| xWtqXbAq92:ƎdAI [a2}GGwq̗4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAq?| >txߠdwc9cX1֎DY#ae` 0ʰ̃N᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dwjЭ5s ӸWxo+x+}4[ǿ_kAp/ =/ _G2 ;5wVxW9i?᫼ B ¿̃N ]_{^澎dw8B}&`>KoA=?pqRGqzgȶ1} tgqFOE:X[yX?O;g?=8㎤}3Ӛ6 #=CGq'KZ,/Olӎ QZzX:{`<Ϡ84m:z::;?EkOӞ}RȧKD2=|h\}y? oP@P@P@~889*3S?@ߠc9> w89 Hi=;zL{c˓ds?U%y~nA#wPGN [ x :82mamaGusӓi~?,c89 +5hA{v@R ( ( }1 45 `m EcOrz;Q @ ( ( :w#Ütր `^֞ vCK'`qqМO AwG$=C?1 *~9  sF=Ktqg}4ñc؎8 `N?mVqLJ( ( (Ƕ0AWi~~=;>{)aG7oK(P@P@P@ n@ǧ{cހ 1\8$9w=[y /;8;v z?zvZ!ן_NyQo+ o+ =@OOޜp(Kcy8GqiY @:˷P@P@P@[ֿo^~8 =>/^ (jZm֟im}a}cq5gImna)#7FGNJ e|?>&wK6?|]G/Ac~t[B\OXY b.z3evxo K¿%_x9/o?z>mxrWW隋_ZjVh6(ޞ"IVH"ŧi}30j$^se[jZ?k}"U%65[$$>-ǔrAcؾ B־#85]14}/w|3WFi%fqii`toE! 2/iy_ӷxx |)}4o鑛i:t5HjZƹws˧ ?cN;38>xcԦ>$o]xˬxn)m^.-t{[җZF (a#̝-m6G13f2h;__Igy;y?1l/9f{{'2D3-o}Cx;+4Zamz5 ; Nmo^bG"C:钔ki~6Wh nZO>^ ֱx]*i>-oٵ=B+7V1V/ao+ukPixP5Ma<_hM慦MĭٴoWe7ߨ\9$_-dXKV,Gϵim"OQP@P@P@(|\d_DHMX|1]g? VX'ľi\EocZZ{wNoPيZ4G5G]H6=sSԭ5?χK<˰oN/4ҿFC?a[/Gt`?>b/c)}Ē[KXvKǬ?[+Fޥg> W燾!~ jx6/Rr˨i-kYjVo[|;DzWW-?cSE mN~ O 8fGhd,XO1^^ 5G¿Q<W1ŧ׼^^)|/?=JG&C/w}o!h7>T8OWW3x>".UqbѼrYx 1kcPZ[!M f,< >"Yh.K"PE; XmYhvڍζR,jU`Z}u$66qI}{{^V$棩\\]l|˛W $<`H ( ( (w1`Bb ( ( (?jsP_dZ5i~*şxh_%&2^C݊%G_xZ'|-?~ 5O C}/@Ş$3Ჿf-5;=FCkz,^{2ò_#ӾE{{45 kmKQv_ğ&>kZ_ \|deyݭb!廷٘7ږ^Gh<;%wu>]-K%S\E2H %}_h۾'] ~:|UC1DWQ>Xۡ eM/DKwU>$/>0k e=z\W)x=CAt+]7×z$6&=mo绹%Pjֶw*5Ś7|_ xcVw? wW%·$M0TYu}SZlK &H_/ J};_>%񆏨͡x>}@߉Ht?zxgD]MORӯ \ ~S^hkyX_6^h>&wxgCKxW@NQ/|CeNm&kQ;~AXGش[׆s'> Mf\/uCm1lKCBRX.d62՟a5cP@P@P@"m zwP( ( ( ZEQwGt.m)5zx>Z{m ynᱱ^(#dkExx^ xn~~#׵Io|Gt}cVѥoƣ}YԒޥ\Ʊ,t=c޿o7+?OZſ<=KUXj{a6k59wچmthe cCJmq=W7ϊߊwixƖ6Zo/x^X[kt#~5;F!eVְ/Ǣ| Di>R->BMwKy1R=5Ć|3[y A_qxjğ߆1uG⫏д SM·5 [wOw$ oEmt4gWIDuvź]OVZ/m(ZmCnfuҖ 嘑cV[," {!KJⵉ 5Jdk\O4i&vc@P@P@P@}qZ|7e㎃=&Yrا񦷣i"ѵokGtCFlզvX}XMk4^bu 0l2FV9viь?ƞԵkMO@#х:w]ƙVז[__mw} zwzUyc@n|Oh3ŗ uu ]`կkM6!i,[Q 2vW|7^-4i xPe5bf~qr_fHi im`Mt77G:oOgkR"T̽ xjZޭZ7'-to>Vu7V%ao*>ر#DjV`YվxWh+?_om|2[O%·o=|%oFY X/xGB=t5 >kY#eOjvZiCq3sL{>VUI m,VK4vⵆak6vV>;mu;h5%x-*5]65D gBh~|=گo}gㅻgZÖFAdǤX@UӒtRD.וoEi62suoZ}ķ/gZy&`}ԞZ=F?ȱOaQҖ!m.zv;(( ( ( |G)lGjv .4X5x ֟E]U:gm㸎IS,o<#W^Z/#._ Fm/ j5e/{Rտe ͨ }]m`$V+Ş֧xHuƫo X65=rZO:mxUL]гmڔZmGW5#Mu߈_Ch6~ Xwo'Ƨuk0Iez"\f{yv oҵ=Z~z?ot =oĚJIsj/3fX"xVV-xgo. 1-VKҮ$|G?igiZ k^+ͶA2@R'G$QepZu}릃_mO0xvuiuZkqxXJ7zd[2Jͽm; SּKOjzwjBP@P@P@SU8ÍF׾(s# @ ( ( M#Kԁ^GFܸ2D@-6 i/oRcӍ mR}MG{cg&c3lmѳ#yX6~$iMk/uY<3.RGž$K}[ڴ540ݟtj/`U :]- 杭JoOxfox}DjPrM7FFȲ&<|p7+};)*/G?м5}xg$&S:.@>ËooRMQ \xwTѼO jZHm5m}}kO.w#z,,'#w1e%qA-ݳ&mH/j6dXiZCm,n ( (?.G<m E=;>zYڏND ҏ,\ ( ( (;~&n<7swyjNxk^濲NO5唦+x ^r9&קa/'Yw?| xT[ƿ&kSھ95Sjz="4K/HKoԢ^j:in|B-KB V̾|Qy7u+x"ZCެHcKJ'o[|}$o6^+ 5?(֧~ӠY~$ׯm2VHtUy|2įmb׆~OMk}lt*G}_qٶֵ⻽1d=$"xDrE] Q|e^jWh5vWiw_O]W[5n4>kwL..\6c<kľv}^:ơ?lc5֖Q_(R0n۾/-imy~gOczc֩+[@P@P@P@?Mτn<7sAMrP/ [V˟#mѯGC!x7O+_![2/!ru{x x{G+Akip^?]s!:V @q:V <ǼB?Nn!9Z [ @??1ru y2Cu,?_>1r/|P?Q CG+ 2Cu,?_>1r/|P?Q CG+ 2Cu,?_>1r/|P?Q CG+ ,P?<f,3oqpo, 9Zk&^?";qz |?_>1r/|P?Q CG+ 2Cu,?_>1r/|P?Q CG+ 2Cu,?_>1r/|P?Q CG+ 21ŽVAku?ᗾ(~GB1rP^?:V/|PqQ;ڎVx x,9Z [K2/!ru`^?\{'gQ`/|P/x=h-o+2/ru,x xh-m.x !=?No?ᗾ(}ca`^?{x !:Ve/x9XX?ᗾ(}ca`^?{x !:Ve/x9XX?ᗾ(}ca`^?{x !:V{8λ<w#u%m/>jo"uZ G![&^?{x !:Ve/x9XX?ᗾ(}ca`^?{x !:Ve/x9XX?ᗾ(}ca`^?{x !:Ve/x9XX?ᗾ(}ca`^?{x x'{QZ\?ᗾ(~GB?M?\{<G+Aku?ᗾ(~GB1u,x !'QZ\?ᗾ'~Gr]{=G+Aku?ᗾ(~GC?No0^?{x !:Ve/x9XX?ᗾ(}ca`^?{x !:Ve/x9XX?ᗾ(}ca`^?{x !:Ve/x9XX?ᗾ(}ca`^?ǯ>y{}qF>@ @P@P@hZvk:7Zwv~iKu}jrG 24$Q쪤&a8 5\n7V >NukׯVqJTԪUVN8FRڌSHRiS!w)4ݷޟs.>`W5N Ƕmvxqߥ ZZO=]zW~Z N~n>Zݷ[vh]n֕d^wOGFV,.4KFZ}3FVԢu b]?LYHu[ m2rUpM)ߊm$4O$7iƿleǝeXǺ C1ʳ\pYr/a`k´a'q\VXjyգR1T8SE$wZYJKM]5}ztߊkm>}[u7cׁӎ1EҶlv6򵴷m/oͫеմNQ5k;k#RMN]}_I[yr5i8SN5,-lF 82RB5aZjю' ʥ7(*zԛF2ڋM٤OJF$Ogr xV;C匆Gyj~jZ.MOeƣq 7<;kZchRmi#r7FY1<԰jؚ+a9mapbq8zJBT8:)F+H=dּo1m&zN:3>O%ߧ׷a7SǶ1j/nW\6~;[u=[L4CY5-#G,n=WV.3Kl Po8lbigD8gvURGN b;N :׫5N4SZIF:p(1mmz$],ȼrg+]m姘LcҝoU_͇*Lq-m>O뽞( (;moz}KRԬtST46Qu lt(+KHmydWTDJBJTҋ&Fr)Z1b|Ufta0n+ qʴa)ʵzSFjU*իRQN"9ǙnlzgWtwnu֚o81~t^IbnڮKKOaQO XU$yZROu 1+I"DBncBQ\Ӓ"yJr"6qXڮ W^4tӕYƎLV*QTv=*(ӄQҬ/uMRL4KGQ. .on"9%i]6v!T'VJIR949JRcܤWXl [էCJZ՚JT*jԩ(ӧN6ɵ{{^w[mm6z=[]9KO]U98)`ӧOM ( ( ( ( ( ( :J67;OqFoL{c?8zp03Ǧh=1րv~X(?(3nLoE.q~Pd ?@>3vw [WҴ +RuNE4]:V5ZN+MP+#)fy8UIϼ:I5; wPH//-n5V/S%4'QCѵEĒ-ypWɶCw3cNlv8郃~z%Wƥ6sjN ֥ymcm&zG=ԈjuyD`  q(?ǧr5UwE5.].T 2Pom`PtIFwU+ ÌqӟPWҦo4(=>]oM[Pѣ}VK֮uk=SCqka}wk[\I<%D֓/1Ҁ81@ϧ}h?Iմ{J5 SYմ}cItWJm-7PH/.&xn 9QљX{Pvg:?€3_L״3]е?Z>Vѵ&Qҵm+QNBI &xn 9QљX|x~kNK:?gox>+5W/S}xY?gOh:Gش=:_0>^}1h ( ( xh.C+{A5/u% af L>5$^ס];|M+\+]O پ+_3]Xե>-aaʾCK1򥄣ρ"S37՛v*Q^(6U~o')|S]?7l|oW 67z߃^)dzZݾa.ojR^iVzՅ.^(TC ;R*x\1Yʭ VaF8j4Kc ](K+/TM6h*)B?c~t䜜(Aŗğ|[|eW|7߃uM# m>%ϡjZx+QoXͨi־=ݶ.6mg~qF eqMnL;^%g&_* asճ:ټpCK3<'.jԠ3:ThɸƗ*n򄢹;*oފtҔg* He:sZ-^_jƭeb]JL{h'n| ּ7꺥彯_`5ܒocM ;_Y\]7^u(.tb~xc/֯ UcX|=:1+Uy8{'q1UqnyүC0Â< i%(NQT0M7|jib2V^;_Cv:?{JSP|5⧁ "߂<=iT߆u/F^wM_T.|V޵Jf~ xcZ_χ<%tMC[Ua<#<6Z|#Ü?g,s,+*x|V2lbS1ch3L jYC~խJ2pNNrrN}J|KnψxkiGJGei^%&t+FZ<t ŷZޓg2k֚~nʰ*U0k`* qX| 10 8{ʆ1SSSSьjTzsJi1e J Q7{UrW"OxCG?7> SB xk֫~r/ \rZׄ547V.[S#_?Gp3Xoaqͳš09'*U)2+01LJ"ҢW?IW Qާe*nP%Z? |@ƻ+.BZ4}? |[o 0|-i+j2'ĉՇ_-iqǣK>hg գp;Vkc3̻5g13J F?! #K Vefi}_q:>xsGlsܫ^3f،ڼf1c<pY.sc[˕9`iSRUUYSPJ4U?ԗIs'߉ 7KFttKٯH3yqiLoz,<_qy X/n $gUR>85,! P%|_尩Ͱ3b>J_W _8Q´O FueVJsJn44.|Bſhz>|,?4 "7wvW:;tdGVWp<fT1t2^'5Le; t0ҩBX- } UGVZj}Ɩ"9FU(RNWg$Ug(?B?;Iﭵ `I&D/cԿEØ(Q N*箆c$yckxUG(O W_~VէR5+δR2Pt;rSpVNJx 2Ş3[/OԴ?c i^!W47zCCKk[idPl02<.Y3ڴ+R̰If8kb+g<6*+S*yԡ,6WU)%#VcOJT!BiSFi+{G xGI}U?ĝBmx3⿎?Vaac^|14?E6><_-"Y,kyg qg8jMpvK9+#Uψ>qN+bqb3N#e8rǾg U8ҭ*h҅okWR5Q8*PR2,$'*ҧ}&r~Y~؞6B<=g\~u B/ w~>jxųj6sGm3/ ;s0nO:q5o".:<;:9aif!qVx̹a8ZR#WPؚ*XřCShեQҧ4iB NgI?v1+w's_o' sðΟڿ#LѼ s97ɵj^$CZuy|;J[?+>upG51^;OX9<,4Q .g#..ieK`iZhV5,#K:˞𜛔.I($E84s\_^|aϯCR:׋txt .lɨ' ?co KՇx־'6y x< /ar83#)SRuZ1UW,a>L<+^|Дۓ+Q=ĹR>#W_#4MnO][Zr)D]Aseaz>޶i ZǫL+/x?Cg\FTEekxً:XZUe԰xeTn0%\NS/IIRE8a҆2qJU9f.y~Tx?vyc|I~k?|_kŖ|;6m;TMԮ.hWrwqn rt3Ll6IC9be](xLn Jq-{zxڋJxz5i-oUӗBIEM9 [swX:|}5^WiZO Zi5 |{%͏kZΗZ\?œyJAip˩18f]1bs م\VkYEPefSlF.sJXJRjN~jj֔Re2Q~ѫNu'! |Q~Ο~.4|ycg  =SHo;diaᬏA/qoLFk?`eT̳V*'Ti'ļ] eZe[dB*΋d:zIT|גl⹔_o#o*J_%׍χ:>|XAMĝLߦoIhYR5_Õg^"p&]eu13Fm<\s70(,$p>OxxZTҝYV\&*o(EÒ0QTӺԚj6_k_~w5_~6j?iG.ZN Mh~$E/[[k(𶧪_Wt"&~d)iw%gUjU,G hSFi^1P_iZ'ѭI~*e|[v^#͸c˾ř[V[`T<,elUZ 6Ӯ/Mu/Y!mmsmx#޷$fi,Y]ZgWMrЧaa0Suy3T8t]P52\s4ڄ(8N7UkB1ESxI' 1x좽/\*~EoK,LjU>Fk,\C"ա(3*KZq88ֆ6HakeS]wx*CUƼ8B8eMªNU=Oѕ|gi'Bf\3|_尩Ͱsb2\xl4~N0}}k#ܾV,}0 ( ( ( ( ( ( (,x'/ß xC:|#Ӽk}k:TQOsDO[\\Jo#{t]J_J_)W/mv^>k 㿂='}7tm*,.5Szm` g/Oᯇ'>/[}gW |'O)dkԚݷÙo} RѭIkNkn~5x o -~ᧀ.OZ2&ץ AŇf(| ߝR}GDt1Sj13 |/ %?8|@d~_˲𿏾i t qijf&`5?4? ~x74>. Fx3"O}?Ekyoe{bIu$&/oD ~'5!/^/C^)oOz# Z_9=yhz}录ρt/'Do?R[ϊ/𶷡~o'/tT#g'5[_|?ִ}ׇ>4w3]~[}ٓE| kOeVW~~ϗ*;/3m4˙"Ou-:^kMz z+˛&+3N?@! c㿋gx}~7i- _› o|$OěHiTuHLO~mz]6?Zyb 6'>|K_ŗZxׄ1OxVR+ IRQPWSu Y I4[ ڟ~~|yX*Te JŚofS2|>#c/S|uZ65^!m6/|`lg]GGfCS| 3>K |Z~#k߇_V\SXW1ϣ`?Cx/}|cK?MKi5V~+|s=k^xş&P_|dxug,Ϩ:gjyek1S-ͼEini,<1}I [  >83wV<#4~miI xsKK=in(-&U6߳ kZGψuk}]s\}Ӗ[K$;Y׵PčgWí⇌>_Ʊ]xHi}U⿌Zm '|?}m!m*E[I|o<_{cV^ ?|Ai >/h1K^m|=[kK[xWI˿Ӯ%5ǯ~6|.Ůx>d0|9m[7GM"|~-ηhelM7:dv~?HD>2x1h o|}RǟxOITn+=3v\^鏦:}֟a|ic/Oهӯk⟍֬‡".s`}:4gS@(x/ zbu麞tWUԼ>[j3[dO!tCwU|oſ|txWR- χ3ռ[o폈~ O> 'IoD~}$ޟܟy ak&>-?^ ~d^Ho[ᶃc{W/N+]L[.ç0;N#>&O$t? 7%/<^ Sf x'-j}Q-9>;>'x{nkx 쭥)O|If?/MCP_tz/ uu-<7O?~=MCW^+ž>烴4go^/E|`ڏ4k:5k!A"l6~3{>㏀k jxL,ui4EW^ym Oi照8PEA6Hf-^nuMI9%}nyΩk%]kXiw<3ke[W,tվ:V"%V:N]jghO62x_ 19ecx_f4sx~.bU~.~l~29V(,3Qs7ҏ%ۗW%ʣ2|T/]bP,i|-3>gj]ޫZ7X4ȑEȬֱ4;`! GFl2KV%Jb4Ԝe:P!ZOj))1o<7UD[CA'esE6wp_+T'0_ZZG\HtV~W>"Q2_ 4sf4&**3X?µ7RJ<'8)'Sʤ:!-8a+t[XsxsM?t-#wךkS խ5 Cp跷V ٤2Hޠ̷:1(̱9;3<(s+У<5J4qbiRN:Ҝ) eBB1ԌiO ԔR.^e)'mZw)k߲ͧtT߉,Ğ\~҄>z4{ӋD2ҌNYF4RVjK Ԗ&^$|e?^7?x[k'%?4뚇-=NyÒp54,^ֆM]ER,=.]^_쳮xAT~0ⶱc5{=TZgY|@߆4=w|%cjzݥ4z~gwswo(X-3W/y5:GTd9f]C'!8[-ʰzhB[V3 bJMK՝Ey(JQ2>Nyf{E6"~Ⱥy|INoN/? dEhi'f; \[(L38,\;l ^<^[^;xZb%9V0<vٷ}u7|E%UaWʚ/{ki{Of~DKƼ[ë6ogKş `ٝ_߆??t~͖ZCo6;ky7\EUaWV }T+p ~Ƽs[7k8hT.=7z7~$x:?F'AU$ioՂtX߳D0V"Sa4b;۶yxј_crJ]q}~z5lZsR>J.z4{Ӎ] Qi֜Ui%-XTv7:XƧ➤?G~OT%K4ks0i<6"%Y}W y.'͌+,٬KG~X|uav\KC>;ÃZ͍Io?<#s6Z]qw*̱U#iG qN:B#:Ыx|*RBӵ*42Ju={s;gs_dvJ+%k_;Vz@P@D/;䁌g?z壵m-]ܷᶾ>gɾ2w½Ś?6G*G;E?l?:"Zڭ_ܼ\L4&'"+I9Gr&-vKe|e&esSQ8Jh$Bp\q~jwvVz9%Վ6s\ݯ,֖,0+EVG^!TuVsP`✿*8-ny3|w- 0`kWy XL(SWKrt[O ^cՆQ:c}jJLu-r63brs{R3jz5ż\ ^YeMFv\ITT5jJ)OJR&ܨR[)?n-F .,fl>ohPh0ѥONaF)!!Ƣ*jOkg5"c/Ch߳?ײj:fyVK-" S˚G63zXzؚ<δLÒNp,<|Qmjї_)ʱUV[f>yv< ĬEgGᇯBk{Zrs"7w{+ ;;>n-mm/}M)igeãM"["lk҄Ӆ9MԔ#B2I')*q9kHF?%N.AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$ cEM>AwZ_$c|@aLjt<>:Zvm5=FX㕡k9dTG4,P@I0 2:T.˭i~ĶVXiZΓyZDg {xXEyUlsJ&_x?:u [6VҼUv-xO wP߇ 9=NHVlgx[7Yi i^s2 75A.<3wuKBs%֏!co ,+Mկ43N׵mvZ OVB=Fi$[]&0$Co >|.gPmOMFֽ ~%x{㼿]7L5^F}>ROS|oɤiSjZf6jZNKgo&zΕyZn-tB99lUfl5y:UiEZ5֋*U NNl#.1"y: 3?@<6/Zǂ<|=Nt|kEl:W䲗C"ՔjitvElC e4u[W|;[^IѴm&NҴ/MOӴ:8౰( 4(DEU@6~|[~|0iäZ5|SV7Wij:uqlK g,luτ|-{i:~oiasiZ?)٣^}kfٷ860&o]O zxi5}SN|?{Μ%í%C` ~^sp>CLyKn~ k3Z5Ğ> ]ie5Q4dwwF6 >xW4/x!gX/ 躶WKm4OĚuk74E9#e[F hx[^6axk@tfs> NuΣ{#IϨ]B湚F千c? |P1}xo?gb]#KUeӬ$?St^K+g,4jI=A7}ރMsmĚ-ʮp̓c7ˏj^:ᧀ'|U5LJF4{S׶^uZTyri l'*o_G_npxwo#MfQQlv:^hfMjڴek$>z]2\4;㯄 >'eimo?5:?b<76f%6_&KYk|ޛZFKgm&akW:M泦O4)mcMIUl^kmk}擬Vo-jz}R}ZM,[̏U&xG¾?ׇ?ohѴ+Ƣʹ i<m4*~^iݶ6 NۏoO6#?IJ4- MӴ]D총Fl+IҴx3M >(5UPlP-kU[? ?c< jV ͢y#so  <'k>k xĿ?x@o4oYa5l7۵k{k۟\K<06 @Mž)ּso׍I 3|T7΁wwfgco!AB=s=~]:{P? PPPP=ct>t>t?P@Ƕ?t(~ҀҀҀ?ϥ=Lb:P~t?t=1@P@P@P@P@P@P@P@P@P@P@P@P@\}'h~ڏ? xTTP@P@P@P@P@P@P@P@=Nmœ|v %D!~3|@W{ڗK{τ66Z>gb]+-j@lwVkk i~.ž?幟ƶniv_dѬƺei#GQo+P@P@P@P@P@P@P@P@&q~P{c@Gdz@Gdz@Gdz@Gdz@Gdz@Gdz@Gdz@Gdz@Gdz@Gdz@Gdz@Gs*sZ0=>WMG,t{wm~)5^wo xͺ;/y{m>Aa5y4[䉝&f6^hڝgi3]ij{klK"&zn[]X[̩4.pq;cPp1:΀8>)x[>:uta??lk>6/5h閶d{h?Ҁqߠϰsϊj>u{__𿅭ͭSm<-_Xmo|!+:gVCqqStxEө;^9Ğ*|[j":~wO~h 𽏓co4o|M "d5̐,р. ڏ^8<-kMx[ľ4ͷ;-<뷂 N\}s {s>*ЬUx*7e27V ؈lmEAYy ?k$)&8?TnǯAA? b ==h:XҴ&P,.B]'BֵX4O\M`E}KPDuAIH*!@7c߃#Z욇5 xZ췗ښퟅK[6Hc"Ow$76Zj֚vZNke{muqiZfAn& opֺS N^-w]4]FumcXկ-/IҴiou OR8,l-m y]ݕ_|griZ6k<)6uVom! 85-+OբUkKfM"uDMV`E 5x~hkyVأQzzΝ7 }Byf;{>)мa\j5xZ췖^o> M}o/xW 3}ηkyb@~X1( çLx7ý.]OrHҧlɴ>]oM-'O%}VJntcM#=ރOsoJp4'JhZ&ag>gmiZN[gifgv}0 qG "* A#ր Z2=h#ր Z2=h#ր Z2=h#ր Z2=h#ր Z2=h#ր Z2=h#ր Z2=h#ր Z2=h'9-,Cq>{rۇhk\q A(Ϟ~/a@- |} ?hkߋXP C_=#_€ZH>zG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€ZH>zG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€ZH>zG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€ZH>zG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€ZH>zG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€Zx{WOh5gwzus{w.^\:!C~G ${009:~4L]xB;mWk ||-OF 7f}rmSUR}j-som=&xW1o?^0Կg>Ze}m~.^;⦭YMv]&[/QcR~e+\.#x7m dGAx/ |8vcLjyfL_>q#}o-ǃ xR@_ dޅ鿲>k+x ~:=kψG-b_3.4O?φ:)_ZZkZơV_>jִC|G5ߊڎyЭSWM?+[if:?`iQ]?u;Oݽ%Ӎc?.w9׼c0'~(x?R/|tϠxUn:'[Ѽge PXEƱi!1\i~](>,_/:x-!b]OGG? x>s-y/W|}})|tΉφ~&w6keၢji-z~[ǰ~͞%yO57YvyYoMǃF,4F}?C2͞k{[}i=gxZN_W!ᙼCrwrk,CFiVǎ=:}C^~G_֞$U[.n|c5ly~+i1>|oWC_?VL?įC]qxVOi  ]#Q{η͠_)/>_|Z|Uj_~#Koj0|"~VxMCiڇ젏ƺeΗHmM ?C[k1ӼK>ŇlOj!JC~~А|9%j|j#/"kt-b,zF V}GTComc="kׅ<_7c_;Ncx>WM?j xƺ~"Oj&]GgXм_ox7~G?Goڻ>ɾ->xVOڏ_kAnp/ ETWeԮ.|gG|o8N~:^?ǚ/{k4?1|T{oZxf .`=ݾgzC_]O17,-Com=?p)⼤?|G3ኑ+?^(~r<#x;3V]I|!|[?O|M _ӼY/|=hω?Ro[+Fڂ g~^)~5E5O QcRcmWoW 񕵽ݧNux\,q;^,ƽ  <m^ G/i7^i&R5/i:MF]{O'zþѿmOx^4M>ý/P]'BֽKt3+ zD:U:#]|[j!5o Z6?)|M,nFG_j}m+?٧^-4Ųo'oG:mm`X4?X_kz>_oۿj~1|w)\Kh|}xol[Z,s;f6 ?gj?3{MsO߉U\x5е}v'#!#om=€>8n =62xrWTS+ h1szG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€ZH>zG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€ZH>zG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€ZH>zG5?,(Ϟ~/a@- |} ?hkߋXP C_=#_€6~+uG1À9#z~`({QZh=r^a}B?B|mz7Vi{`jz4:Si\=H :( ( (ol<OT}3žKG XhmΫyW7E%ȱDt z=⸓?wRn}`<„+4eZTթQV.ө86\uj*Ւn`+Zm+Knt=;IFX3˃^GTduwntvi_EIm7l9Hqi/Eϊ?ԍoQ4 5vi0%b {K=jx"P͚~CU'T8?kO̽NQ꣊^V^l v3GQyWϻiVֺv߲w+h+R}oç8c9##ikeguUkZ//IkMnw9SĶ:Nޥ⫭N8lѴSQ$zTJ6_qnwB,> !Y֓kap,eN>%Υ:O`@Q(YUQrsZG݋S#.W!ll>nqv>nTD/W{5ura{jznYs2]^O"kkI$J葬lX0xLV?jv2(a)εzTTQFeVZ%t2ɤԥ'%E^Mm_Wv;.lSZ%xOKլ/OǑw;Z1Ӭxa p+q 8QckYeyⰕ`s"jUJ,F'(Κj᪾JJT+F[TO.3ªJqH+$֪A5ktmm䝛>kW6٥f쬝Ynݠ8#d?J<u>4T[I??^wz f<=_b2KŪΚܓ;iPr&1 ADzU}tªn*Tl:Ui^ﴄyvnɦu=>^8H= :Y-תkfݺhm` ( 0g'ٳ ^@T01#s61^(鎘?~i~>lt-bP&--5"v5ݥ>Z*,k-C,Olu cۦ{?֍AЬU6xkV KŚ~3yjmIRm̓a<5] }ƕz@C/ ޗ|K-o-1xWƺoZQviWx$[|W; x[BvsxvizWoڶZjma}S~-̶Ԯ=V:.sVRO #Yem?|=x7]S:ņOwdkY7:5|M mKm#i>H׮)u'=S–Ug.ly<{!ֱ`Cǿ#{c<|g8xxD bk^ռi6Hе%t+gWh؋mɎt 1: (8u# q ( ( ( ( ( ( ;zg=Lk=7>%1<3OU¯-ׁ4oGMcQ_/5M:IҴ-:_I⛫]\6>^3נ9FpA6P@P@_/O_.R/>=Ӵ>[Bºo5ԷqČ$ƣs~,^G~*L"zPFgRj\aJ8SrRRQ5)O)scݷBJY=R/čNмi3ƽ ON1ƛxu ەunߟgXω+T$V&4+H`xX /' B )IPBW LcNcNմ;΂^Z:Sr篾ߓi <:w!'5ԼO|j ?>  R=~Y>2%to,YYlvW0ϩY 6eBX]3jf̋(+<ϱeVu1Xn+Oi"k Jg(}^%5BX+KFfJ %Ώei?mx>nLּSww6a^$? ƚ/55)mCZeݼ/? Ar%kθ1CsoxVc ~#鏦j o=&n 丟ii3V91,64aHfip<G2}gZiRXYL'|t)%hR l5U?aһ Ķ_|_A_x熵{k>/Xn~6 4_x'I,dk"Þd.+1]ēx<7'L¹<):M<^QنSK;g:l1s.\kQ"g)U\C" sOQjpBJxH? b5+cXkue[YO0MY]WF=텗,5(;c_c&֛m.gxx50t8BX."WpCúI?K?mziOZ躄־/5xMeivlp[GaicYF5>1,Uib(exlr*eisڦcW<(̲ 3.wcGb$)ynxeQN`׶s(TbsE,t -KcúWi?uxK^w_ č/gA.@C>oqmqwxO>gXyce٦g,E.Σ9[YkW=BxF;O8kRG'F>ң>hzNg%OWk#6bߋ׿5iuo[Iήd Kn$OԲӣXSs6&8/.bJ<+<<!y< Pa\dLF!bTs*8>~ISʼ{ʲBҤ(QRRmEsF*rGEh/|Yi׿ƽŢ!:ݧπ'տ2,5^,Okwpˬ^9:e8%x9V; e^>3lI=df[YL'W! ?ٸJp:YLJʞ!a(Ԅ)S'RUKԄjVf8ԻG㯉_\So|]?Z ]ǀ.i0|? ,m%4M׮!VN$௫K.ʨ|O=xc&Vp')3n(Ȫe|el 4˕, F/<)3lMi*RBԫj휭hƧ5k˚_M|F?>+_-BݻcwO.t=?A$-tWF5"s0$2#`+pLH}oKrF3B0*KaTeu'KJ9OazrKI 1Қ+7Ik]| ĸ |4+~?ZoO|/zůմ^ռ+\xVqZIem65;Kٲ6I<&ayv+9w(U]yeG%իW"285xZUk~uXAļJUԩ5b]:r'roмi^<k JLjt_k_;Uv<5xH5_./jw o5XE+r ?y <WO6$>#%?#׆*8|3@ȴχx~H0Yh/:h<;K[Q;g:/f}qȸC-lT?rGϨ T˥[5>/|EL^' ˉӧPQu ERMr9)FJJ5^)xsHhh/5_=޷~|nLJ>kMCA//#E{N36YJ|S`8'¸q^ƜC0JslDe**ict$83N.x'SU/eVqSxzPTIӧ8Ԍ\]5$[o/4GW׾'?=׼Oq3ᇅ  ^^i ^ķouiw3rX5xi,29Sϰym<-&AӾ}⽯_k: Ӯg4]b0O  p_e.;jg.p>2\M)eɰuH(q8D1XX <9]g5)ahaP;Ny/.jIIE΍O_|Amv6wzxᗏ N–:մ;Xι|k[ms? 2Gb!~p'C/Oyy֮?Yaq%**}C1 FI⸲,>"jWRZ)EIڬji?߇ OxVŭU/<+/:~'-cj!蚏|]~9xPӥg[{y`ôWNtsLpYo `j Fi䴥P`hf4z8,&[*xHtܕg:.XjN)ʼ=*r;.wө~?ŝ7;S6*>$м45Zljm5H9Z\Zөf-Fı]j7?&O ^,w3khc~i;c[xBkcQ6fK8/ɰx\Tp6G֫7-YY 2sǼcK0gQkK bV'RsJ*<+RUܩjPC10=ƚ%NӶm-ֶ[뮾m5LJ( ( /UI?{W$W1WOҀҀ QlǶ?(}(6lt>t?LP*:N(6lO@1å(}(@O@OTOҀ@t:Pc1@J:J(|AO)Tؖ\_fMc>?e^ 5ϬxCּ;8<9q uG'[!D{k7Yچ\>*}~'OZ7ZwxwǺ=7-I>k6ӢxFt2H6麏-sΟ{cܯk;+_àAmGҚK46O [Yizum`ǠxoYú?sE/iJ5; {wI24jf6VI_KxOoO|Bm5;4 CXgw6X gw |_?/=|/x?7?^xn+= h^{j=ռ9/7 eҤo|@-kxqz[r/`z7?{?ֶ~.|1ꚦ|cyĺk KciLVvn !?Ckǿ 4 Ɵ_jյ?z>zv "|C)[g1[_bkXyGGh|}>/lj5?~'l,,~j[7^I<|{|DNqz7wnw9u<{~c>spxJ( ( ( ( ( (=A72@(tqPܑ\h'AGt8 \FP@P@PV^&/Wt[6Wnqݹӵy1]F$7\2= 6r<~3sL^Oa9 ^L.&Ntj{:qd2h⊕$Uӽf;Z$J8 `p+`۷G+-#{O,w:5ukcg,OIb_n[5۾p')V+TXmVwo~$ 5Jա˜f^8zu#SB֣aJgGRU熧JxTPc̣:PkNmV]ڋi+>R 08wI `?1V_eW&Օ65~WG,gwkox]]%}ew~G(OE88s29#09@%NF9oŏIx*7kh/짧+VTTI„?&s%aN>|dⷊJ"WF볲JOMl`sEifۦ?B ?`:wl/ 1`c#σ$x]c1Rz?x׼ ͷ\eX4PiGgZ%}Z/\>x@`cyJ[^%qRQMAX߉.vVN7k8enUN]mu…?&*xsܑ gE<c 5>8fRmOZlvkY. 4F'^/p.Xyz)J⥜Qe:u3oai֋ƼhƲ!RUR▫{7{%f&{ɧ*施Mm |2uwWh5M>R)WLŞgyO*15uHXիZh8>NRj R-,>*)N:Tb8 "(Nnj7t}=]`/׃ەcw015̼VIr|Uhf.EO8$=Wkt$ug(Oc.?|rx;tP=GN09_+6W$\Ahh5r…E` :y~eP@IJ?+xG+|5-_p]0t#eҍ?uKu\Z„YqXbB#[Y_. vcvTbP%oKp> o^Cdt(⋵Imqlj}n1ߧ\`pKlwZdO OW> qŽ8A$^+_46ղڻX j*KEJwYB| _7\?3 n#$WK[xV|6jOڳ+[EtaZ{:vѸ9_|o gpaoa =O(k'{>!+Y6t}Kbf䅕}މRvZU`|˜M>r6>\ riR']ӿdԭkuh%̵Oz< :*kUZyٝ|7]'Za6 h 2os-^l2Zw"s7>$RqgY h׆1q0Μ+F&JJiԭP'(&IJxl= sRNҳ)qͥj>Gmӧ~=_|%*OMUWh ( ( O6*qF ('OR(| |>o x^/d_iwvMo׶佾$nmq +X?Cf5[AMltU0kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*h? kٻ*?fⶄp3io GC|e j1kԭ4Aa“239m=c0z}?(~|9p#R9q1pr ⍃a8@Gzc=o^0y`L6@P0 # g@81🍵uOvQi<_OIgĞ|Km-Ε-f~p:q098'<4P@P@P@P@P@P@a|bu瓻VyE7lPH?qv8?2c=Gez? ,hYe({@/1@#Y??P@=_bG,=GeG_b6;0=Y6!?2ˍ< #OY z/ķ|[eڋ!1P!E`qp.1lh6 qY6_G|>?˷OcFV |G\c w`z? 4Z2˷ҏ?q gmG|>?д~COcF?~?e?Yv4[6#mo+VG|>?ѷm`w`z8m4l_?qP>? |o!?1P!FA_G|>?ˏF GGAGm| F?A?.?mG\cQ6G,=Gez? ,hYe({@/1@#Y??PRoò#&rFѺ2Ăa#lR]?!տfzxoNY}@dxX~|o#$_ȿcoY28oQ,$?/XӅ\&7 XIBY`4_ .e_e}½/[~_ ?_?f=痦(OK_YIBg]<> ?_ ?5@2OP T,|$~,4_ ȿe,:_OMwYOޯ̿b1afqQ΋ۧ[}vl~? ~c?i6 T,|$~,4_ e?cU&|t]I~?e/ُօo^o+нeN}/nma?dq˭K۷fᶝ[H1d9_uOXmרhV-$/Yp?d>&:/m4z'gukÇF?Ϸo ?_ ?5@2OPOHzae:?-~_ OxY痽?M6c' ?gƨXIBY?>пcje/c' ?gƨXIBY?~1sab1Ba?r8GyX |,/6G4nϛMl(#hɀp(yX?e( B?]2cӏ#OG0-fYu@#4=\x301}'~C,t€Geoa@ 0;iC  =2 ?/P;0=Yq({@/1@#Y??P@=_bG,=Gez? ,hYe({@/1@#Y??P@=_bG,=Gez? ,hYe({@/1@#Y??P@=_bG,=Gez? ,hYe(U 3FGzG6>{^Uƕ&ONm|߈ֶ/tÿHp}]ll$Դ5@:P@P@P@ 'o {<5z\7==uꚇAOoď E\#hL/[K_TX@45:uޏFgSxᦋ|&VO|Pԭ~# W|?c Ǥki[v? {x?i^$:x+f xhj -6 þ,ž<%Kq\[Zj@P@P@P@P@P@P@P@1>tړ[-^zk&|kDx[Hş¿񾏣[h~vgj x|WC[4KM牼YZf4,\}_ 3|fAEŕbq3ZV aptm<N;8h`hKc)®&IrIZ2*S(Qq'vV-~2_AX>*MGE|G7ljSY]xLZ5Cok_fԯ;:{l}V6iu7{c:r k*fyUC3a*b\Eap:)WOV\СNQOE:ftqԧFu'F4'(N1_(˚pi9ޟQI~2T;^׃|KnRԿ ,ޝ {}[KKM2x/n4-NI` Ĵ3\' *yvkx,V:uhs<i|=l]:j8&#X|M 5X*s8Œy QMrJ*n-FM5,m>/|=s׶%C1[+ ][oXHҴՙ$u;+HIA2 Y_SQ#O bqԕ \nat . _VЧ#J#B$ڧ_{(;{vzɨYj~/񾉦%>K [GVhOYC-z͗mد]Qx<eMZG$13T7"XaU%_ 8%T*X Ft<, Nx(Sa;K7diV[ZT8{OMGocxG@ӵ[oY#k2/P}Ok gGWUZy&gX֥?e+xsI$=uJĆw]Jض~]xkV}wCWz/xT|L-JS_aP;l֑ ?W8ij\??p ?Yb({*?YH*:ΜM TU=ҍgJsRqwu {\u}gW=_zm5=0|87ٺ׍5ka$2kxJ܍O֮|1⿶YC9=ڕa8[#}"3jPeug};įoZ~k4RCc?++Oۿc3.͵c&x~ugGʎ[*̳"x\({W'տs+ѩɈoQadҊNdݧSi7]Z|OxRSׯMιA7ٮ>0 yVpK"]"^l,~vYc|E\&]Eb10t)SNeVp,&""n_g¥iBu!I^r劔b(+OyJ+wey#v?Z򇅼oKY>?g\m?3Kw|˝,7SX6aeK?q>E3%*rb*WzQ売9!|i[ğSUP׾hv |oxƚOßk'V^ Cu*-{uۋ(h#S F+2vm,2ɣO'Tcp6kנx3jo G g\u;; *{rcev.tӅ*0zӧpzuq8f&. F+VIǢs8h{&K$QSrm(<~Umk:ψ#j> >>~Zx\ׅWAχRº/)c0q\$#^[ 0`pXc:8(rBt,dR1kv^啋%x~#\]j/˫IWS4{]G> |q2P4b=6}{NcWGMk%C#SɩQUXWnWGٕ\5N؜TV"&!J>+n}'?,|M?J\,`Ů0X *7Wa^K6_PS1}fOc_PvM>6xf$C OT_oCă(GVO K *Py_\acUoϋ=͖;K$i(`gn_Z$F_WMjMmuw™XXjom\>>wzn隽|!wWզ˥e \|l7q_ ]lXG+үWMd)`+pK3ե=mUv+TQpJN/uϙGTTv bxk-{YuD ZsE]ZxYaJOI'k"(eDiթfx 5*|;̳n/*r*teTK 5K 8J}B*Uj4%N\ҚN $ uZ|iZ׎<'F'햋u/|Y㆓Tvm|'Y9msP֮#M?Ld^fkxƭvAfxYܾ>,~]jמaC< G Xc?aF碩ƥjӤ'>KBsJ9&mt;bxCǃmOZfzO@kj:mŵ|E?ll.o; 5-/ϣp~wa)㱫άh_W(,D0غ\S BsBUJ7yJ*p VSn>=4]6WVk_j$>i/ڗ,4t˛SG/u9X '{kK&$NGî*\Z l-*1Odx^ha qհ8aqU*`#O ^P>AF3NrR**rIEp`*iJ?t/w3_y8X[>xz4_T񟂼<ڦ5Sut/_k,w]+ZZc  IFyWb8.2ֳ<,9aqXbZѥ(b17ʰI+NS.XŤӺi=VS#i#=GA=ѷ`P@<{'[~O@=xDg|/c_u-;'u}ծm/]B-3,kc׀ㇾt67K (ѼZ:\ZYiZ6jZ-|l8(8Ƿ|g~:gH$^ vB{KjTk>Mh^x=>!i$n.a7^zvf)GSD$+1ⓩa5 oA )H SWl)xϲko%?ijXC>B枿da۟& YgAivC|c/^Ff|8,ӵ *ҵxus`ej0?h߃7Qо#?<hό>+|1uz\~ K~?-ޣ.{G^;4ΤAixh ( (OhW#խAwfE'k{hڮm6VD+-]x_<_xwI>0Zj{m 3ggq/S_NnE^t+xwOrq:}CMg0־)|1}|u]x~ ?&|6E^O_#MCHy%\! wڿg}?v ˞8^s<-X3O<:YQ͡`\)A|;4kۉV/'$7_=ʨoCͱOIFaW(< F3/^^RRcGnXlG;oeiJ55FPmvO |^ߏk ~ep/|<&?h{4O%櫭4v a5>~yx_z]~(pC9>qE6q~ y%J z p&Zar'br~JVJx{:q!ZJ-'Rwo\ )jc??j+[ G&vO&WQC6N 6w.Y6WS!rbx`OeRpԥFnuQrҔ;|UEM>)x__k|pulv)/k.Dkmh FfӮl~2"጗YxW9.88~m~'p!l/e$OB]yѭ:QIolMTY[FNdo |3kro oOF=lD7"O6Vwmfk8L_jqni<5Ƹ #O U4h8ׂ\Y*ŸNkPE_ }'rM?}_  Ts7Õ~<#.pG;Z/CS92&̾ Crz5U]pxiK3tTt' r JXlGlpISVV')TK +uZnn!ZMKJ6i{>*x[,x Go Ys٣6 4XVV=qgc-Z)cy?9߃ \ ̱tuO4f،V8hyeUhԫO_^TB%U)Kk7VK:i{/vE:&}Þ)|tg?ڦxM/ 7vƎuojA;;/k:R-B_  !?C)b(Y~6'~.ѭ7ŭOU?on_[լIHЄˢK&ɪ̺Ga2J-ɲ|+I$?|_>3|$ mxWÚ)6Qg7zσKN^FC#.ٯ8\%/ f\eXocqKpskEU;FaK [{?oN0|!>>jěoR'EҬ|'ja}Ę^\o/xv38k8WS81qa2JY"KFU-*Y3;1n5iS)KRi}Qr]JNNRsTMƗ%#8y ''gwe ᢿm N Ej>΢r'|o~`Am ¼rw 3_go 4m_n??y_8aJS)Y¿GոCKX坾7ҶJXg|՞I{J?]b&Vߵ>*u^ ǯu)/<+u*,[g ->[us} 3 5:9V;a0]as `>֭JuRcqيRj>ue__mv_,o{Cg|^Mᧈl=j+fE"m'wԭc|[ x7wڧzmƛ_޻6t_SstDwI9m9m/xs,="βlcҞ;.+BQ0 ,'2~O:Y⣎S QVXV~jP)Fb7NNJqM:J< IJ\KlaqcR>*RhM[FO^:MM-Rßiմ[*k!o[>\m NcY p\R?pتY_B8F]xJXJNI֩'ty1tyFufUyZu,gEJJrWjx} Y.zu߇nͤZx7Ě.h60Gd"]: ZH] 9!cYJ/G3gWĥZYbUyԆ*gRƤ#t8(A+F>T얉rIE%.[^AP@P@?|x? h?Oįx"ֿi)&Zv_T~ +fij>02MF~|>N}ς| u{.kh/xx-._d iV^ վj{$}>Kxr[_ڤ76$< .&n-$kiZW|;ZWMA<$͢xa mI[{hdI.Oe-MA_S^ {=Cj?%Msu68;i LiRIeԮd Z M%&Mjj_m~4o xǷ6]GcNÏVX^⛿i>!|],Q:xmlJlOz ( ( ( ( ( ? ( ( i\LzG +{c_(bA QlP@P@P@P@P@P@P@PH9oo+[\>։/w 8SqthMm_wqyzzV:cn?._n彽mݭk}}ͧb҅zYjEN f_^#9qUflvM-k~_l*\zg usMhol?N~'nMoZI_נd ug2Wjm:]SL:_#=83JK_M|;<:q4VathM|{c( A=tom MyX6` et}iY^u-<]6TSZy|?l?vto ZyX0om 6 41Aҥ~[[H6.~ }I3U-?maq|o+?Ji`N(>m 8$uljLZ炼tm}oA /&t_hw~[-S[,,Ԭү>q]l+)e<~{Ȳ"f%Uep.:`lF+NK ֭PIu-*_iER$8T3Z:^'JƓ⇏C/ b:;J>,l|aqaڽ鸇I5R+~ج c3\f;*?r|0\MX aa3؊|cU˰j'UcjKƔaX*TjYϚh8J"bt暂rKwmWu_.o=^߀o3oo.Rkm\կ {B<;h:n.ܙ{< `ٮ+?3,,&"پ+^kxO8:)A25Fp乹/RIV8N =7|E'x<hmOS. O5 =&wl]%ٵC[:OE`d14Sɩf<1U?s_1y N1~-\bS(Ї=YFnSp9s'~i_\i|]ZA.qxgI/Ỻ[綳!I`d,jKpI\$P[w(Ѕ|v&N*T(N]W U'+k]5F8Jm-ZV[+۽>񽇀#x7¾A_7 6LVĐ<>~.}ᆱU5 ؽj^mÙ Opypyf?ael+xlL:V T&~ׯ,&PW{(Pp&4 8ۚlOz'>#\GE#36[<5{8|S-K᥼zuPoMzsmf|(fyJ3l6e_ kMf}I`2zrTzُ Sn"*(Yv lJ|!R 5$Q-²aʣ^Q uNz swW6I|K/kZ~)>owֺi*$g; 00)q|)#ØKpIfxeN8T1A¤`ڝlEGѦT:9rRJ^2\Yy6iڼFxgN>GNU#CUAX4u|(fKwa黕f,>ʎ{vUC>O =|gԽNi|~'/nxrl_֡<>Ňb0P8c'^ڻsNԽ/_O>N۽;l?tk~#c@h|zP{3ixKWZK)a['f1^ qY#Cq5̶*+x+aXtq8\TqaJ7M0%V+&%Sj\X^M8v|[~z?|GM!#ڗ|M]]ĺnle5ݫi[Ia%3QuÛ<`835~rgYi<~eŞ}W$x5 fG,}XbVt,VTqxUJp)Ԃhb9R3SMZ r&O\|ai/ |:m׉lg:H {N ~]^\i:Lv_u&K+V,ўoøN4 Q׫Z_b?lX\,)R/*eXu^Ú-5 J ^*zYoxş</ß47Ěo/VjM7P]ë8Ҿs4i1hEɸS=s'js p40Ȱ =] RƼsP hOΥUR* u)ӥ~T;Ҕ]+.^g]8ܯ $WMųꟵ?u2mɬ'j|y.h',-rZ-֖mu(^KEf|i^cKfLd Z9f 9DqXlFk K^#k`jH:5rS&助8SQ\1HEA K=u/LY}.5K'W,6U9t8ҩ`chNP擕,MJtaRXygREsRI%s]BqVi6gJĿ|W |G}&Au/ׂh?B1;\#~7ӵoxZx֚2jʸ&s.# .5:kNMOүp¯S[U*,K6RX\F:J=d*qXVgG8ɬ?5&n_mN (ko궑CkZjjz^jQA{o1.ikh%pOr)0XNNJ3 1TMӓԫQn-ӯVh5RIӔdش8٧kX ( ( ( ( ( ( ( ( ( ( ( ( ( ( ()]:};/7pW0y1FfuFg c8Ǯd1o:Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@ :sFJˌuJ-m꼿x|uvNjK{]-?A1Bv_~|> 4~1MRӺizoݼe:1ـlf~~@,/$ig~ 4~?'g6H<>?9Nz6 =5;1}'oma_3ƻ1;3gV]Uq G4~֝-n`@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~Hth3CsG? ge@;/?@$:4?!?9 Av_~<` t=@^#4c``c}O)ǧ=(,dj25?嚀 @Of'PY,dj3F9# c8N׀2M/:%I1ۏV:x; Of'PY,dj25?嚀 @Of'PY,dj25?嚀 @Of'PY,dj25?嚀 @Of'PY,dj25n83?cFO-gu:OL=gd'vwi-X;}63g=;g8R'Zi|#}ܷl$4j]UKtWb;zUo:pLn#wpO+^VJKw{mmڽݢhV@pI3G\eOL=J[Ϧj[Kե{.49n3LOom=v4ymUO+0Ԏ=rO {õ/-m}ӷ}j!uK[{|Adf.7g;nui5ݾw@83xm =lqs=7i;hbۺQoX=9aӫ$1fS򗮝wjWZrKwmAI9$`J:FV6v}֏fk^}:zu$A=;gqFz+Y6[dnDV&ծZg8=;(N6Rkm}d䒽կTm]]-_Qt< FON`硡tɭIkZ[]C~XsqO#\%ӕKZ^h6\0RW'9 sPk{5RvݳDҺҷ][KM>hs?7gWj)ꮖ+srb_e:okm}zO^8,pv׫uǧ[tZ읷KWPӲ뽗ɿ׌ T`%ïV]i릪Pɮo~_[Zjשu\9 d]]7k+M]>m?-i5v8:###9k˗+k٤{&5˽nI?Ɋ~\*'=;p9֕ꬮv_V&$mhi0G|GRp1>}p:Nz[F֗[YmwVZ5,DZN+ FqJtIjOg׽Uqkn^]4B~GRzKe_Ӳptm:د#8\7<bmUo_~ /k_d##ӥgmo25?嚀 @Of'PY,dj25d #~@qxx>xg/ 3o5}߀x=,iGO[kmldW`@W|)kמ)?-|'xV-5gO~:_O ZV%}B)4'NCcX n.=j ( ( ( ( ( ( ( Lpy8!Sm=fђL]wӵK4c=o> W[棪+zw?^#tohqnW\Ԥ.$χM +KkWv>x_xY&NP;qKMBQX|~pqNJ^ʩT̳guhX/biեR,v9fLnc%Z9ҥ*ۅ8^.svik(E^촧 nrW_9Mk }B/Q}qv沏F[֭|[gakp։m?Ḵ@> S W Ɯ9~Ln[tpX\ic'ՔyUZK'%%R:zQapʕJ(EUi<#Q?kDZxꯄK aXxj>ƝIv¬?tc G"E׳[J9| G3'}_7^[w/~y>qxX1p_q8JSm6ԦO3^t8[Zf9V;~VsXuc:ʌ' WmT䔨9dRt"#Uouɥo]ރxwL/ڗΙ+v^.}-S? ˎkzh; d9cSY+>ac OV]VSxX{8Ç֥O.&U2M X|+S͇R9\pSԹc$6UgJEh6l{K:L'ҖK»q/Z[UGtpT9?L\?q}*QƎ_[+ D}?(Aiza{ii`V<6m+0>d#(eJ|?RKebO c|]gr_S B/ bI䲪V,\Aƭ2UJ8\J"Խh_?$M 4-<;x6;Ö6(ּYxn/kv /ou#n.%w<^Ϗ m,*p78s[0_,8y4)e̳: E\L02*uĪIT9ޭEM)Τc%5Brvt̹TVJd8o<;?^]3R]Wմ+=w%bF/_Lu`^wpG½ ' `OҾjY7:9_/2JKyW*?R ^g],lc-JI$"i:jqM[P2~" Oƺ,-‹ɣ]QV/bݵ6u8-5y,qpfu2Rhe|cOK2W p7^H)b';<}nb:t̩$xƇL?2|]?_G~?fJNtu7_'\V7L|9uWۿY?~sJO"XG_2IβUqq}YKUrjҴjr)'hIk #-[>!࣡Mџ05)7sY/ 8?jjyFsR3,Tp|V+%Uƅx㣚9oLrJ"r%~NXI)78Fin^KKMsu|q}5T𧍮|_d%ּkF-4D]`G|u 8bxs3 έ:KO)XUae=jTIg WKNxʘ-i5^iABRR7RqN3BV-mY'V|m$Ě~uϋ|Ciϩi~$4|-yqi+:Y2[I䵂E}~##Ix[ZX)/3bҗUτ.fI[;=촯:5_2•f ʰi T}wDo fhlfc'5e>&/_5MӾ|%64Z6uf%,𥕴R[.=W:qu [_C?n4K,rm q>0:~`qG` ( (1|C?t{|)־_ %kzw.o<#$< g_:Ť+@\ CTſxX)i;YMQ)[V9)%y0-s2q/0][Z񞣨'׌|'iW{ɤI.M.|_t0,?Y@xx<9]_%| m }xXZ}B5Pi*XXּIxY' WnÚ> _ xw\pUM$W7F@P@P@P@P@P@P@P@ r<tiWdqg["T۪{5m>q$j6 Ox}ִ-wJ||:/EsúuѶX 1qsD_rn%s|Gg43_i`.p;׭>;VYaƬ(eө+PsPJ7O̟6"Wqt(E8R?ό`񭟃izi[JFsmb=Gqȿ6uo&w,K罷93 Gg,1U1x? NYO["P]|GO3F//J gUF%G+N0mɮW)5{.ed^qsB|'+K??jֿψ^(ּ>l6]n_x;\,ufys^CeU8G5lWxLYA`_V˰sMlMLҕYաiӄ!:q\VN8[x/j"b&{xJlחF@_i]Ak.kj[KHm|O_Wò &|:Y$x"3*c+u1ͼK1ؚXf]~ "p0WPe:77jj*E>|җ3ˡGM>Oi*@^AxYOƫi:l[Yei'_iFv SNoU_ eupX0m8G(ajc&+Vp9ZX|\PFxiɷ!$TTO4]SZNJh'?|]oi6NυeiV6s{$/u$>"oodY/X~)[`J48\|4o18LO7TکΟRTU03d*5#,70Xqԕ BR޾_Uܕ98JW+xo|1W– ~ ]hZ޷6ju/u{x Hg<=xtiӬfD4/MLtgY_8f]<6'¸uNxUe JԳ ~`cR3~` Tb(tøINiJ2m( )ڔJqz}'w|2<=GQϯO} 2E_Sj7xVA o>1fF> 29^o n2L6q :8 8xC))?.Id\o"hN"AjnnNU'V~j]7Z_]|Yo>c0]CDƿ5<5yitmm&KdƩ3d]Oxiˇ`xr|oQP0<aq̱y~'=Ы 6":8\6TV% ބ5%# E_ ,[[mlr^#×-~2 GPͮ}_n'"U_A+%xo͸|/`*0"KfE9:n__~R{jm{ώ::2!P{u'[N2j ie;MVx쐛XM%',&+0g52FW,Х+W+xZ~%TeU希W֫FL}Jg,ͷK) SST){DMETVQk|9]5|*xZ*!b )--7^7_ZO]_IwZ[xqbL^sO(˱]x\#)fq#JjY|X=|&+SpjO5Q{:qf&9ٸJ ZTG)Eɵg=όs% 'kPվj7>#>4e5/#-e+IL TiNlVBשҞOf?a0C`:|7|䙶>U?3 X!(?,G:5*UǩE_*8JQUeZe*WwAO;} /-ė^{φ|_[huKKY j kZ+J[i{M r[6/8ω-Fg,{Y6: ߈lS̱5jaT3FxL?ѫV'֧K8x,%h{;…gF:1zVm-Ir7,]8҃RwOX/t-5^V?QytV>7L:}k[k=/K++o!c<*e6*#+XLr<},F RqT|]QULEJٖ),ժ†:J1k&=b"J0HόCQ _q ixTxxӪ L~/W%6~ gS\apJ?ڙϣbNU^&,D1 C@^O+]3 R(><2%??j|F}:Ft_?Tkebo 5fa}Ua&`ANb6_,X|RwMRUIW9ܜ տ}QUOuW<=hZe~/8.sFv5-5kO;jkɧE!A|'p?e,O2,E|Ư ^3 K3xC Y. K%TKՇ$Z󴳼QK2 <}*{ Ctּ[~#x]U7NY [N1HՌ+Ƥ}2^Vys^.nvW{k<_ j8i:!^i߲j/1y.|2^|0ʿ?#qKiaxCxn@/yY*V^hڼ"Dʒe*Uh؄cV1RiCUxl>:?%ms׎5QxI6Qq{ω#fgYma({%[IdghͲ, -i2:mlxgzǚix_V"UO Myf ]1,E{okx̓c׾UA(w oA'(l_ nFM=So+^?Ywš_ x2O^6im.mˢWZ뻛ɼ9cOky.Ĩ`;HKyX6?{~{P@P@P@P@P@P@P@?? g<{1~lOQlP@P@P*1? ~J( ( ( ( ( ( ( (x?@ 8дi}{5wi}9A3׹ N@䴬[U{iz ی8'C9A*vVM V~ikvum&cېA!C'$CWvjP-vzlj鵪kK=zt֪6ux:r{`u=ɣh7Ek]kK*J\NW-V~&ՒjA8_zcv:md_ҳ[\%~_K$ܶ'oV<9aGZݭopp:zߦhJ+Y+$}D[E ot?y[`:~|=Fml=NhZyX6ҏtt?L66kN:bim@ ( (J6 Ql8lOFP@P@P@P@P@P@P@P@[RӿCg7P04Lpq&<KN,]A?€?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.j:1BHcMKh>d"wV"t^ևn4Cľo!'Uj]vX3j> uMGC>qi,:WyУ q/S倛aWqf^}[(/S5%^C 8助✨G.ks%J5iǖJr{59AGjMl[%*x~[Ӽ)i&O^-{_'(4$4f]B/X +JYWClNYfN)<>+ܷ`kYkoxcJA{-Au 7ҎVlj< `uIJDSi,󺔱\apٮhf1VCK8Sfae4r~*K0Ԇ2?RGNPTԢ(4&TIb\%3%;Ϗ]\izivײw_P]Aos Zsor$3@ºnθ+&q'fkQWbP)֥C: R:PsR)ArMYѥ_M{9}d$-$6Mktg :v/)֯xU|r| ?5KkxK>Ke|ڪޕȲ[Do>8YO8<:'a2Z+GoT'C7MLMFO XelUz5bfL=jtVzNJUI(:nt鴜+I' '/|IC`׍oKlB|xb(=ޫ)$W-ZZ|N>-W|&8fW:1s>iUψSKgUN5Pԅ)yqki2Q%NcEJ ԩlM;9F .StaOw|} Yhv[xK{յmO V2]j%Ԓ{o jw)ˊ+Yj8eZ!N%T,Ja#:di0>#/T38PT'̾,4"sdRTW[Jqc%J|aGN?::k?Jsۥt{i}ׯ+ӵ~ZkZO?*.?@?@?@?@?@?ˏz~-|GOgPF ?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.?*.=!CiP7)  ( ( ( ( ( ( ( ( ( ( (P@Oo} iOi_rtSeKcO蔗K_Ez_o,;PL/WOoM)mO"é dxRV^C跬/,U@MTs<_riDtZP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@PKjmmPK.A&OEBPS/img/obiee_impmeta_datasource.jpgJFIFxxC    $.' ",#(7),01444'9=82<.342C  2!!222222222222222222222222222222222222222222222222220" }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz w!1AQaq"2B #3Rbr $4%&'()*56789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz ?[VH9$!I\k'[O'h~?kJe Wi丈JL3yԠ:oMm?焟?[O'k̻d#)үU];*8$zr=iki<$ ?\7R[yDK%ܓ^4 Y~n^Yki<$ ?\MJFuw}zTmE,ESrN@& :Mm?焟?[O'kt/ ?!/ w=Eu  8nsG8@u ?G&O/{i}X*NO`$~g<,c-::ϦA84Yki<$ ?\EZDsLyp@n9<rMxI~#?5߈ƹG5 >@LA2p;5='7[g6Օd}hxI~#?5߈ƹ/#]f7 3s{\Q\781x[O'hxI~#5==분Ȳс#$Yt!2ItKEcbm|QdM G4ki<$rO%0P6g`cQ*xy )-T$ڋ :oMm?焟?[O'kD U0pzz,f@#vqEw&O G5i [әY@p@v&w@]Xo ~Yki<$ ?\z\4ıViG֥JPfecc@u ?G&O$uܥWb#u[2 %X288#" ;Mm?焟?[O'kki<$ ?\EX[O'hxI~#(v ?G&OExI~#?5߈Ƹ(&O G5QE5߈ƏMm?焟?",o G4ki<$qQ`;Mm?焟?[O'kki<$ ?\EX[O'hxI~#(v ?G&OExI~#?5߈Ƹ(&O G5QE5߈ƏMm?焟?",o G4ki<$qQ`;Mm?焟?[O'kki<$ ?\EX[O'hxI~#(v ?G&OExI~#?5߈Ƹ(&O G5QE5߈ƏMm?焟?",o G4ki<$qQ`;Mm?焟?[O'kki<$ ?\EX[O'hxI~#(v ?G&OExI~#?5߈Ƹ(&O틿_+k.oΓ ?ԿS8a:=Բ]F z ZY)#K/l]R\_xQ}1|4Kw<8lr63O_xMSPD_0OěJJviSt򥾙y3dXՊ6HCGj틿_+u?j>8мG>[jaݷGrrXO2:틿_+_U6qmc):AҵG;JM2keY~ʳ8/,;~Yw5|?d%Z?UE! wk״HcoְP7"6}ۃv9ֵIm&R8 `Jw!*w"y^Wy{lϮ7u 2V5#'e8&2NJ@9tZ6#HA1?ʏC Th=}k,&!F`J?Ak8;`!Q?ʝm5hWc@Bw9\gQXcLul$~A?eFeYt>9_=ذrG3Ԋ5+[h Knᦍ(J;GoO?ʏC Th$Q[`Anp{gǿ5w~?Ah+P\4- 0Wv9sӷӵMW09VL`g=yZA?eN }fxnP(nd?"jkdz҅8@kK~?A쨺ṛ0Cr<;.`(Qbݸ ?֏[**@U]* ľlj|q}bsa\E_ùؿta_4(QW/?]w?.;U;GNrbss?9Ӹ\E_ùؿta_4(QW/?]w?.;U;GNrbss?9Ӹ\E_ùؿta_4(QW/?]w?.;U;GNrbss?9Ӹ\E_ùؿta_4(QW/?]R -/ILFPRO ) ((((((((gu:Kdԝt`oQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F K?F4oOѿMnQX{S<4o/F KVF'-Ixyh_MZś i#ЂF8dFd feQaEKQ[u/EhiZ?" +A_?V~ 2KIs"c%ϯhi_"2Aц\daRn:N4u|+ykӴ I{If >X-w[H5{VY  )?%&bׇo5yayA3Q$dJSG^)ZËVlu +;"dLƛC@99#W"tŪzۥ.w F}0jZѡE`i>*o6n-K4kl*@ c;}_LQ:&NX N{P%.Q\i5Ʀ5{ؠdFK91n1XKH \kėF*JdWePCxg49hs#rg iZu+[,+) ֔4u[K* wA!fѷ9ǮEAthQXv-/5[LKItpH6,AD<6@,5}3U?K/:ɷ9vTdAt˔V.c],U+pʒy#]thQLh hdIbCVR2#"HaEPEP\!v =toAԥ3PI"D2aH>z˴&>k\̚|0[cF1;A8II'KЀ@2@ {+q-%>Km'QS~ARI0Gc₎V"y7t Kg8@<$c=sB.&bQ6.V hp[8S&1fᵴ<W,$+)&NpNry=R_Im XÔnW7t^/b-]YspC$6q̪} 5-NN1K*H`GjӴk|aw1I#Y43Gki 1ʓ ki6ެ$su²(۳=6sO%I#(dDeVa]x8Nz= LesG Qo\4ŬﴵӮgH9 8WQ9=H=2KvqdlG~նu>F3uX/,m'>lN`d`p0N{W+iZgҮtBE5vżD nolwOX5ۻ+XS4GWb2$QE((+f:Өd1`tU \x\Dpʖn\y.<0$"c8~u0K+U rFHUܳx 'nӂxݜG9)e4 ^ ;'s70,}{~ sJFX/ԁҠT\3fq4%yCAӰsg g͎==@uәPݠmA sty}j>T^ e.ܜ=H)2ܐvqzgh$҄RJ% aNN`cӐPcl&V2ѕ!Ժ$4oт8fOV-vt0S`Fdc$~l'Ws5HZ8]w, Xv\ORi3I֠}9dyؗa `SKm)mqr(EPEPEPEPEPEPEPEPEPEPEPEPEPEPEPEPEPOs># e]To5E&#@ ;TSg* )L:>L$; <ힴE]*T-n,/ xq*N&"`\?ȣ/+rc"\?QTGsOTT"GsOW_"/xwS5mpE^qfsMnge$I9c+.d {eHDu_^E6/$k?'wo䵥@M.[ (%arH#^-i/j仝^L&vWFqiefmF %ʙDz(X /^m4'PdM7QIasl݃ 6A,y\q4S돧4vUb?1MĦr>5]S@^kͰ0D*>c2v>EXr#fEխ8wOkEuw4^teb ap0~:oA+G7 -+$F r=QEJլwr}CE:/H;cU%@w\jM-zUn-fHLDFPy^ETk6{oxg\ZIo4loq*o>@R(YFp@jgyd,{blHl{^EZIt$'r:]}fK{=t,64L+ x'Ae6Xa:Q{MKfh2m>rCdnOQR5|~,4h 1l ǹw(mQE(+.?'WB\Bzh:hE @F(epASp@#ЁO(*;KP.T)?sSQ@Q@Q@Q@Q@$1,&iqHwnlp3A>R]sQt▉3]HN};S@m>~m4y>~m5?crϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`'E'EwW瞅(пE >rϴ_rϴ_gexy_o"= Q`n +Wzm` *н8* C98_B{Cxyh_M -bh?/i_*ȱP Q D1U7%PA$GK1Np-iiӽލ=պʣݞX?ȣ/+rc"\?QTGsOTTVv UkWN_KE[i\^M`ُOE$_ںw]K=S!ѫV貰OWN_G?ƮQJ5)jkhӿ/( Jںw?t5r,R?Ə];\ ԧ_WN_W(5)jkhӿ/( Jںw?t5r,R?Ə];\ ԧ_WN_W(5)jkhӿ/( Jںw?t5r,R?Ə];\ ԧ_knuGx&U,gstjoO Yܥ4[$ȑCy$YėWI:2gʸ4deCLOΫ @3kyWۻf+dgέ\M\qs#(Tafߣcp#n³-{GëXl̗P2v @ \淦ItYdrx𷜭um#`#*8@Υ[(ٜKpNrXt8}{GOPVK)[dw-p7nx QuhP[wXr[EpT W1}kxU[卦W neuxm8ۺgn 5>c~[y 57界1O8LC 0^*g..RH%kt4O}Kh= X֪qBЪ1q=+LQ;7;N'W@X6"vs+&E/|!e&2MIʸai,G-Hޫ9;k]J{-/m&md09K_NQdPFP:}24^"&H쭬.-UQBb6015KIuXr[^Mɟq@chd$h7`9>qhժjFeTԀr_A")kI ϩEewi 9nUv?)gB ?U5 BۤKvfPI8>>"`!+?(_ܬ?ȣ/(@t7GsOTU-\AtW?VMW1v(UTCOpORk[J={ýt_F)ͦsmGA\L^R+cbu>S?5;Z-V76Z]tN|QTd8=fqiv -Ĉ0 QWJw)%{.q=Y&6*?]ȕ>\d_,caU6&F +qM.v?u2^Zkfmŭ^T[Nedq!Ԭ|Xj miJ’B\K|,A®9$GUEsv(Z~A g(xÁwq|Z-z5ygwX#33Dng*p?6YW*r0:+ĺs-AI/RVt( m!UNTUTw^TNxf[y`HFq1*8AttW7m6/!oPi/-"o&xkM0 eI_tLHd1I=ԓ %B#J\.j\ݯ9;J@m%FUC3xA /ֈa4"FU@ }QEQESv,0"6<`ώpzq>YWpeu,!ZOڴC3pz`tGЗ"?\ ]lG ]lF: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThK{qOqOP: }?#ThKYBRdgH hpvz %ȏQ/D*S~b(S~b(o }?#ThK|& <& B_OUG%?/ɿ%?/ɿ€z-(D7VQ+n 8gP܌Ψbh?/i_+GOoOo M+]8$0S~b*s]4C0 V88@u/EnVQԿ :**hi: +@_+B׿Z {/AƋԓB *m@A]6مHF3߈W\Sc&*c_sյ 4PGS#N>jһђU,iΌO]Mt*Z?]ȕv=BѭgPY]]܎Xg#!G8KI{,p`=ZImVLo;FQmHlrcz-Sm^NH9յ {rd1q "(',12sE.fUA-Cڥ@PT2<>%huT`V ;RUU~F\c#բ3 hiZUFcsivfnsǶe쯟Qy_ڛYJmThA-Cڥ@PT2f+Wwv١o\|vcp:VEнEsC\si [ӢK`$p2 12##iPX,#y/5KdINN I<6:.52++F A,>N''$`csj0Y_ [i{my̯;1DM< 6|ք7èZSZfLeE EzIo٧sqGl'>_xTK<\RG X_xo5v)ݏ.sހ&??''X~_NO(rc"\:rCq4EER?QPAE[W?S,-ݼn:jkaz|CVU`ُOE$_ںwvv Ḇ 4r8,gӊaEPEPEPEPEPEPEPEPEPEPEP\!tОo 7{taԫETQEQEQEQEQEQEQEiqE+ojcT c?T_ZC![Ui#_ [W?a?Ə+4j+_ [W?a?Ə+4j:oeG_ JmVmWh,t/ʏoe\??ѨX?_/ʹ+4iX_Pk*?_sVmWhҰoFck!Tk*?a?ƍBABk!UiX_Jm[QB쫟ҰoG5 _ [W?a?Ə+4j:oeG_ JmVmWh,t/ʏoe\??ѨX?_/ʹ+4iX_Pk*?_sVmWhҰoFck!Tk*?a?ƍBABk!UiX_Jm[QB쫟ҰoG5 _ [W?a?Ə+4j:oeG_ JmVmWh,^ghUQV^Cg4vbgdh,I`GOSiX_Jm3&Z?Q 4_ ´?a?ƀ3&Z?Q 4_ ´?a?ƀ3&Z?Q 4_ ´?a?ƀ3&Z?Q 4_ ´?a?ƀ3&Z?Rxgu'{yYDY1RUa?ƬF,k$lʲ8?PhՇu/EnVQԿ :**hi: +@_+B׿Z {/A{!k-kFsV =WN?}r/ PI3$Jp])ڹp:ӆh x$IbCKc6}U.X܀LaJK b|8쮶P(L1-NI7hTl6Zn Fb{fI21`g,͖܃VT^Y.d}bH/,q( :avڥ̬gdp@9;P QDz:Mw!\+3)ʐ0TqwmxQw]ǀq"^qAko幖 >\c!`c, 7Fl^URhڄpO$S돱vKKgӎ[索XGI2.?Cח:^[ԝBq.|ܦnq)chiHKI=םL,O\B5Kۍ~}6Cue ?DCx/8NU|Cl/r dPa gf@/vtQEf0({V׺O] sڷBz:0g]aeqyr]m,vv{-W2'sgǷʌyu=>EX!C2xɬ]Q٢-ͥf yb &C)(6[%U毧-hJh0%9a;\8z;^^ygv 3P@R#ap8 @X?dlWR|8'n{lqi+ 傜~_J&=5ʷ!sZ·$t+j:7oXmd)^ED36` V=3H[[[9ut;F@l[;<`Eև}Z\ŪIwۤY86]1? P@vz%xVCon-%K]]:7bTّWۅ;C6Ht=-[hV$v#I$(QH((t-'N]hi~\V> ɷ"L 8iT>ITͯE<9@ + ?4+^dxsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?G"qk@xsW)i_V?T0ƑDEUCWG\6Ɗu/EnVQԿGCq4EER?QPAEsGމ%E;Z4׼u'+ѱ}~"j]֟c/u{kF GJآ改fVV (aEPEPEPEPEPEPi죹Vgw+V`q@"PEPEPEP\!tОo 7{taԫETQEQEQEQEQEQEQE :6& Hw|H$`|C핢^??"ȉW?EB/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUzD⫟B/OUd^iTA<qcU`?ȣ/+rc"\ n?֏[**4׼uSJ={V!lQE ( ( ( ( ( ( ( ( ( ( ( ( o 7{=a?'Z+$mY0魡XјMãvWEC+|QgaEpPZ̗JO}@eӽfb>Nq6nd?ܩֵ7?fH/ $+yxKf \}׈5+ Q%Wxl[ʅVv&>EA;om~6\.u+9ҐLL|ժ*CY" Y2s( ,Ztk7ռE-([(!#N wk77f]9B%2h1~ uTU]4ivz& 9>p7^8R((({*M{ڼjڍV۴ik"%;ttŸ7xZ)*/ Ct 1k?7]es?bnAS-w묢 1k?7G CuQ@\OT](.r Z*ŮuP9?AS-w?bn( Ct 1k?7]eOT]?O'*Ů ZYEs?bnAS-w묢 1k?7G CuQ@\OT](.r Z*ŮuP9?AS-w?bn( Ct 1k?7]eOT]?O'*Ů ZYEs?bnAS-w묢 1k?7G CuQ@\OT](.r Z*ŮuP9?AS-w?bn( Ct 1k?7]eOT]?O'*ŮHe6$yjT+nw0Z+?׏Ԙ"aEKQ[u/E$Q?QTGsOTTi_xVꦕ {/AnB(@QEQEQEQEQEQEQEQEQEQEQEQEjoO 롮{V׺OGFJSCAq=;V/Q+U4}{}.:[xGڭ_خ{W_/hX~o=jb\yGڮ?缿٢cU}{}q=jfV/Q+U4}{}.:[xGڭ_خ{W_/hX~o=jb\yGڮ?缿٢cU}{}q=jfV/Q+U4}{}.:[xGڭ_خ{W_/hX~o=jb\yGڮ?缿٢cU}{}q=jfV/Q+U4}{}.:[xGڭ_خ{W_/hX~o=jb\yGڮ?缿٢cU}{}q=jfV/Q+U4}{}.:[xGڭ_خ{W_/hX~o=jb\yGڮ?缿٢cU}{}q=jfV/Q+U4}{}.:[xGڭ_خ{W_/hX~o=jb\yGڮ?缿٢cU|CJAAvq=1y]ُL+Ck?(_ܬ?ȣ/(C:**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hiF# 'j[7݊?"6ّ_?Ɗ( ̋xI4fE|c"\.ȇ(%| WadCKo}}MEadCKo}}MEadCKo}}MEadCKo}}MEadCKo}}MEadCKo}}MEadCKo}}MEadCKo,qg4( ) ((((((((((`c"G.?֏/>ދTPgEWh;z/QQ@_EEKw_Ə5#BQ ª$?:DދTPgEWh;z/QQ@_EEG3KދTPgEWh;z/QQ@_EEKw_Ə5/㰳X.)4u/EnVQԿX~1GR_Pn?֏[**>*>*+#SKo nx$xwI רo- O26Tp{P%_ uHyM4!' i;gˑ%AcP*~.U煴{l=*zhP\FP\L rd<$ o-/p1ߩ-ivcFNw2BH;dUTO'j!@l)IA6 zJ4ZKm5Y wdrBu8"'HO6DQE 0;\WCusFuxw~a9?>")Xv9 n~tK0v17Iˆ/K5 qhqZM6Jŕf$uR75w6)&
D\.Z I$# 31,c1@aEKQ[u/E :r+?(_ \AV$&ymWL`(c}ꓴ~ ʻv 9$~tgWWw%RP_ S\?Ui}LU2k[Kx$yԻr*I-(kk_-OpTkk_-OpU~[[H<T9n"y1hU';P?mk.*mk.*&O*C mk.*mk.*&O*C mk.*mk.*&O*C {Z[tkѴɴ9..P> 6%FN2Xs*,[J iCpl2[4!8# J*6G<,2(t.ee# E?~dW~d?߹?EU?߹?>&O*-QU~&O*C TU_C ~dW~d?߹?EU?߹?>&O*-QU~&O*C TU_C ~dW~d?߹?EU?߹?>&O*-QU~&O*C TU_C ~dW~d?߹?EU?߹?>&O*-QU~&O*C TU_C ~dW~d?߹?EU?߹?>&O*-QU~&O*C ڶ.m?q4J%݀qt S\?Ui}LU`?'@ S\?U S\?Ui}LU`?'@ S\?U S\?Ui}LU`?'@ S\?U S\?Ui}LU`?'@ S\?U S\?Ui}LU`?'@:qk4Nb2gPdcգ+ö֏56iA/n2H8Ms2T}LUfkk_-OpTkk_-OpUs2T}LUfkk_-OpTkk_-OpUs2T}LUfkk_-OpTkk_-OpUs2T}LUfkk_-OpTkk_-OpUs2T}LUfkk_-OpTkk_-OpUs2T}LUbQt䴆gv7hG sԊue="kYhFpI?LU`?'@ S\?U S\?Ui}LU`?'@ S\?U S\?Ui}LU`?'@ S\?U S\?Ui}LU`?'@ S\?U S\?Ui}LU`?'@ S\?U S\?Ui}LU`?'@<;ow j2Z4: QU$us]Jn9*Zq27s2T}LUfkk_-OpTkk_-OpUs2T}LUfkk_-OpTkk_-OpUs2T}LUfkk_-OpTkk_-OpUs2T}LUfkk_-OpTkk_-OpUs2T}LUfkk_-OpUCZ[tkѴɴ9~&O*C Uu/Ei}LUgx/^ơ~KIg9<ψ_+_׌K%EKl:Ǻ$ghiQ^\D.܌ d=d/淅Xp<䔮C Rn;ǽX;z/@][*xt3PHn摮Zy- :keOp>uKk5xfDӮpf~cc9,z5m>Z|A)%Sx p*=xfkqN"xѕA<(kkejך5lw630_w+RKkswat.rsys ۲gp0%}J[:-P;Sz*O=.c1O)XʄUV\0xPj/-Eor2VE(  i"U0r$G*J^}OP{gy;cvs} vZBz i s<8{TF}&owz-#{1,p@p`&K}iKXkqnἫx~N0UB -PmK~IyM>MAg\bQH<ۥkEb BjYtx=I$RMpWG-дzyI&Q,`VAqf4M2 F,ac:}́I$VvU(W,KbxgJOOKV%B7n9+П~CҤ|c4=ŏXϑcg~A@&[ 7Qm4FD*I=O\6J-Hʌ ;H#9ϧV؉^7-ݢmt%r9L῵{k-.m♑Ѱr2 @'AuZڨt[M.OyˍK` cU4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWh;z/@QR_EEKw_Ə4/>ދTTgEWjC8Km} #:c;ܞ~LG7S4F7'=y^jt${`|ɛ[3L=8 ǁґO) :p2|I 2[; o,_h?Ȣv(`X!ߑG,_h?Ȣ, ?(QE?b{A~E|C="(,_h?ȣ/QE(|C="X!ߑEX/Q ?(?b{A~EQ`X!ߑG,_h?Ȣ, ?(QE?b{A~E|C="(,_h?ȣ/QE(|C="X!ߑEX/Q ?(?b{A~EQ`X!ߑG,_h?Ȣ, ?(QE?b{A~E|C="(,_h?ȣ/QE(|C="X!ߑEX/Q ?(?b{A~EQ`X!ߑG,_h?Ȣ, ?(QE?b{A~E|C="(,_h?ȣ/QE(|C="X!ߑEX/Q ?(?b{A~EQ`X!ߑG,_h?Ȣ, ?(QE?b{A~E|C="(,_h?ȣ/QE(|C="X!ߑEX/Q ?(?b{A~EQ`X!ߑG,_h?Ȣ, ?(QE?b{A~E|C="(,_h?ȣ/QE(|C="X!ߑEX/Q ?(?b{A~EQ`X!ߑUsj63Y C*`#EXPK5ٛPK.AOEBPS/img/spgateway_index_2.jpgJFIFLEAD Technologies Inc. V1.01  }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyzw!1AQaq"2B #3Rbr $4%&'()*56789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz ?̛~|<9֙C*oT^+i$Yysr_^jXwpʐAJN0 dtvRmGo෎KY7$"Ewg6'/#|MXGmrah_(!e !Nj tڣB{HRyڄ}Pcѥo/KuoVG/4ÎSA%#8I KZ摎4Pri۞r_ka.^i)Fyq\(qqӊ9P#qןJ9P`c~)8}X<1z*X2#sҎT+~B=9b#}x>T>H8.;~r=:zqG*$<;:G*,Zp'|/kF8\`AByv_nzQʐr}_74{,WByvʇ|AEoŽ_Qr\cǡON(CGa^}(BK@tor%`Hq|9aO4J9RX /`HO4P#ob*$m} 9}E |Bq=8?iy -JF8qғAȗڵ#Ǯ0i!<;/=(H9b/ۚ\+!<;׌SC䍾 G/aoQ.1㧧rG#>rr} H:Rq9Vyc8 >Td'Ge,Wq{sKr}'G`||q\v(,- %8ztT>H0yvuҎT.X/)JN6"_j4pʃ̄H _n;nirXH^1O6>߅兾!DCQʇ4ÎQʅ%#8I KZ摎4Pri۞r_ka.^i)Fyq\(qqӊ9P#qןJ9P`c~)8}X<1z*X2#sҎT+~B=9b#}x>T>H8.;~r=:zqG*$<;:G*,Zp'|/kF8\`AByv_nzQʐr}_74{,WBPrd+<4`qxoM>ÕP˰*ч^6+x7= xbxdM_F0P/%2ǙmsMȣsW?<6_.V:0{e+|b#+^4=3⧋ {zUZtrku?jw= BXn >C p?6]K&V|oϐ5>?u :e+ aγE˧mg$ymaiȇe]ڎzyQ]4#<j_ >9{KFῌb>MNMf i0ơ'F-v y,@!RxS~/ Y֬u+GE齾^ogmHҾ_w5@EgDŽg[~.u/j6|, S][]ozao 2,z=z|$~:ju|we _ui^<7n,|ce'ýOT#h<':xִ^{hS3h\[]y/ g㦩[v^ Pvψ3x6R|;M>s]kH&] /| 75|Zk_-g\vj>[sh$uڇh6 o]3BeGYῃZ|);گ/~Xhl~ uZ++y1wq-0]vJ+t5K<a{Qڿ.4xr_i|>|_ռ/{[];ZEX𶃮OŚlﴋtRt;_@/owbj}'f7~Ѻ扨i6knmRLCnLյmLK?jO$j-ϖ;ZV_`鿴G|=!k ?^}_߃~$¥g5yU~Ti54)t;gJԵ_[C-n+5Ho7ai~UFuv|ƛ.wKw_|;_{I|4Zhz{1]+ v3VZIA~%97;G!BNlajehԥ5)rYtZt[#PG9簝I,1ς#=ݱdG}+h秲_sȫKi/2O1åm,W -oQ\V]Xk{ 1OY߇/km!(([<ƅ6/oO0~^c[|ZO5Խ]+j/ߗOnzvҏie.Ku[qb;앿ࢗV/?LS{+z%a71KtHJ ?j5=;cM'<i.aǵ/hE"Jڋ)D-?!ۮޡ7tYl 1<~%o(Ջ淰ށnyqR/ڵCoaNhSka1?j-˥XzqK5}Ȓt=q~.O|뷨yl~([+~_O5G)s[ )ub0s?igr[c{~cK䄠oP[cӶ?迯ry*JǞjuMyu?ľ{Um?P [Z#|{CӴ{F_%񎹥\Ht[+ScGfHPel5gkX_Ě,1[k>luK{> #N~]UX:<)^mGőbM{afBQc̾7A-ֵ?w i {/ ~}k4ƛG?YX-r|>[EdsqV:|=~|V,|o9=ƈuo ?^x^oxfQxoxakYiVf_<樾 I5aY}Pψ-cGD>4ANxx7~"r-MzNҴV5=5ֿ5E`^Vt90x<{jυ^*M[ 3㏃QeeυzO<x6:x*AŦxFqh-o6S>*|6:m=MƗ-}7YE,4Nλ6O)4#QnZ,5O-5_ (ӞɢVl'Xno(/h鰚}cgc7k`c?5&{?7߆(~&M{uiVпz5Դ4}f#|<_Z 5oZY/Oh߄g?i g]oV׼[/cJ}cN!_Zӟ.϶òKgo|k^%;ZjJ#_xeY/W[⯊~$x?~.i_" WG :՟_k6~! '~S2h]\h_#zLw!ޫ=G^(o>(k<=m↟_^ -Ju /^0֋x{ICxcL6g&}㏇텿/ MkOk5\]v=c>wEq-z;|<߅i_Q+?-[t]/PɵEf+l|7a.cv&I`kWQ<<nT{I{{IfԭA'MM˪k.d|mGU*+t}L+ZT~=~ +H,T) //??p0H pF=A1Q[f/'觷i] +l`O9y<~5om[y|tSlRROnhlcz<<}6vIf1'$gWBK:~;u<H  %d vϥV+KgQ{t9"At͌y$H'Ԟvtw61GY prqe\ vg gӶ(t`<'&xb_xzO knxzM#M}Z>.5[IkskoPյ[DE!S2Wj>Bx_^Fcwޗ{>O7Im൴]\KDd2yt̗071ItH_6!yS 4WTXucOj?!/o۞?TIk ~H/ 4zEΡ.iD,[>#XRh3`|uqp5\,\5ai9;[f^o cd6+nWھi@/@Kڷ?a G]M3+?~?c9K*=[/@?cۣ <S3 qLנ玜qMd$7xTIǧ?f('|ǧ?al"TtO(|SV? <+Xzs7>۷Δ5O~Ɗɇ3wZz_Oδ5?ŏ?7xS[=>5qMKOPŊ;_T7ͭ?_?cAon:RZj/l(GEOK¿n:Pv$Xb~"TGW/VK_n]=,mOtҏO_Kj~"M(|SV? <+hzqa /*z`Δ5-QcG n1TzQO_ر 15KL\=lZ_Oڇ,Wl"omi*l'3=/Z_M_ʟb ?faJE[Iŏ?fvqT1s/?_??*z_OGQ#KcA [oOK|ݿtioG%?Ɋ?fqOK韇Zk! !z_=?7ҏcŏ?cjOε/q"ر_l'3%OK|*l"T?u_ڇ,!Q7x1TMEH/UE1-[ .7ͥ/cƍG➗S?ti~_K!s~"=>)cGVK,W?7Ss j?j/hj_"T?7o(Zj'O/V?XC?`n=3 kGSjذĿ/*z_OGVWXC?a?D?xOZ?֚}V?=l/3wcOs??XA/,f?icO۷Δ֚a?XĿ?᛼D?|>n1sM_XcG1, OL0s jC1$"T?76T{av!S7S=?tO_ر?fS7oi>*OPŊ;o=3 iUUT+XC?aD?c:*zs7=/:Kio_ڋ?w=,zžom(Zj'O/V?X'쿸 nn?ira/@Ss j/UA䱏=? q OKv҅5[ <+hzqa ?x䟶@:iOPŊO?fq?ui@WXCo=/?t o/Vѷ"?Zj*l EEOKv֟WjرŸٻ`g4Koy/VЏ=?3w?~oJ?֚ 4oOf.3OKgIq%U[%OPŏ=? f Zz_֚Oy/V&/*zXǯ:C⚱]jر_ln|R~\MOʿb CZz_OJU{a-oy/Vر_l"T?ꟷohZj?a/*zX>n:Qh?b!#J1TO(Zjz~_?a3oJ?֚9Ə?a !zX?7QsJ?֚dXb3?|/=E1O /*zXS?to{ڏ?ln=,m:_ڇ,!%|xΔ.*?`oTOho{ڂb!Of?i|>n:и k?cAOÿ᛼G*zPZRZ+/ʒ䱏=? /*z^SҒ⚿ %DZzfƔ֚ ,!7S?tMX&_?b!/*zX?n:Q_ ?+b&#?i|>n:G%?15OLoxօ5/=C+bfS76֊ Pŏ?CjKi:,!n=/ Cazs7SsM_XcA/0ٻC?t?aJ?֚a?yY,!s7|S;¾`4MX&_?!ɉ /*Zg֗WW nqmѯ?[|8<Fvv>/aij&ӭ[WvGZ%7V+3owK%cSQ}ulXc!D (ϷLRk|sLh:$P0iome}t->ꝱv}o2)o4Jdl<jG{H,4k$ uX.]Q%Yjn c.mnK5sD7FeBGV!mba.`AϓpN3Ϩ|=*/=>&<}3ujt}{€jT6۴iٷwHMTQslizZ_F4]>VZ'M/N{KS.8,,-yn't8wP6{v`x/_GK[w8i.s/RϊtmV-g4)+;+ g,Wp]co+z< _ ſ ~!a #xGşQ_w > ﳉyp\v`#{ďGeE=6Fq W{v3k?^kt]Oյ}_VIt+K廿Ե-BR >g8wr! 3]oBαk}k:L^fmi%͜ 2R2:1R 6Ӣ8Ž2r5eBN,[|Nej>Kٺ5 ,ç\lER[FE T~]6?%?~rmIg|p-NaA0y&⻆{+ SZ𶏦Z֥AXX[mhbP커QC;n~4??˷nr0ۢVF~ebK7K-5]%D4MREqi>NC y;wFREmiveN z@4_rG qYc9=>nYtJ޿tJH?G12F2ayPpr-H>KO w@@z@A@N|>ܽ;-G? OiF.~ׯj_m|PLKE:G0=c|qg>!Y/oh`z9,d=p֞VtZy#? ټ.5kcy:G_ TN4yEAv/@- q<1׎]?#8et{ ioP-<=6 ts9+%!hU=eq.Gg#ERӢx{@Ƈ4.Č|Hgeu[=M/H]NN[;oa\6Zy1yWEv^P:!C|܍Gtzc82:n8Z* / Wgl`mlmbi.n5K{xbI$FUDMǀH{y|܌} ΈׂKA?YH>KG?/ bnhM1s{W󿳬e DɁA;]jUͥP$ lKEy%6?Gؐ099*'MZHͤw#C@-bֿI.9eKX/Qp[H#G-Fwm+y"u[XYmv/>tc{-KMn/,ݼ纅#dR Mj6i}Q!t a4o/."}İā,".Y_r+xK05'.k GZqѥpO meBpI?X<*fsj?i7϶ifW.oGZȞbnU2T-;/_a+<{KA X}4_eh=%o$l&W4ŒYwpY]MGCm<}#g:&&dtZy#ᶓZρoR1tmBmUEY}N)qwE-f"M̬m[miS?t;k\ov#uꚟl/q h͂hd@"aOZi3j -tkO\j=pEFFdZ\Y^G&~ʿ}ƺx-,Diٿ`D:IٖI%Mn/,ݼ纅#dR0/i }O[𾏦Zy_i5H4>M*[]*Eq,1 g<ȋ` p/j tk?ϩ?÷ iaM{kˤk{{MQ W9YbaJֶw`5o~%Ú~vDur\oc,Gn&̌U@I[ȯ}sᧃ.tOAi=R3o<mrl4i]R6@HvOC-KGaGٖ `W㓌rrwaD{A$zOLY" #$y K,1ן$ђ@Z* /_Gt| X#* / XZ_iMGT 38TTb->^ogK!phغwQ m-ΟG,ಡ#% -şll|MK;;Mgw7WWw֖]3s,̑h 30 ▋ek ^GqGqF0+99^^V=%o$úC,A8sqM0ۢH3 .99y$9N3ԃEbӲ ? qΙf0A=A@s ERi}= 8m28O{y[nO$r&Y?mmlΣ{e4$7, y6Je44O@|Ki%nKhڥWQm%ň8+y n'<{tZy#cZ8 }|6As?g@p|uVKtZy ?P3]/-:/-q/hV>&͎hھy/8Y]ϧ|-.Pa"e+@ֲR6cG|:VcoMu$Qo(yX&ʃnF~8m8^2Y/kz|H'?uƟ?ڿh6cSQ8EAv?m/4?~?eR۾onA8 fX$(prݿ}tC;fY12~H?@R1FV NO# O!_a&%~ϰ}~m[_tJH?Y~1wymiGL`BG3x<_r3Oj%.Jmu*+P>3^$M]C4LcxR*DaDG0=2()hAE Z±/mg$_ey?nbD827~VmzvZix/S 'W/b O7\7o;A6Ax/Uw]BIԾyYf}uo0oO6]G.r_hh?g?9 io7aE9>ᶫhZUׁ-oKO͠]wاKKϷiE܉o'압 lCVKm-_E $WMik}kk{akWpɍ.|܎~VWe/5?"kK4no|9o=@6˥ 77 sonΌ"E$Ӷޡ{tZy#bZO-/hF/ V?V?'m+o[a²][i+ Mn>j!ZitfH}İwM2 4_e ZtZyћv_z;{׭n4dtmR[\ 2 v]NAHwA/Da_r>1hF@G _KP,q>_ C^ =q}_HҮCƫu[ⳈncDI,#w()|EOOOR|\> ?u ZO| Ax?\1[[fSkdM9[@V)=m?I~ _^'ln4kP.$_r,1hR6U{n7%j#,g;RhhCPu[[&@%3\$ƆOV3uITecך~nٴe/H.,5o o,*1Ga V;t=E.ٕ@Ji#sׅ|Adt wzte}g|"~cU'2#o_^| ҵC46y횄2xU2q6WbUkyII;I? \|"񧊵K< ]ZJ}U:Wv )W*F3C)vwV13P bnyIS^(|Yj6^ įi>I^wǥkƔӬuȊ̋P>bc-o>Վֺ%e,j.䷷`[Yúğ<76NyZeyke7[\MBU![kԎ ԥx$+/+F~/|/㯍6~Z-+7gG/~v=7zŖ 4|p|eXǭ:7?<0xKkM.[֞KHbx[6x7ſu lh> ִ+^k\~<9]\i> o.n F9]'kYO(ޭx?-=C±xgx/^\hqU_R:iۮqz.%Fun \_Azes_g jWß wž- ?(|5+ ^-լ~5] ^ИuGꟴ='TwfW:χo^ω>{ o Mۆm*{T W*_!+݋"gB|M |—>[x({Xi߉i|%r-Z5w/iMo [5һHdrdʀO*A@Z}:x0|յόO:z?<=]jzfHKɬ+")?Co'CKafx^Ӽ)yr7PCѭ.eM"yf u?=ѰxZEkWѼWgEt4]:Yq$ 6*GE >9i3d_:O»]f]+ðkZ^x:ӵ{9#B7ybs soxB ս|>ƥc* :L#:jrΗ q6xzKkQy,^"/KĒjt}"l:xÿ>]m|gUaI.R;[=3Vuw;Iiq{{8^IJқ=z'@&;X|OѵkA-_Oѵy|9c3u.:=Fq2xOeC<-5a?L :xmk>4|5k&$I{qp|?a8>%Ƨu .-~i \xV FOM[@O)u#՚nb{#a~@l}_k>okjb}>)/S?"/#i?>4Mq׈O)q[}ǟI]#K;L[?Kn&ݤOx&IF6#C.?P;q(5=CO5 .LMZ}{ -esk (G`oxu/|KO.$Zq5Ɵչԡ1no6=G@F△icoU֓Q5/ֶDw;kAc̾EP\D1}Wş6ZxcŶ/-CK߅tt.Y4W[y:/a3wI%r~_ᏆuT|%/Xqi_ZmCe5ť~xX$o|vú]n6 <-kO.5tfkz iZj:=<"?Uɴ3umlRŨcwmKW?i붶WFUGO6Uρ/]M-Z]Mj_I*ޛy? rJp|C>MJk:4KMq#zvil5]>[<4ܦ$6/h7W<7~}LB6I^ۛ{keE0  GB Vn<3x'Ŀ'j>ch˽;_x^D_ˢٍe'6 tz|KiWZF zω;Xws9xRfEipl_M񧄾!|Q揤[]м/+DdOhi /5y/xn7XBv9v#9,P쒣I%Pֳ |3h/YxBKU[M->*Tkhmb)--32q%cs㵎ul}ox^)t}6_8tWZ{-[Eu7|IZgPN=…HI)|iֽm?K,4/!}:O]oE3%RY}: }? ^0<%-4ֵ;T5'Gmǂ4m|?u(LB>yX^ o@&M.@,VaZ%%Y dyX=֭/x=Ҵ:HľjP@zwZ^\ZRZ#~$J|pռ]qj }A4hr²Aqe`/#-eZuwoiOuOiw>;@ex@ ׃I/O|Gg዗R[k=SC}*">c Gaxwuj:/x_7Ү$fg%慢ivrtol$h$)rōZWY%{#p~_?>5on\iڈ:@t+1\_X ;KshGk]qz_^ ߄k׈$Q2֯%qX5<=6o2M?[XVke쎕=Y&_^%>=4V1x^J|"7[x[NMCNn5]B -mZ{˱=ӏ?Cc^6ŭnx>GÍ[QO iֽ3Zi:> SZCc&i*E<L]LJ|6 G⻵4MuDm+mask-\,v\C{d?ŏT&ԭO'ψVZ}_i t;Y j\ifA} !-lJ\ <1X|yk/|~~uPk]7Ěχ;v5+ƺΗY~ҥ./Lm(1A:YOy cOMg¿J|{icKN𽎕Dn𶜚ej@[Ecv7^b{q~?\ cúf ?iM=?Xuڪd=/Vi>-B >ȠKWGmyx^8Z_t%𽭽kAn ubSǥZ I6 4n':4xV||2ռkx$>#q)\Xp1#ĝW$;MqeZ(t[?efc]ꟾʳaf 7<{/>&xy|_w&k|8ú+v=M  zE5[ȗә  Ǥxƾ W@7]1khNRa-!ta[נEbf=NnEma?>kz.womq=ZǙjOztY ?: ?/5i_~/<;ei^vi:LjtO1ma*b帹 <,xmm)<;g߉tSZ.<=ўI}fd ?C<_o|9Y~kEM;Wvψ<-ڬWycn VF6P|C HŚ5ޡ54>.}'OST69f񗋛V/7ÿ4n>HMM-5h |9^YO5iQlOun_O E|m#58mKxw^ _C[7ǁ_8ϱw=pOǒz%c;xzwšOxgG&k%uCui`hڤ YOE>[yXs$ 9K?CQdB叒o`08Ƌ| $}K-K稜K%i"6E @?UQTg"V߁~03P_!rֱ^+y.'$hmx^+liv*O2|'5߉V{ ]vi&Nt[/ |N5ij 5O_5g÷:&= :5MYV9on w6!,yYEʗi%z>Xںm$YFMx04[ٮ<-/tKOCiZq ޗi]ݭPHcG+!cE?- o[&m֩f^f ybNEoijGBODxRP{J}wڿ5$[D!42 [;DvEyso<2,|lGyqjAҵ3Pτ!ujkdח)A8:h&x)e ^U5;Fԯi֧i;|"P|J._i _>+nuiIOW<7@O6]uX{'|m7E ėOtƃÓ+^x|]}-?X^ c$ :\MtH-_] ?>6xE_ }[Y}z mjZ~hxOa\P-ӝ4SM][YktG}ٓ^/eXuA\ƥhxt^%.#-t }8jIDZZ4XNQ̥3^>xcB~xBzE[xK|5x^ZAj֚dQ<1YGܒʑ*W&Fxs]񇉠oX^k7eg6V6iHvKDn!E9m_;5}Fŭߌ-FNw^wx]HUKm!.lt[{]CJY-5+`EI mW7 -/ *;it? ap_YVy{h#- Ooqv ) k.sHе Oĺ淬bе +H-dvװ촟6E7su l_|)q궚ƟZOL Jү+|Mjb_ȱ[Z[$IF7MKA C>N^~u[~_xĺ^]}mJIx@[@om; R ɣxKP5{OZ]#u_X?:k 'i"kKx%yX?|)&/.07n/=6Zm5+M4VK+c1b43`ᯅ]kBOwDҮljo?t<K]&ijM:`5O4_HoX4"X6..mOoٮ)kYt粘6,GHΥ|/{?pjlԼ/cqgixú#ggg#=4Z'l9xk7֠ n+}DFu[-oͬje'Gլ>5ig#+!llM>&\ –L,D{#sm2^4*k1~ٍoC 񿀣<9j>%{K}BM2kh>uVVuƏim4W(G81`Џu-ux:θLփ״nmҴ?`ĒO*][ow#ȉuqsoii 6¿  [Gx7«-2Oqw7!/-ܰi⼐Cd6-<^DֱEe}mֳK^I,zWYדKcom8g6<~.W:𥾛{1iwS붚 д=6{{.qmo5ЈXHm Gχ*>"ŦFn"ּAxKm*O=Ԛ}R_]k.M ǟ 2~i~%xjZ8.<5[MlZ(QOuQn&R699_ G5gHY4 j:4ui8ccƏbD1Zk,qX#% WYhmx<~Q4}V |IOZ4v>6M LUG懸RMc/tm;J[[ `|bRH$K4,3,m G|/@WAD>75[ j׺߈Zt.-OT[Y4ƱQȖ$Vnl|½'Iot}/~0//\QF[X뺇/VNMO㳓JHm:HcF{VQ86 f/cW^Fx]PPҴ?=Έ-3MӢC_j6#)xS4:um'PLkޝwMM'?܋]W%468Usмc2|ua_h޵DdOVNz^ ?['N=ŵG=ܦyFc6#C?%Ŀu`m^AL Xi@дx&Ny6:m郂09x$o }[oך!񾙯XSV.Bϧ˥Mqwjzɦ5FD"[s`s3y{ tSĿC kku6@Dэ9IћKwJSW߆dI`'$O2*2 .Ok~'uo[~ ^ [;[~ŞK ϶^ixxtWOK4hnmHf{~Okƞ!ߏ5 `J[ml/gyk3iij\O{{{[W`6 G]GFdSkq/{ojrmm5 h[3y uv{.0H Hl̓cо zڦĞ/ tj{TτׂBBPKu oVK YA^6^o|A񽧄GЛGeoi[1|9[[n%I䕦s!6 Pi~yl|?%oIkYco% QCn"Kp|?? t닿xJNj^մBYQLyuWZ./ :(6Ǜ()x>jw%=Xc&t4Tk j!4- mwkDi־%j:ޱ?iWEƱ m5=B J]]jΞ ]WR:=ֲe1llx5s.x'CMNX<9=Zi'og2+I"CѰly?o7 |c߇xh=FVE:J]{GeҖ+Im4ūݮs61GfzFo֧b]\>7%ݽ^ּͧ{^\ź$t\3,Jl|-[W%,`o+׵OYY֡ŝ\b[c`W~ WV|/j ֡gŷzkW5;MGK[ӥ}O[+{;[{m (I$c"xr]~&j.ֵek[%˧P[G j`[EAlgHΥ|/{?pjlԼ/cqgixú#ggg#=_?M/>C CO?`' 7[u Y+K /=&7daeYLf 5oYNJ|O%:xtxZ}QcYc^75xu^(&0\$;Aȯ|!fϬkp?Γ_aqCǭwquvjk_CLNڑglhi~k8`ּg|p|W/l/!OJokO6:{Y >42]oqnl/ZX|1vZ43yZEyoA}֖zƩ[۹,l6 7/Jе?m+JYuwZZLկ1 w@Pl#,n3h>#G5 'n/ XYjv cs24B86ǜ|[)5C2x)O*]Ye=- F!qjzpxZ׵k~&+-YQC<;i66zki.L6:Q-NIټ=q{}#w|@Ԧ[t^8lWXyh%6 G”׿5X"jŗFn?,YkLv>ZxLJ.A j`?&x%_jJ1)~Hh @O:\^."յ]B-1.vzuMi֌t}6;ׇ쪯u?'5xE50cJc'ǧA=U.?D\mX?!F?y|?ci?ίVhcz_ޗ>CX?eiUԾȇ7 oL:j6?U.G ޗDm8ҫ>S/$.7>?D9h\myg=uo)i)[C zY `uBA~RT=v,+hRkAV(!|?,1j+jZuX_PQEU u5(1f81@irT__ UoDOLa;78=t G%O/s/|# 9*ϙ/CoAG%O/sCvo ((>exwc?̿AQLaٿ7?9 ! ?;7GrT2_exwc?̿AQLa;0Ozr4 {ӧrT__ _ ! ?;7GrT2_exwcٻ$'$G_\J\LsnQQQ˙/ (&0?|%ߚ9*/s/s/1 1|#1$d9'&J\_˧CoAG%O/sCvo ((>exwc?̿AQLaٿ7?9 ! ?;7GrT2_exwc?̿AQLaٿ7?9 ! ?;7GrT2_exwc?̿AQLaٿ7?9 ! ?;7GrT2_exwc?̿AQLaٿ7?9 ! ?;7GrT2_exwc?̿AQLaٿ7?9 ! ?;7GrT2_exwc>G<`wϮNz_<z! ?;7GrT2_exwc?̿AQLaٿ7?9 "c<# 9*ϙ/'&0|#881|_D?1fJ_ (&0PQS||# 9*ϙ/CoAG%O/sCvo ((>exwc?̿AQLaٿ7?9 ! ?;7GrT2_exwc?̿AQLaٿ7?9 ! ?;7GrT2_V7JYb;-5]>RVҵMT;Դ=rRѵEX][$-Bp?/x'ycm5m'O_>?zVM kQV[_ط3G*DB[K"[T΅M}o>:tmW/j+ ;?? PnmKJԣ4ִſmJN~|XЯ%GuXi*kzo4 ]U dž=}Igg ;|N!Kgn+(.v!`># 2pz:^PO>HNHL~Z|;u~GP12G 폨Ӝ23܌$c3{z{g4?LcA$12xǧ4t>'?81L ~6@#ׯ ߽rAӕ?=N s~d|A,ÐOB 6۰9q@ Pq@ WU+R5KOѴ]U5}VN/NP-BXl-m yHwvh xB5? jxg_f }SIZv~~uo7uo%DJov_Ե+m&+['MD4OZΧiY-iu<:z]A%-4T_x s;~c࿌_~ j?XԼ/i~.ŭNJ>~$4MKE2ktԋZ$t!46yX'2y~ca~zt< 9|c?@su8<\5]JE-b;4 J'J5R[{g-7CЬ5-gPhca isus#G2"Ec3Tx$tx:)q`N:8Iʞ(=6#pq:pO+*{FI py{3NJ:`Pu+mKԵtJ 'J5R[{y.M+;KX(Ccagsuq!xeUB_=#$s8< ᎼggG m? zOc8#Ώ:'|㏇~ZH~?vw}C }b+F o izF9:tWVI(ǾǶ y}KǰxV7yuXk:O|C{(k:՝]h.|֠wO1x8>nO0 v0q Q遜8Gd{4Ϩ`tNA83{z{g4?9y8a#m:~w6H8zr{oCA'|c'9#`Ga9 ܃olg麝o%ͬw錺h&F/t{b,n&仱[KkkG[+kzdc =t>=^ ?/#x FMgI׈.6M:fݼ-y,7\e}m/xt9\ssx?~88Iw8 Az``S{zONFA69<5k^;_]5+ xT&tȱVz67-pդO Cizkah.ykƫyoizN[=ȐiuR-$qG9l;Scydd9oViokS4~)~_O='š V;~Q'<_Go2C}GړiAsu Ixg cA5=G|9ßh0i^/|D|mGon^^;%VuۋfYfӠ掍\,v-e%+C~Oί&mCt"_ kZ[7OM~?i__Vn|GЫ-\ i%yߴ6>$~zi>8嬺_K5YUyXψON:^~Zl>4ֿfi;Q4)9}w_|c^oxDxm' |#Kٗ'ڏ.~){]NO4SQ=sOPxG@ҍwaukBiz?AI뵭D|g']xGG?d_C4/&þ0kmτ^-9U=wouJO|[>"j̟?hU<7|;".4M?A<7z~ޓq⋟KKz_"Ե_$:Ka.^&㯆^i9>8'GHDZ6eo jӬn|[W}ZZu~-x_?o+ /+ x㗀<#+L#@\\> o ~rP=CGGXJ<x׺UmCRt4KX |T Kž~+OګƿU_h_>"|H|f\׼!o?о'x_\M{DѼ/vi3z,Ca.C?i k7_Av|lYce95zwD][U<ukM3Dӭ[8Zٴg5i".9 wNjl>0O?w s Yj?zMs uKx"ozU񏆵.DndlzwO7: Oӵɾ7~hچ:ej:5\j|7& %ReLIul%ўcig~]G~>i <x_ =iW>+¯ k>&M[X-dǂ೓:4JǑYx|kŽ|xw|Car>3ľ.-k?/|2>9 / S> {_ Dm: {Y4_[|R >)|V|Q_/ZߋZF_šOWH n-CA- ݶ>; =;oP5+ xoIѴM/ X4O5]ƩYhFI@u=NSסѬo-Fij<aCOVW~^о|DuoكPxBI|N𾸚yix_StoD-.g-Xc[%hhx;*NWj8_xkW>.M_K#V_[OҼAi>ԴbSҴ]Gí/O^H {+#5>'x}_: |2յ]={-xBկ5/Oxr=/ᖛ!,/#tweNXw'կx۟~~85ߎ|?4h.Mgaf||?ךb^ִEya$߇tGEI5G~?}u_ /jo൅g]"ºI sxsX\ci|; oe>]WAQox?_ bo٧⎿o|/q{V&WZt?\RE omw:yguD;kOǟxžjEO9Cy.q;_hLZ#sZjF;d{mSJI|DW~;@;i?X<)x[GKP|umjķVEYkTմ!gzou]P vءk|~Z|OviK☾#? gּ*oB!/ OZϊ4k :zg[k'oh;_[ZwÿWUԼM[m/|Ꮚ\J!xGgqxĺkk/ -@Kc¾*+k,$~%~ݍotcu_U__Kou=H|O?޹zMլFjmo_gcCYh:|ŚvKj:ׇ>xb?ŽG?-O>=j ?4>O}{}G5Z5VDx#]Ə7~2¿|xF_|> ? xF״cKӥA_Hx[Ox{IAmbc믉uj?Ҽ529隤^-/umwWtM+]֓yiy GCl|/a.zƃ iּ].Ú%zZxo/<{b"+z"}'@>[J6Ȳ|m⟀|V+FO _}~ +h[D捥 ic7|9kE|+i6Zd"Ȼ%Yy/ڇCGCxO~ޙ^ M|u' >"x\WIx M<5dwk~$v'hes?Ikg$_#]3HW֭}M_OŚf𕛟7>/4*KW_4<'_W[ݧ|qeMox/uM; /SGHOˢX.=Bn芵]?>55x{R᷆ωMW VR#T7w𕦉ons_4 /$y_~NC>$_'uc[_u_Jaxo躆)_[?x;ŚeĖݏ<>n4Cn˶zW~*wo~$x>/|4ߋk Qoh gڂ-~*h/d_GGhx_@<]j]xg>P<jKY-M}>խ]xV-WWE?`4M<;k6ڿwlt($dy Z~ОOC^)Ꮗ-| ~ZmHK6:n? |QhZjWխ1&6_}R'$|c'$1ῃGMZMKF:E|Gwie+!ɷixWOƿ5? +ʞ+=*UҼq Ut x?<kmBF-v|S⯊iuyCv;%c>%]|,-kSh^>$G [M_M+T΅j^m-<"uvS/i5nI[&_|jmssKW:ߌ? l4:Wm֩x]IJA7(;X㕇~&վXi =~"iQ~˟>){Rq?tV%ς.ғG$vDQW7?x>!ֽ?>_>$u|[MkZ~<; 3yuł=[Eڒx= v>:?xDO^|ROXYt xcvo-6/ uhq&m[˧Ե!=uω%[>4M><Ï3:cG@Ou~ճX; MN ]OWĺ/^1]]~Xh&x6o'GMӼ'o>(-UOYGȺ >6?V]m'LG mxk]J|e kmu?Zx6{"&cJfFCMý_Q΃;>:$Ѵ xVsM?^%犢kMu&xHG y>!r8Mך2 kI#MZ_muXE|U/5/V=fxwDêa&vK;%hsW|h5_ᦳ~_/o[?C|gB}xp]y?x[D~ya{֖D^2 (V>5 +/?co_ZΗjzWZ¯L֗_g> .{k2?¦MbY|Lm)WC~3挶7o#|GVo]o?tG MK|jk7v>0ꯧXIx},=oi~/jm#?MZ? h5>i>9º~^]ۍ<=Ci -,CIlKz#뺒 ( ( ( ( ( ( (سd+s<,>;O1~ |c'M'-+K7^񌚦Z'Ԥ5Anp 3:wi['9Kr'&ZIjͶgjvz{h& ]BA๹5h{ZD#6 ՛?^<9xo'/k^t+ҵ;nQ<9Xi1OW~%<&o0}8Sw_,oN<dhSf{N6j6c~onSi7CƟ |G-ψ >xZIi/ T๺MP4K$4XN.ែog |%i_wU-|3 hVږ PkoD֔_Ict>jMk)~ˣm߇_j/oԤ_࿂|2wˤk |-isZ^MOhzUؽ݅nK8d*^%!mfg~ mv4%%|7o5@X[BVմk~lX\Ei o?|4/x_I%Kϋ<;|6mGPп"-To9G ʼn ャznZhZ&aeh>emizNAi}qcakk0o iQĉ*mF o~%wO%!m6\?~ klZn,mC#Gb(0,~.?8mYl/zӓϴoZj}ؾߓw7+🅼c~z,t&\_"7u4Mhi_kٿ aߊ'0ĚtXO<<_QXnHD@mO i爼Oxk_|] gtMck@fx{6Zwo.5ϰ]u̞T+Mw]F{y-֫xӊ .xliz޽/s [7Jgu[{U7.4x/]GRI]CSo6GSo"[)kNvOIGkm|ꐾ,:4u B94]!{i$SYٶ#>?,:x |"wIYwΑ.ӢFmib:]vRFiXOO xB<7xZٿ;N-G7p_}Hm=Bh\$ְʠI[yX5 x_V|;Wzok xPskk6uhmw9aL1SoA6 ԡ?_<薚ZO h)H-!Ԡ+"Xċ+4 rW%T_~'mHSox^:+c_?. o+c}-:U Mw|A-l5k>=Sum Ǩ5HXY/VJ bx˵vA?~sž5j7W]oT.4xcP즟@^M6'˱+ z3KÞ(A>yoGm紻/-nca}[E*Hm`N#~ x/_7|6j'"z~3j 4~[iXiф{V0'<n7 ~|;w?tw~$|Qp7o j?ٖ>./%s|vOmt/]- \tgE,.cGլ/Uҵ ym5 3RH/I{yxWI+o+ oZ)]ex_ ik+g4?P/45u SSP nIo ]v0|/g&Z/>"gi,>>-|>|K;z^_Wׯ/W,iǥh!{+v'jvp|$emBUwk+uKR nQ\)ƵkkKq Y$٥\3_#ƟKCxE"խtxkFNoYAW0CEi{{ \$bExՄ$NO٧rKN]N5=[O%UG5]jIu;==iP\ǂG m?b֋ǀ*C|:-cZ%V ,`Wkj?m_{[-S~j? { '-εx_FCĞ" (5Gld6]#vOxGǚ-kxz74]RR,`+KdX'@fpo Mm,`IOڥ/ORZw𬚥u*W&{__]hZ:NhK3-ki0վ!s6~ѭ|cpZfh,Q4Fm繒0MŬA X/enSF/^_dY?Flu`ZI_jֶufkig424pF"S4')>$jKYY-}ƹcҊxP֮ԑ]Mp&ݝuϛ~ qE?ƁxSWx2Ax[ßxkR{ig¾6{_F-&߹SBG/>x^7ޱ&j~ѯӣ49-Gll-*Jm[z-z>aZMjֺm^kޥ,6ѢIk:upqssq3'|5uŨM/Ȓ?[E c |6=sğ >ZcX]WTNךv=BI[n9顷hAӚ5Scm`g?m_{[=S~/G᷃oSn5o]yg:"F=Fr^J>? G^0|3N;XmoTn4Լ3O>~n.Xfj[P/>x -| xkO4ZsZ.m/ڭ=7DL=*U;PX.IoƟKBxD"խtxkFFmAmwgY\)p;6X0v|5ooH>|1ᶡu|;^:o^ t nPM[i$ڿKG#x(k! N_odhi※]YpAx'H[^/#:UԵz>0.]_j66q]׆|56s,-$y^ݭ x[¿?k@$z`hvc.m{koj>mg͑ -@w|/x Ch}xj|:F5Nӡ~ݤ[ |h#M P+?_h*OFڅ/ jVK91ҷ蚍[٘ez5l閩s4wto쏂_E5W߆ k7#E%MF'#~ߨZWn,mD5=,IkOKY|I-{Z m$5 }jPR["f9~|߳k|E8 šjx__?oQ2h_"1{k_FHl 򍼅vw'𧅼m_x_~<]ORm;_е[6_o[i/>iis sZ*0[yXȡχ~ms^jV &Un5 Pl]>իj}IWrs+;Ÿo 6?/H|?G:#W4:+SMW?i_>tHo+~)oV>_xgKOo|/iдwyq^#Jwws/K#b@n#ӿg?G/ZW"+ooX7⛃ecoû ?&῰tu;xS6Я/O .Φ-h~j?alߤj^}B/:浆EąVV9 ςt Cnt_MjzҧҞM;k:ƞ5Gdmu[su*HV_SC,x^נW;k~EiO 1ik~ CTԮ,"I VA%Č»:'K.43Ou 5mEe5-]OIxE]gW+ZꗖRE]9)] [Xx^IOw(m x@ž3 σ|;> ZX~>.Z5IK=çFєi[Z)]e__ ikx~wm?P'jWQj[$̫ Fs`:'K.43Ou 5mEe5-]OIxE]gW+ZꗖRE]3Dž<.4qi?ecr?>hZGn#_/X7l=W]_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ճ_ Ղ_7s3`(W}]^`Ճq=11Ow-_ ՃpNz x$ Auz]6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V sz瞴{ahw?h`{ahw?h`{ahw?hc<A"uw Auz]6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|6V Gw|>I>&?e_3Ov#c+CoP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@7#?7s؝Jd˶2PɄlu?>2'q|.1sI#õhR5 "Tom43L5GPO//o.%Xl7Yqk3T)ΥHRNB1)EFnO݌RmsYJJ)ѡNUV!N#)Ns(۟56A5 5 &+7, 9.k[ Fkm:\ ΓE ́dMD.mnTerm)8FRJWqK'7JqN$IJ0JP'8[sB1o~d\C8u|ihⷲK~&󵮭foTe?u=3䁒p1 (Zt)7khҾ++ꞂըM6'K['M5 5 &+7, 9.k[ Fkm:\ ΓE ́dMD.mNq*J296})%+ٸpm%\\ttKԔ se4!8NqFE?:t#c G8K;kvmZimz;%mkfcӷ` 6y-ݶz7}V颽&%4R.ig}R[_F}F|5j]xQǥjVWwNu^Z\j֖2Xiw_67PsjIar-o(* l5IJ1n)>JN2pjjN$啺`ס, :Μ*(UNeSj>Ҥc)S7:ji)R~2@$~Xk[׺8V~[ߕ]{ֶz-]vMf쓶׾$t܎G^cGZ٭EKukTѥ?[Y^N=c`G%i;+K'e,];} #֖ݳJ۽VOhM.`iGlIui-&DcOo{VWfд+Vn4[Y6Q=[y̐2AoPC$ϋPab3pxQR,W%Ԩd(Ũ%'>U 3cX|1'.JQZ4)YkZ:qIΣ(sTbU7 }5gg~g9?kw6>Bj>ҵ=G&k>@-K*M{oӊN*uN ؚ{8xTMFEg T2MN~'8*J>"bQQBrMJU'ueJ4(_ӷ9A#zW_nOef8赾YɶN+W#_ WH5 6;-Iºe>wolysB-zx<1SW ֖U){XRyA:0rVm69fV~ޞ',%):z|ᆖ)JhIZNy(;]7:*x^Ե\N`NꚎoaI6.sxUݵEq0,aR4" ½i^ڝJpPUz3Pr:MJ 3抔{J* uOEvKh_IXYiiڕ]#Ww6 I$i7 v&hҧ^TgƣNZQ*7W:0Fubէ!wu~^mtN}Rӥnxլ,JKlZmJK.k{+X y4ĄZt+F:8МN5eu*ҍ9U хZ3(F>hi WM4.??9Z-압ji{zmH1nrm}-n[;6ۙkźwѥ'}/J SPuF Y4Ojo%cQh->A]\l)ӧO V)BQFJ2WapB1>Jd)5:Ei|Ŀ|[xN [/on|AOG5 G]`kK9nak-#HyZh5`pqN<..Zp XxJVE(ʽ_e{ETb0~e*I+[K%:{c: }Vj\6֗i$z]qzw:q=#%~2zg,0sBui;[[FUߧ 19 ӗKgk馺4mo+f5ʮ? N:c;rqf;*-]uk&ؐp'*~Tc#j\6֗i$z]qzw^S5KS{Wxwė^ҵmm[CzRK{t{Z[u^vVlW\Tv} r[˖u7+7,j>ѽ_Śwd>'}+KVRuekZIU$|9j'OkuJp &:Єye^Z51aNj%~_zGc"'|5SVmy}9GZi^ n5?{o2E[ 3+3=haJFi ieyz_XrIaMԡIRun>"*jF~Ƥ9EJ0R?3?]bD5 ifҼooxO_hQ6|9y4Z ON6:]x!N'=,NbLf+ir%b+R\)Pj{|L*aק( S^j2q]5,ߔ}/WX5CZY k- A񿍴^x{ "VSӍil.piQ!aTa(bV"X,",^5ʢ k F¦zrJpQkFN+Q{! &kK6 T|}{~E>7xA]jzq5; 7N1zX=*69,=*% V*KVƹTRMahTQsNP Nj2mytJ-Ӽ'-> ?'zTp\Ꚇ0Z50ݖjxSx<-4_kDi *e_N:x\5|]Z$J*2Qz aUc"t-_ds" kz&K6H5]7~5|'> G+h>Vt߰wawĽ>1zx=: WSvr"o5Zu(S ޽9С+Tq{]}/1:TV7nuy|Ku][ūh~4<5CQхzxD[ki\EQ*R a'R1xbkR9VXFW(PFq.0~}/VOt K{ψi+?^jMҮ"n+zصmþ!զ'5}Hj:0u]ZPohmm#kh(yl )Rl<15Qrz|,^#Eӫq(UEPI긗Hsgkwm>i+zO;+쵶VI]kk_tdz0}x-ӥk-myY^qQ9x+W~ݛjZ톋k :.<9_߽9x^;[hgh.̻/f6[*c1sTSZ4HATTJ3!9թ*tޥHFS)Fe'/v;˧~/@ǁ%֧~v x^"ʺjS_p]jmb,ty}F;{WTN/pT)wtKBN0*xijƕYӃikWݣO+쵶VI]Kk_t 3+iI+n[6=շ@մ>[۷1n <,֫Vҵ-6+\8zb@GڍvK'tvڽʕmU2ju^(-:]FGMZOԯt}FMզԭBѵ`H'IHK5t+F:8МN5eu*ҍ9U хZ3(F>hi ku9$Bs'ӵvR֊ww2nN'bç8=1{4n]|m/m۲kU^I)^-˸t?Np?< ɲ[nގ+EdmI>1Ǧ=x;hKuk8VWKݫoݢvݗ+}mFAzc`0$gD$ݺnlVVt[onX-kV7YY|9O jfLt2GVq<Fvwm^ڴkvJ۫[?woǧn8Am 3OE[mbnE{_LKi6]R{> O6k@&Ru֣aJծt&\խ,dlo[ ZQ=\U j*bHR|%O9UeV1t!7ԜNI7+uPׯC*XJuS8TPV48՜}HR ntS821:dH^+ַUuqI+V^-mZ$'m}IOl8=Um妎kfIit%Z֩uK~LzaNǸ~V괺|7m[FisJ=evrM[Im6'{xPHDZt}-d{(7d%k}MmꝴW.>G㿎 j:bjl+t{FJ dX`˥복]vIٮrOAӎA9+1 c9as.KMtiWM"tk]N?t9v'8wUZZM!OPTF;,Rm;.Id/[4;%>+?jMҮ/".+kzڶxH^> =Uլψ5vtyRȳ:'B0V0|<1Ҿ# eauq0(ҎN+ t*\KocoWnz yOӯ8-J^^DO_tdz0}xݖfxco#09#䷗-nWn:Y&|-'{jO4 A{ψ|O hV71Z5͎-Q2IgsX O*n W*FL6#Vu[)ӡ 8I)(ʽgGB2jc+êbKj*?F-?~)ךztȼ;[ ޶-[C|uiI xR,]WV.Z"[H/hyl )Rl<15Qrz|,^#Eӫq(UEPI긗I935߶fsKx<-]ƣ,W t]|X5mzþ4k-ZU~åjMLn#NU(`]/VJBNt({J4Jq{" kz&K6H5]7~5|'> G+h>Vt߰wawĦLn#NU(`]/VJBNt({J4Jq{(xO iMCh_l-Ӽ'-> ?'zTp\Ꚇ0Z50ݖjxSx<-4_kDi *e_N:x\5|]Z$J*2Qz aUc"t-_du?Az`VVWn7\m}{nմ>[۷g4Z-압jhe_~w{: rI=ېJM;Y7e.]hg}-w{h[Xtdz0}xݖڳW\Tv} r[˖u7+7,j>ѽ_1C_MǢwzu棢h$=:]3Fӵ;`ԯ,M.Z$i0 bX\~.!*]*5-֣ Z0 <*iS¬iʤgJ3r⿛ot?Np?< -mz8vWm$R>1c=1^=(K{t{Z[u^vVlW\Tvǩ8==8k]uK[jny;|ApqpqŢoo,q:h΍upFU:q\F'U! ٥t:Ug j'8Sy|?7R_ WH5 6;-Iºe>wolysB-aɳu:upomNia%RW)W<.o5fi]ÙkJlq2R^ib4V璌q|#m ɯ/.4[{k'LO㰿)YJԴņP5<ټS7( \%8o3+[G*U)FNQXr<GIMN1 Ӳ FM+uwe ;k&ZE \g>鼴ự57x\!ёN1|ձC@o||Wug.\rFYIf; Kl4qIoy U_4r\}Qok$|ggGOnԬxGblVM2-œ>M7[k~F%:gКiRŧ9J~IFR6qlT{yv:0prb",f3ZԨ㒪ΣCB|V4*a|6é|H+߈Yܖ6^&|G*jMΟx.o XxS:ym7K#ԫu)MV^ҧEJeGK:TNGOy<\A_+JK3X&&j4oBΎ&RJxZzk]]S|L5KË٪k^x4^$v=rk,m[!.5/)Papa0ʊkżϖIթ6R7OJ&0:K3\kGC,CVXUZ|ü5L5 U?kR4Qh?{>,O#.,? R>!{/v۾(|;嶷yZwMt1\O!pp:J<өJ1uAgbjʜ`S 9 OF0FCʰqJ֣41JP)Úpb<ᵥKE_E5x @5o(69STltspExbĺPx-i\8||Y\ <=\CM}bj h%F:*T*8*Xҥ|Euj:~f 5:yTWC W*f8LM,?0hޅISLCS Qz)-^G>oOoQ  oNh_a4lOk 1_4%[o?Wai4)qMzyo}j/+V+JX*徧S)2҆/a 18\O"{Z<,RԿǴοiխ?BW]x{]-exᾭ_DӼvOvk$Ү"uy7욏1ʭ=|2[V.tkJF>N.c)*.,D{]k8 Eg4%.zc*<$)`*W`kөRXlF5}^hƞ0< ik=,AW_' i}oi:LjS_ ֶ/gOW&gJ Fyv'_uU)Jn,9USu+*c**~') eկœpX\U(`R:1PYNYRН:pi֌wR!߇:U5Ko%ao7-Wmc[kMxCGƟK+mFžs\[CekcsE u+PXGԾ^*_#G:qJ+Œ84\FsNa^x<ጛYmO98FX|SbeWN՞#!Wg*teY҅:zٝ+'8q^CoMt|?O?' rB~_35X;iWul,ȯA;Vu~8W7/_WWU|'?<\Om׌<3Έo}yū]ƻ+{JMuKe/uhmROQgrR瘚x:~EW5tUNxSzt|}\6iW,>c0}e!PO ahb(eqZZL0%RLCs^ix/ O*?><ދV'AW$<i):>ϡ}T,02F%g03Nou9xiTHhUaJXJʻ u3Ӑ\&&pX^ԧZ8e 50ThRGAa}ƭ6?_꿳 ׊쇂/_o_?.<\W^Ӵ/h]N!i=1.7OPUIc ZSW)NGiVŧ)acJM]Ɨ܉ca2j<[ OPNoN^H8b^jq;u}ji.GSڽjak?g4=2,j75wy3lC[GϱX,5<-(sB?+cO㛨 4c{)K~qfahS:sus?~|/]<MkR=+: kvmGoyqfSڧ|&/ORNZt,yao1_pQnP哹QqzPJ-*oXɯaŮi)O!6-Mm;_ßx_5//Cx_࿉t?6zm\(-5<7n#mFÙ}aq8 *ΦWZ_ lDၨpӭZ+:'-TU!QmSq-*z5{}o_3׾ v_Κ/A0DVe75k^;\l|E{/c7Ui 3w,8 \0T0T)SjiT3[E~U)ʦ)nU9[Gnm:|avF7>'#ëix׍ox]LkEƝ[ūSNVN[N`Գ˳>G>&x\+SAT86 \5,2jBt1Җ0ºںU15i^eMJܚ$m}4[s|xg7Ư3V^$| o.!F #SI>-G<_uχ3xB|C- J=7˭Gq p?1OsQCWK/P*Upӥ:8TXLS!UxjQAիBT{WR^횋k}U j|aOU) (h_ιk=?DpNc_ϳ}m WLJ&  Tu~o_Q3;y+Nn]˶f⟋Zi]eoA?iϊues[jVZg.c%ݭƭCK"fʥz8VgR|T2t0J(-J'_N7 SNMLvM|wҒJ˭-tG?=wXm<9?x2Q k~*,5+Ԭ j:']jK[[[↗s[^DTGPXR9N R6T)ա*uɱ4y S_v/rfo?gx?5_x/wXaxĚ1|VGIn{3--K#o~|K'x.hc ]dYLn*G ڕ!B4fJR3t ogs[fkZfE&I" 8R0cfӧRJ (֯S2&roЖ~rw^z&-Qv{>T?gx?5_x/wXaxĚ1|VGIn{3--K#o~|K'x.hc ]dYLn*G ڕ!B4fJR3t'ˈhTUO|z6Ԡ]~),5s>kO-ç!׆|1%x_Eψ?4kWwV}Gapx\^[__(n_sc~˗|?>Ֆ!+ՊR7>Ȼ/FCx_W{x << M2ohx3-{3þ%։mOC[u -_,&u ey^i \pxcԧJlF&xUR&^֛t[fh[JGpxj㧆ե$[[ǍO>xkW}⯀EM}fSź=-$"6 ]0XLe:̪INgcZSBt{_kN1~U98rݧ g|]c/<ਾ-#Idtx5]fmWz_5Z^o f艤h^-V;u/g9~WC%̳fU){L&PΣэ,\4pAag,]ZTJԨ9sӓs.xFi7gzOu-<xUEoOk#x63j !xR@ăy47DM#DjxL9.e2Ka0u֌ib牣: 9bß -[ ּg"Y_kmJL𶣢x_E֬dվ(iw1L9REC c3J*T:L%Jp%RWV֧G&&ik]{+)$mٯ#ͧ^x Ki-|'H> '{Śwa){]`h!e홴^e)2 >_XU'3l^haka%Fq8/u=µL~ 7}osk]wUg ? !/sgow/'n4MO1Pw5I4/ Z{UԼ,o5 8 yF'Vfx&5S O^,a% *_]VYҥF:iRSQOGݽ>[?5t^i[5ZcO(=}F5H|W(R<-u-nn5o]mmy7REC c3J*T:L%Jp%RWV֧G&&]S_v/rf]E~&Yudy}{^?gE杕]E~&YtmN?$GOn<*5oh?jn[ Kޯ u[/];3&^Z^ yV]V,.8^9|TqJCuSp0*`YU7R|LOOxZo|m/xSfKRS6I{gOQk >^-Čܼ:Dm,4^W9mEa:A t59P_$1qLT)΢JXӯF48^3%R}/C>^^jzg߉7?.tm;?g-kT:X幓Z%E#/f- tp]~jfЫSRy4P^-SЄ9MN6?_(Rwݕ-}VO u{y3z|Za_vOHxuq'2̽C0zTSY1ˈ<=,7!fESB.z0z\4;.r]z|6 '1>3ӵX9> wUk=+HuuX?+G(V/.|6@07<R\MjqK^rBO,緛ӇQPI{E{U4iRor| \gֹ+xM|k>+|uo i^.&R,| sgi 2n"0<7G3hF7PC0VU|-jWq0ع0VSJB)sU&I[އĶ}6=^o-~/:h6-]ZO |g{qxusm쿶7/yV&3w,8 \0T0T)SjiT3[E~U)ʦ)зMʇ-ɷ6zs1C!x¾ֵ?Ϣ |c6-i Z&~4Vƛohz}Du?.$ʲWcˣVQVtVUU0u\>S}VTal8 x j3roಇq$6'ix+ 9K~sں:>}||i5>M2j=6277Y>o8 Cyr|#(p*uSJ>WJR!OVr:%mcթ\[D^\8b&kЧJ ^m 9)GasL5:~Ό-Ci4nC>b\sAxÿuS_> x;K_ cޝiuC<ƟOQi/ԴmWK>k]B6V N0<08"s|M\V>(`Bx,T_.RX\El^2*TcRڋWu'ĺKxh{VojwhڮOoOq3&W׵ CgmKzΝX!<;8G j׎q1S:4مZx`x{JU1 Nft;Q|?b'ñ&|kfaφ eW=¹Ӯ~-xŚ`MKZx&UKU߽5Z,JP(93`KΖ6TY^ +OF>eO[FURT4J*M>],׺ݫ/xT5JW}ڔOV~+1q}Njiϴ[D(S?oX[hW?۔9d{sf(ҫJǖ2kxaqkJSF 7 sSpr:8O>$|=ޟS£'пZ\h؟T`:+cԿi/J`Įi~S31ؚod_WzVZ?U?y}N5:Se9 ._ cqDy*YR?oaiiZ?j_ C[[_xZJ'}[TƉx#FI]ZEu #iXin5cZ8zeQB\n1qNL}:8]2RT]jXR4qJVhK% \1SUl<#VxI/RTקR<_l،,jjS+Ng[}CӮukm43M"^(0LΕN64SYrVUU%uNTNS)_'K3హP¥ tb<&6E>Eѡ:tӭ~C9umcfk: iKioZ33N^o ?ėVڍ}24[4*xDWO ^50ש}aF#,U9F5F.tV(WB3qy>iC 0<= ¼y6ڟVsNqTʮIѫ-'#fk豘w<(_~X?kG^vbcp8co_<_Կ>*skYO xo7?g$Ꮘ5[[]5jq>b]kw*-#^o//<=s5bOrfU'(• ?j%e$,8֥֔Μa,FCZREgP085PסOZ2Ӵh%cJ!VLzy6)F0]VM~5Yۯxg:Ýuk5ߋWWwVޛoˢ^iz._.W/V6}M"ƯZwl?麭T>UqA<S҇4)sf49СF:J7wfn ;ӡQ~9~ GYч3QrS~ֵ+_SҾӨ_ jP?Yvo;ݪ}o>nx0(TOǖiY;% 4Ҧ|匚\Z撔}BnM&܏59w|e^: / Lj,WXjm&|OEΟ⋽.RS~*M&6m<9po a+g|g𲡍euN :թBxrѕLR9')BҧW_F:}{?e /_Iuh>[? >#x3QƵj:Ι^:x8qZκf7q[-1 rc CB8F|eHs9e_RbSPOy6ӯ^Nw ,~7jm~xc~? |:G xvwuY\i %Z4jA饴& K<:>slNG1L|ci`R/aV'C)a +SV((TԭɢJK7|_C|j_<%eO:ޏi:4ߋ}S~"дc~̺ݭy' tY9=1 \ :XZyaqL+9bW<>uZ,KMW5{9.ne%٨G^8$] W",M M $?챘7چ0|>[x~l~G[OW{qk;?X7{ܻkj>)Ɵ{iυQ{k^3VaVQ]G5ex[Qß -[ ּg"Y_kmJL𶣢x_E֬dվ(iw1LJp9 .%Υ*Sda0*Q`[HN"\n/ ZZOZ5iJI+[G.k{7cUgŏ7I_iqit+fW:~/Կ>y'Ŀ_|W즆?0іERp[-R#Fj,;JgNgdF7­9K1JQc[M)\~8?>CG߆:Ǡ.4;tz]ZgUĶ:Mqti%ӆ e8qY|ʜ}.˫p*:uasgoӛ|~+ P8(r?umZ_.Zh|q5]s_ >|=Cÿ uA]iv>'xwE/tl'mt4=+ 1^y'Ŀq|W즆?0іERp[-R#Fj,;JgNgdF7FNRRy~~~ }7xž ԵO_/t>'xSOKog9CV⯆ϭXNJ!<5Eޯk:-]+pla[qXzz5qbPYUӭ .qxlFIS/IFc6~W=s\|p*e'}SWS|K X~TWmJ Es_>-c'&4R7:RxgOw1 Tڟ&6xk<'+|IMz~i ^1Ҽ<7m7#q{<{? KÚKdpA\,\/ `< y1 ǎĸyD)UQ1q|0XѨonUahss?v/}xk^Q=_-hKeh!42^V}]if]Kn<&?~?[e5jaԍ-jjb[^2xSTc&t(zM{]ZRIZ9u]E~&Y!xGi^ {ƚn_ G~ D?7^f ^Y{fm>רtn0#S3LNgKS}c V*xWͱy*?YcQ4# &qyW9=[]||6Ix Gh//Gi 'Cxui!;Wwx%fzO2,N19,#Oe%Zxq^6n^ThgiGSЌ+T'~ 7}oskȻs^*|S{ 4⿃-&-2LMZ< 7=v~LjA}w{&JhM6c\i*V嘼St)!c#{)xj4YY:XiB7I#OOP_/q iÙ!OkemuK:x, ҍjYzYRS,-\<¦eR TՄ!g*Zj٫ss)/vEZ._~!A_kZgR14 Z}P-[?C+M~Mt4= euk:Ek q&U಼.;F]gZ˪v겣 Dg8Sx|nM֣7)Z (~|Ps㯈YO['m࿃D5XxL#W'{BHkBD,5]:~7LLبǝөRXVEAͷ*./hz|gXύ!eso#þ~ \w!τZ֨ts'ڵ+Ku(ˀG ^.5hS[}XsWS6ZҔΰqj?gF!4i xO -_ |(6ɼMj8'k?#l SԴ7z}Ex_뫉 q,,8J55a\atU:z*R 9C J*QrҤOظ~-?~I}o~ Wo5'i9} *~2].nGՕQtK}V0XoUY\31.<*`xl,h7Ҍf΅bTR֝L,&,~(r_oǔ/|I;X |A׭|'> x,5sK)y|(ۿt= k{( T/%WGĴqEI_=08ZنcLqt+axI̩K`(ʶqJXʕRQ ˽??u_wey𦵩^jBO}R w6F#8|OS>wytHxEB*rӧc э g^S[r,bc҅UyiSxXM}{ .-sIJ&njnnG ok+:Y?}=1~-W:=ĽgOk9]*DtAJ+}J1^_ Eu{>*o%j,-9(FgF0*mӨ2*JC5g,W'Fjj1V:G8`NNjjY^[/i(c/G3^W_7(o?|=c:08oAN}0y~  >ettZ+[%XKWd}l >\I#@ɦϙ[;jm,Ŵ쒺KͦIYޛ_R_YǪ^Z_jzl0G]iֺ&UQXRIMbFٺ:r,g>iBnN1sQ(ŷ6n1|tURrCӜ[gR2pNJ.1:oiI79=3NH#99ZWihz;hW-wVߙ{ѕO{k'{=G=2zR?{=9RWmI}J7Ǖ[+$MivDMiS5 "U/4/K5-GPNn.$H iVT8 tάN%)JW4҄+.]^eRVHQNUNJ8S.XFҼ)5ܹ[E#9 %fbKm+-n䴲r)kT߼Jlq ppLLu=-U鮶ۧ[@g=~Q{[D~|AַKYo}ot mz6+_mMEd羾~:qϯO{{뾻Zƛ[E诿ɭw cA_]]w%%y:tzG\\ym$սiGY^Zr~ϞyemvlfիYMYɵ_{h36饚onZlfK)P~?÷s ݵqﯛWf;ictGϾMJ{$ގeKwO]`~];c8ߓO}6뭶vNlmi`g$>v^;;+_m^iDE5%tMBp1L}z պrӣ]-=-ܻ1qׯs$N]wYl}${9z]Y.j'&Z[TW^ݕO_^I*1(Zi;[|M骾uNN:gPӗnkt_r1>*1(Zi;[|M骾uN1LϿʿLqtN_&@t_^MB1^z``cc9idmo={=; ;zzGCӸlZkUѧkm.eahmzqOE>\vvꕖhܫ)1؜|*kYջZ~!pFaYv_lK.{_63okh_7}zͦictGϾMJ{$ގeKwO]~=#Qd\5ӵ62hm:{ӎ߯=[o]kyZƋn.v @c;}G<н[Gvm3Oԓ@$={tݭvYh;%.ϯ˷c8i6[6ϷR=A>t˵hݯUK~X888G:捭U鮶N_嶁Ͼ3zȢYe[DunZ}Pl91mVN;+%}|,tӟ^߯Bvw}w뽯6e_Z5Ƿe۾`KK:/^GSލoֵEbiZumW}pqN8?/@Hvo-Vߘ^zM(6]V)(?L98^i1:ێ#&%tnoGtW%^ۻ˧X0?. tɧ[KukIv|t}GLZ;hկo˵mZv~ʏP<: $efuP]Cǧ3)tF[{[/vtϧנ][/m:5ܽ}˰t?|=mm-6?lt?|=mm-6?lt?|=mm-6?lt}~OO_aW|v]:=N]}[mzh@${+-=U]6<: $efuP]CNtinϮ0W}c~zg83K$ky}PӲ cۧCOǯ4]"hYo}g~}> 9{iѮn]?_8 M:'{]{V頴]5W׮{Pӹ:g6xU_v=w#LF]< }w݆18HAQ6ݴ4+%eh=>Vtϧנӗn^o1؜UֳM߫wZ~!90I2}hvvWVywwjڵmzevqJlZkUѧkm.eahm:g^H$=$kK?oqC;}ֿ~[]_K/W߯=x[ztK;ik^oRc~zg83iI+YY-Ӳ c]y#4uM/wmjŨi[\1؜UֳM߫wZ~! s#?ROz4[+5Twwڶ N1OAw큏:PݒIEc`vp1ӿ&]ۮ_MkgoN+1c#;_P zZiw{nK[/:zt]:&f^\6yh^GsA9ϩ0uKu+_{4|;j{^J@|?H>:tɿ@_ϼk玠j1fav6ѿ=2}[G5 /*NYH&"KkGKH>YE.Tu ?b|}TeH0Js (TsrT_4TNM{:= {kiMٮ4ρ<u_gƞ4Vuj~}{Lj;]t#ƖI30Ej-iˊXhӞm,jЭ b:ON*NRN~X A<0b!S1B!VkPj0RFU%B|)kIܿ_,t+/&_l'7eC .Agn'P4b(T&Zu*tlLWCjC/j(Э_3cajk9QUIʼiQ2+Z.QS3l_|2]N4iW~z$$׿ 0.Tҵ G^w%ePumV7c\gIgMcqrZQZi5! n8ƩF1Ӡ(ygP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>:8`11(6$2rp =y AattA=UZ˯OAEK.e۷rw=ărXd;=7wkhFk%[+9>k7fމ+$]r;U: iwj|Auz}w2I0mgwgs>dBjpO5a5ofi-%xtB[N2M7FFVg qNO..§+*ӕ:f &cqЌ n0IdZzM5tҖ/v'tSwM"iF-ݸQ >]*K8;K=5K?LNm,v: Ңypj2|#p'-y#7'9N(BVsF}sTU9ZTSpphPSQe(lۀ1drFs犋Zu׭|WVWݺyhap38!Ӑ6u8`vǘEWqqf $|DԵM+K.ozܖضZVB .-tmReJ{jK(мeXwooґ%ޟykyiڶijvjV֗uyc}oդG;|)Ϲ= g{oZy5ay]ťi:=SGz%ޥ.2Caaiquq+$6M+==:c' ӜFp}: c'F34 `pI0Qcn2sc4<'|+𵟋ߍ/-+6m'$iszh焵Ǵ?,-B+kK]TIxzlѿOxk?|[+Pm[AK}S <- ;Qt;V/5VUL)]ooyizuƕgXX:%嵭ƱAj:f Zz.ߵAk\Xm%t?A~jV.kjY:}u%3EzޣNW7dit\i6wauV:֧Dʯ&on$mۆQ !}: ~8q 'ᎃ=3g fwRun3UM^ҿ-{jI zN.l[YmJiZY埈yEҮo|OKcq7t5}:>qQSp{eEAo'5$[9W.%եXzXT:N-%y{XT*|eMJ7R/ZZ\3̩5;E^%>G (JI>Hͩ1s9œJ~}1JOg:%N2+p_hSIǞqRPIJqR9H>`:c8 KeO_"+m%RP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p( |]Emjػ M> nnn[Þ4n5=&{]uȴaL M<%W*4jx3UTǝY,=\5d[*QsFqPjQO*;\P^_?ykgL,.t? M+E,||+xsRӓRuȴ wI!մrT0uxftRr.|Biխ*Ts#8ϖ5h0ժKsVT(~QSLD3 iR嫋R^JeS[F*^RS؟aK"5ևv EOg۹,xI7|[xwZ\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u W8CO0cFhf*0R(CF4۫J=5JTy)8u\G *x<-<ռ?GW#Śx>__hĽ 9"[վy5FIa7U:rRUZ_+է^SEt׫Z/Ky} MWA;L&x^/n}xPm.~&WuύΕ' Mg?Meqa1t^ec߅fm?ź.^i|Yᯂ_~(Դxoit_>&z:WoU񧃼9c1}B|!B5t]xg~,2OJ4ߦWHWKw4}O|jtm~E'ÿoZ<{MxS.S'[o:ާ4彬V~S; ˬ~:λ=BkYkX5xW~'>*J 5wxY|7cXO7Iia־K~0S6wioJOuH4GQV|1ZW; /Zk~ -/lKҴ+Vt@W?oO?~q$oovx{Yc[w:>%4O wM~|^V(k x\K>&Nq'ëۏ XxW-nxTu٭'ZѼl#y-f/o+ ]v3}w|/zͭo nqM⟆>źƹx |;q>K5mVV6Z +s5_:Uulόޝ6{GW}ώo>tj^#Ь>bӣ:'Ԫ|.~?~ ʺ׀-f?[s-6OזuKx J~+tcPt fM"RC֬-?ei24>4;.WǾ*-u_׾ k~t/"S,u[,t@chx3//<ᦣ< kLK[UO| & kRuo#X4&x!nm? h:ơG|1/^4mOxRk +O6 1!{oXtiڇ~O ZNT)rփaxJJ{˞N_ Y*԰U\Q}*X6X:ʢ6G.e`:x=$Ҿ|4CL9tյ𝇂\鶱xHԴeU\xr'Tl]ݎR7XOwWFRuF񩇂'VI:8ՇtVB%l5X% $JPKS**xUQw:Ԫt3gkb0Tk'<]="z“|(/&ުomuV[}[K:TGi-I.4ɱ1{yGG RaWN)ReO*Tyoh$B5Y{TY/9S!fU G),NN. ,\Q=jTS eCM;\Cgz'ܖz獼o>-<;wJL"<=mU;zzv ? ?pϮy*1K|}f)P*V*gZN)J*ki1l.w,^Yse'eO Ѕj㽍jbW\«Fsq|.-[^(KV5]WU5]JĖ%>.}WǢkVukkk!jqF2yc4pcG1Q7ERaB51ZU<R̡O1-*8iSi㎣ 00r=<=8zT凕Lr:^闚>kj Syoyo3ڡƉi2['ٴ|o̘,EZRF!?uNU\#kRlEhV*P|Ν>XrZW3乪Mz ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.}_jvSIn2I'^)FEԧR: ƝES\jқQ|)N Qpg$c6_RYǩZ__s ]isi?=ƭC4ț RQsQj)E~N\ӌ48AR䨵Lj^JU98ʔeQBU=S:79Is玼5#:֡i|E5+I2VKMO:[[jRGߛXo馺oYJϕ+ZZM+]&h[D5m/OҬ->]I-o/-gu[}+R%t'R"4}^UUYA-<-櫥MXXi彵Ư[Z&A<&kmn"iw% 6g ~)$dg6 @.#8C?VtKsRmFumcXկ-/J$u-OQ!Y渚T$ݘ*J\w ?".G\ wMﴛ32qrbt]SVv] )𷆼Qs=FV?t:R&}זYc'9 /^9$p[H4 o!F0F gX uombizk槧&Ưi^]zgaki qq$qR;T%[+oZG 5Múܚ4;oiW'Nu4j4nu{nn!gӲ aV7vh^+K@Nܐ0VVzY6VI$+./ֿ[[v`xt+Nh-o5 B 7G4+LֵKVӴ#Mo,lm罻ixw_~-wvI$v}kY%dP|uC{C^m[Q<{|8E'M[RJѵXA 5;KH#F46ow-i6Z6[-thw>,E/^4g:M7L5U=ѼAirF]5麆aesv:]k}5ciղON]wF- ~~~nI-o4[:Mմ[I-LltGIԭ/ k{Yam\~):={|O y5-z+Ayo_Im9׵4B[m'L#mo4-?n_^jZ湩Xh&ھ[VKujZy$pXX[ZC,\M"GDX% ŏ >🏼KxBx?5D ޡ7Ju=3K{4^ॎuu ȮV: Bc? }I7#^X^Z^X]ڮi隥:}awcomyi<|o^H$Fx$dϿqG\c<?@펽px9H8㌛yX6CUԭ]/R.&tJ-'IuRK{+i.MK+GY"uNd6I<K/qۃ~fG`H >9p0R~~՝B櫥Maaw_ɥhVח֩kiTH_jmoʶUS !5mJE-b-Bk='OtW]$hCuA>] # . z ?KO`}:vzA ~sI1=$kew{[mui.gdq7WtZ'uf}2 zN.l[YmJiZY埈yEҮo|OKcq7t5}:>qQSp{eEAo'5$[9W.%եXzXT:N-%y{XT*|eMJ7jr䓊\+^:'$Zb(}W{jXթIա7ROe':5R)4*>T$m oto/-/׺/4mJпqiksip2dFV5xs$$)$#6(֌ s!+I9\*U=SP8|U} M'yƕIA'y(MJ4"A\@t둌%-W?i~h${|ZYMHA@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>: (:WTVv-Tψ69Km&[nmݧ\FzO>߀UrIsN֭g9E[ QHKފyՒVJu W:mw ].o`%h:Nue a*.xZ.u߂V,|9 _t{gVxVм9{7@׊:~kcKheZK0؈)fJX%sGF]URN)U9Дj xaSb0u>Fu)x*aFjհkS^V5c =J>\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u V#1d 4iƏb n(„0j4cM)CJx sTK:y)8u\G *x<-<2kzwOGC^>9fzƿ 9xBNR1KXxoriڎŏ{Tm|=ρ4_#i> louoZ&g/&Ju-mZv~h|]5O:)~&jֵ xnM>Y~=ޚ5i輎 uۿ _'Ե_V~ʵllnƟ>sٞ|s/Hm&M$ӛyXW\Kl9< =6B5gp%ǀqMQX,u w-ek"[o6>-:uO*LoPG `NtKxEΠYMxυtX[UċF|;|9'|-ΗM_%hx3_-.-}c ǃ|QZ]xa?kk[ο{$ZpCo||9O?YZυEOw'/T^6 Oǟؾ:MxPK/~-?Q귞a%=sľ~|+ڟ-,T5}j~ ~> gź׋>)?-^"ֵSTu[uOHWId]]cauz_X+zo-GC=RTIk u x~6Z΃ryR8fx/mxrG}PѴ}Y<_o5l| 3x?m~x|;a`!ּ# B/xi5J_ N+}]٢LoOg4+wSxO>'jz&$W׋l1umnQ/#,-7hP?m/x$&Oo\D~>:дiŸ#ޭPyw[ká>}jMZ}gjYOᵻe|K=?|Gmuπϋg6~= x|S|s񥖁MiZ7BkZK*-xK[&x~?~!_Y?_-VKOv7s®4K ]o׆ _aUt.d{gxQAw[OuOdKon]\(}⯌)-^icҒd #O+ |*e_: &`:Wk;SC/|4&KĄ|Gu[~\Vg|K'φ <=//kg|xxළ"=0DzZiG&$⼬r:ÿeCύ|mm/Q|O #?<'x^ >!7h xVݢү; 4/]w-5ׇ~s C(ӡx?L?xzÙM[_zGῷhK{e.<RZ"Ywu4h~S|[> U|&D> 5ƅiZfoƱ{gMm`xv >k~}߉t2xÞk/QtM[Wn{}D\V3_7Vm2\7Ox|M>8}i+.~״-sZNj4 Emu l7שtfZ[>7~,[7/'45|WQz֞wiumgZ5wj~ޣeBK=B @cc&]/Ǿj;-Oj:W}c឴ ֡oI.oKᖙVhOk:?Denʿ> 'WO|"o#x{]]o?M!?Ŀ ZxQooK|5y 7:X]Ⱦd?Í LjxKB/ž֓ßu|5t-3.o4mzPo߾v^[| ᖡo _ioN{⧉|Gx >"E)ϋeI2oiBBI]/s%: ̞ 3_\ ~=c6^hi4_ m?]j_/8$M ,M[^ltˏi!5K ˢ\ N1y~xO iχ~'o ~ο|kLjm>Q~;j:mao XY5.-;W񍭖Y1_7 u(h^[!.!|6+]|?CAӮ4kkD.4J{=Zӹ#h[-w"+M~#i # sK񮵮x Ӧxo~=?b)kM6'.\<l AO7dx~x'-?]h u-GVEo < U@|9=&>T<]Txǁ߇:.4i?o[ZVoeb4>AmgܩSgöU|<%/c*\VK_ Y*ʱp5\Q^**psPj5ДQJdeJ b]^ όN[tƑh/ӾoNm wio--*1PteO(UVZVpQTf)N\𢙒 G3թ ~i~/%Ϗ|ox᮫뚮oK{>Yc5ຏ[񵐃M5]#1d 4iƏb n(„0j4cM)CJx sTKjy)8u\G *x<-<41Y@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>: ?Ldt@'װSSj_꿺??j/ZۢMik-M /m"u =5bK 2}>Rfgiq1O:!'VFFS8I6rJQRjj7j.ɸ-*ƜF B\sRtnXJeT 5Q/|1g >g5+]E廆tXifuX㍋tάN%)JW4҄+.]^eҥVHQNUNJ8S.XFҼ)5ܹ[E1RæH'9<ޛY=9c{&v;hVr+iJukwrwu0!l=N@~>kO+1Lr{=-<~zzOÎyx6 cۯqǰ3NzZ6 qcn@l=qtOs6#`xsP| 1A199yol=1<s6 88:}3smS魓g~ַVN={ׯoӊ[[z?YgY$zk_  c$J&]n7foK-aۿdr/拾}5a謴m  ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.}_=>K2[lolrFwp3Ƣh&mqՄdKO\4iJPRFNQOSʜϒp&8xS'K%m'V5 'LگxAi<`,'t;M"Et븬|#c&VcJp՜sգQJI懵J#RIWK7ΥKEӣ &%ʞ [ Y\UJNPJ_o:/ F}cƞ$-M^27ּ;⧰ÉnaƉn#hz>^,|Djh )a0Ga&*c4^W.jFuT͜lk攪`/rPTSBX K2AQF<52W/>&_ķ>:P|S?'^Ҵm ,-6+[_0iTa^:9RX0S%PBUjSIΤT)?3Xsl,bU*tcK º؊Rj:|ҬU:ҝn4lg?/3O j_+5?=zhFKI$^Ot"5cj,4i6cz5hVNXXBYT'Fp'JK'?fFXάpQ^ qRakOF5ԨTʵjIa)a*Eξ G $n_/~:[/t62ɍޡ |g buxO~iathGCF_k::j&KFOҫF5!ϗ5hVg10LƵ5*gpkc^ ڴq^ҭW(Ž /߾x U.?=lxkk߆yNiZm/S׻{}2z(:6q{[xH3VҞ:J2Q:jR,EtoLux[3&9IB(׭F47VXcTt}z ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PPzw'p:d uh][U~g֩+4lm-ee)ɨXEkI}gyi}esz֟Mjwo kKk[Ky2>jkP(Iӥ(BRj*DS|H¤e$֊GNucIT8j-34$#ΩՔSmBוo:cL |c [evejmf}kIZ6˷k'uu^d{s'r]%vO形+MvQwy67یbja}֚^\eaXY%{p4 ge*U:T*B+Nmڝ8~\|kkFZ)UkI8BdTڌc%)Vmr~pv2:ޙh-ɨunDwJ7z6M(&k٭u*N駢ǡ `OnݱG7}v9ۿ+ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PP3|0th2kjϩDG2RFppRo*N)~? n.4_MS֟>eiggg,$F%յݥԏ s$Oq,Մe*rE>G r>2$\gU9҅xSe*u9*F/T*rUB=HN tm?=5i:ĺFuxm kčĩūI[w=[;Jm7D_V*8գPP.w m'W *reOu)89K]UGi?ka*|uižJ(e|8wZndΔe_ jZo]SZo+υ-nJ\Ks# x7>#>u+J +M 2bb3U3XWcw%+F{nw a(ЕZrrT7Rs:9JO3Xsl,bU*tcK º؊Rj:|ҬU:ҝn4lg?/3O j_+5?=zhFKI$^Ot"5cj,4i6cz5hVNXXBYT'Fp'JK'?fFXάpQ^ qRakOF5ԨTʵjIa)a*Eξ G $n_/~:[/t62ɍޡ |g buxO~iathGCF_k::j&KFOҫF5!ϗ5hVg10LƵ5*gpkc^ ڴq^ҭW(Ž /߾x U.?=lxkk߆yNiZm/S׻{}2z(:6q{[xH3VҞ:J2Q:jR,EtoLux[3&9IB(׭F47VXcTt}z ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@cl|`H98< Ӱ::pܪ}eקw{ӥt9SvA9,2gtԞUwܕ5QqoDҕ].yIj*эKBR4;x5 >Yn=CHk$}T2 EY!p8Ti4IOFjT!+I9E uU9qt*S)8Ni8s4g(}Բa6s玼5#:Ρk|E ,ZoWGڽ[ d]WE U&Ru tX`ҵ-v]3IyMKPEѵY%[M*gk+F~K/H888~MA?89gP( 5]WK4K\5+ Dt[W5[};JҴ6 n KR/$ kHekHH 'ƚU瀴ψ御-~4-~ omWZ\ >u]o@4]k?<%i|r sm4- Km6B-R٭CkzO`}:vzA?L!dqƀ:{cc ǧ8/ |L𗋼 qò>,?Big|k>>|P&Lt~mom0o~O`}:vzA#>V*RotK[k[cTJuٴ*GԵ]Wk{a,K`J~=PGAT0om9_ x¾V 7@MqQHwˤx_L58Mim i$ .wOAFNN=xA'`z23 Qs?$m(jizjZ~hjvizuw^K}5$qE 1gcԾ,xk c? |i 7X/c+RO1cj:vGvFPǦٮe[K9wڮozw5jWP:Ncm%ie{WNWW&8ܵ 64:{cGHWT' t4:{c[H2 ]Ahn 1kG7L6A֞p2xL./a^xW T$+sAsBQ;{ikAa1ޖ#TOQ*\td/3|it~:8qՓMRm7}ZvL+ew~$iꕷ)a -zP݆6m_ZZn--num.T Ư^|#䄔%$$fќNr%i?g>JUJ~O8Jt)8ҩ($% F\?H0Qr1{ZM/doK6~ɿ)( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@P|H~*+m;UP^mgsucr嶓q76sӮE N'z`Dbi*TkV3a"ʤ%EN*-{8>gxOSþ*t-3([÷gj^0#,e |fR⫼.._Y T%J(UJ/֫*~ҥ,<'i1l.w,^Yse'eO Ѕj㽍jbW\«Fsq|.-[^(KV5]WU5]JĖ%>.}WǢkVukkk!jqF2yc4pcG1Q7ERaB51ZU<R̡O1-*8iSi㎣ 00r=<=8zT凕Lr:\Ɵ/?g j~ huO|KO:{4=c@Qhz]lMXxɫR5*NP <%*~5< EjUN,=<%nZT]gNխWWwV^oEcফ߇_ !͟–[Ӿx?>&/~44^5NKz?ZtwDz]/2]OMzuO>.u,~ڥmS| 'A'I]c{|2׵m 5t;9|4ZVMhjӷ'jUcgWt4y/P{;gZDci7Wzo&º_{gŭ'TΩ_qj(;.<o7Zqe5|SWKk-_H.RyȘ_oI(ks\x -ܿumCƾ(|+^:Ŭz߂~$Z4lX |x$'Oo]B~>;д:Ÿ#ެ5?{߉4 ޝ C jRm^z-.[7> |J?eǞ k m .Z} ?x_GǞ>aJݏ.!o}"w]h|X<a~? nExCG|I[z,|E7L-m]VuwZ~i xkP{7^,|'o=>w-5?PX;W^uI^ޣp˛q^G\nl%V_<}߃:.Skκ~>xo Y|&|+]^ 榗|oh^za{|̗]CY7mC3w_&y]o'*x?4kWv,[zZwZ.dM,; K@/ϠX6٧ſto־n5Ğ&7}6xխu薚=ŕ9ieh4fEw4 xGK񇃵^t+O4h;m>^ kx]k?CW]u:)߇Z#YS|;ձ_j.ߋ~#|-xt=4oIk_OYMZk4[-1)-7Z6<5-~Ɵfnnѭj.>*kbV? !Z-(1xKϋ$-,5'm+iG=Ǎ/#h6_k x×Z7KEUg>dg~|*6E [MoNI\#~+|>w7 tYx:?5/Vvp1iN|̗S|y ~Oٶ -xvoٿM5]OUMY|;sÞ>t[j>=t {\Kx#VA]8׼IψO xK(^H_nĞ347@OuxROΥ[iMAQlc~Gú[φ1EVF$קL߄{e/uk8~$mojSRma~'@x+>(Mxҭ mt j$X&:Ө|:;x)mkhɪOau{}n-m\ ; W݇# 3 gxB|wx}]inɬi&_ &f|CoK^sq6IW5W]N4>(Ѽ xJRXĚϏG"z៏|)GּC k 敧\6]b9OӴ}~ˡ~?ӯxǞ垯߆ڷ|w΍$Oc-f-}ibk^+wJ[w|/|UsF3xCž>? >FŚ~x7> u~3Y:ׇ_ gtTu}A,ZU8fu#{Oj~=cO  |o-S|Fd^icÚj2\ZGKtQ~c?i^/**J 5wxY|7cXO7BN1ٞ|?|am< &?dh3&c&⯆v>_´Z^k hVỻۉ|8+QCFOxB\i.ŗZ*-{8>gxOSþ*t-3([÷gj^0#,B#:YxˋiB RJhҩuʄbK ⦱1<#r˱8RxFT:|]VN;֯,=L?):phU*d`w7.>g:O"ռEℹo\5u]sUԭ^|Io{{]cڼUy,z&aQv6il!c'M1MF4~StU)F!QmեQJSK*^}<^:qYң<80KK G+QNXyTg/SyVv _7wh6s< hFLS%}NV<ɌU*JaBTU?*M|ϞVjK OJ/խUs?K=t׬p ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.}_jvSIn2I'^)FEԧR: ƝES\jқQ|)N Qpg$c6_RYǩZ__s ]isi?=ƭC4ț RQsQj)E~N\ӌ48AR䨵Lj^JU98ʔeQBU=S:79Is玼5#:֡i|E5+I2VKMO:[[jRGߛXo馺oYJϕ+ZZM+]&h[D5]+NҬ->]Io/-mg5H4G\LҡT}KPEѵ-4ۆ  yzuƓgXXkNky{αEZf 򬚖6on(ү.64g4t/z1{pmsߍGrrpOR3G ~zkah-徝iZfZG25$qNUA#I{-3&-~3-~ o}SZ]ƍAx/.M:qrYg8m)՝B5J|/ck@G|YƝo|tOiy0KV|cI|1j߇c=6['i|r>y[}6=Bms$Zm.%[Ky{|ޜ#$pIntЎN qqp3] :t*A®Ѹq@$xkg][_ͧ%G_,P;xx-{AF{,{yd%wkhU_s’< /:{ -ܒ pH'o+OGppcAn9'1ހzytPF8<8^9 c$t'yX^Z淩i.ij~ize]pi6QK4SPJic?š%>ߊ? Gh#+ScNmĚIcx%_&XmǶ@zHbGCyF6}步Uk:%孭ƱAj:f zz._E.HXm%t?@ RztN$l~a|ox^exW7u]~M5iTwˤ_L5)H77wZK7Sm*N׎?9A88l̂_8pp~lO^O `cu GՁۆyXj^Z滩h&cwkƭ{ozV[=u4HG/$1~gsgZO'xxBjST4w?<%ez* RStK, е{,t˻ i#: ZWZ֋ukRxڥ}{T4O+WHA3[$b` ^!m:c=F9} n`/'K ۞r6@:G\Qlgl8z qx{y$8Vtc@>rOP]\tQ8OA0n;Y'ug{ZUh{Zui.gdz&v_ֽIYY =+Pכ[C5mdVK\^Yxj:ȴ:MΝ"}[ 㜣ʺ5SU*({:N7~IQ"ΥԋS\۬6&Ua+F7R^A’;ua:rnoi7%%xWA5_D4[GMo8!- o1-4VVIʝOc9U>xKN}c?) M[iݧZK}znEn׾.7oE:{w㌑Oһg[iV-WEe[k!P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠 (ξ$x?Uv bx/U63 oxIk9wi"ц?ޓ=0O_14\R\ӵC0kVrTeRudpՒlGUέ]CKX9IF<'U2մ/^wM<5-6}V 6"4 YqV8}fQ<<+WUjTFSUhNt%%ZBu=|6yfgqX-zyOѝJj&x=Jhb1QZl5Z׵իsXR3'vm"B\~꺮V/>$.s^*Ֆ=Z [_Y4CUa f#45|Qy8SS3[@x^w|9 |7aY Ěڏ]-/=Cm_h:햫ezoMZRra-SYb+RZuia+rҢ:t^jjz-5_:Uulόޝ6{GW}ώo>tj^#Ь>bӣ:'Ԣ|yzo??=w[#|!vc^,j|sM:-i:uO+MoPG `6q/h|9V;)kgº]YlZH2Ar;DzNS /)/QCX6S^nus$wSj5F]*A-ek"ѥ;b N-Cǁ%?~zޅI־c u #I^Z hPj-eYlpݏ1kUK,8<Ư ,jW|t|O7utCcCLJ) wW_4o+~8gஓw/@>O drjO(ߴ]4{7m}V>-xFuᯌ$7 ];ŏ6msxO}wD,5o+O'4>/ǿF7+|Va[w>h_<|PDŽkiZ~CFui[[5YgB'kl|x$&Oo\D~>:дiŸ#ޭPyw[ká>}jMZ}gjYpݏoMoٺ\I?ߍ01wm෎46 sq_^m|,S|%s|74y-t5_ ]7xlh|?_|/'>#@ksZg-\h7})/SX8^k_ /?>.|wĶ7|9ƍkPޑqS],}?QWa>d+@a/+ >< /S[~-NwM|C o[O=i2kzwOGC^>9fzƿ 9xBNRwK.d_k/~Ͷ8moxó~k%)zow[ݍTԟDΓOhv2]v(xƽO|@>:xS^u?|/BE w~$񞑠|#C—u-bNlvz Zc%4>վ"|0񏌿g}/6>(> 4Z[?|Q >:g'(Q{Y&+k{R4o+ 8g~9?^Gm&ƕmoƾ'+kKT"_44~9>C~ Km^FMR}? xvKqi4t8 %k4xϏo:/~~>8xr:ݗ_M?D;t{7EFO׼ ]xY'mnXiJ?E__!c?XxC{hKsuMcH7WPi0C6x^9m>c)kiJM{oMĺ_| {>$t=i|ACޓ\Z-2 STѢNuu-D'enǘQ|{yž?hB,W|G᷍m;O4\BoGZzn}Z |7U K ᖫ_x.LFtA̻,ÿ Íso? 5WRS=ëz7_þ-솿ԭ|;Y#Zw'g/7t(!xZŷH'Ƃ9c!|QXKjU<-i9SZOa)*3e.x];,=|5dK+RUrFxNR`Rk`G*SۭxQڶJy_x~3IVv ysM#R-GVqxtSIWvv:VצJXzrb9>)^#IשƦ ZU$VӭZse pg:O"ռEℹo\5u]sUԭ^|Io{{]cڼUy,z&aQv6iq3a7ƍ7=TaMTP FiVE({)O.jmO2<V|O:(úQp9SU2{^hUxWM忝 {\<j&'lfӭfO2c$jFJP9Up}J_9ZSAҢ:ta:kU\+kG5#( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (c/G3^W_7(o?|Pc#B>: VEwU~Q}$nK]YnoXEmI{iwg{Ys_Xi0{;K_Iy q>j24j%V0ŤݚISRPQ3QvMIhTtV4*4Nj勚+rrP(27hI?|C8o%𽞃>p>!]J/-7.K-.Kl%'zTMOVi5xԌ#+(AsJ6S'Jru(ק$;Tԡ88U(rPb)NS >j|J]))txA9\ -}$^OfYY((*J6\WMtk}y_ޑѫ:g }<E6䞪FW{/Ů]ZV{^N!oRIY]&ҺZɽ5I6Sj~a}ꗶzfsiPäV4K3l]FNue t)NrP"& o')]Yr.*B rZrP)R)r0)I&J.q/z2A94։G}ҵ[MRWkK]K+At) `r3I''4~ZyX1:`ӓߦyցl?9;ր:~slwǯ׭c#=Go+OoПu8: Z{a{gm^iVwV%- wQ7R۹Y[;Q4FZ6FQsj2NTu '.WFQ}4(SJs (TΩԌ]N~tSiNkw<_^SQW%xS&mW]mg40z}ȺuVVJG^tT W8QXjp9ڌѨ$sC%R)JsUq+֥gRцOR{JJ*xQ'(JLƯ#>WViOx^M&t/xvk^SCķ0cD74=EӯnFU¾"5qg|#0p 1T/d5zu:*UdLV65JU0X9(`N𩃡BQ,EEҥsxa\ejpr ]R~ ~^|+h|uGWz[ (\x)iZT}XYZhiKf4fS)xK V)(Q*`nRu*sԔ ,U\9DG1*:1{F]lE)N5>iVu爪iN[Sxk3ux~5 /}]C_@Cq=wNm4]#H񥤒kn':|wjZrⱵ4K1ƽ+BX,,!NFS8{J%y|Tp~,~gV8(¯O8)b؈TfPUT*aZL$ѕG P_ xZx÷/v?  GÛ dP3_KYp[^YP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(=F;8PrnrH it2v:4.j-DkT}6KF6ݔ,"ҤTd=BO&;[7M5ƭ<HRI 5~eMu$Ҕ!)rRn)>XʤaRQWj2vRkEF:*g5ϚpNT)!Qҷ1ۦ>RK`tڄQFRtHҵ$e^۵_R2N?{sNnw{kW|y^IID1u5 BHuKM/Lu KR.`Ӭ,Pt XYfuHㄳB*ʝ*pIQ!'6NU?yY.[dʣJj*!J2u*NmF1+G6RrQvN;zOoL[srd[^vwwvOm"nm&wֺە'tz{cА0zgnأvmlDnz DP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(?ڃj:]o5MVgsuF Y[`E[)\7ٕibf Z/Ui&kOLVBʲӳm HG⹒'ϖQsj29]B؟#UE.3*ҜB)2:ܕ#*s9SSjΤ'oq:ox64A]#U_ \;7KRZtiyJ~2[ 5-7ன7¿| 7_|%wj%͇҅<i:Ko diZ*+G1^; t;hJX99*r9ԝJ%'f ,U\9DG1*:1{F]lE)N5>iVu爪iN[Sxk3ux~5 /}]C_@Cq=wNm4]#H񥤒kn':|wjZrⱵ4K1ƽ+BX,,!NFS8{J%y|Tp~,~gV8(¯O8)b؈TfPUT*aZL$ѕG P_ xZx÷/v?  GÛ dP3_KYp[^YP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠1 ` { 6 AF $ ahoct8`qOkinUeֲvi˺}vmmvycq jOM*ѻuJO٨iJI.\xhƥaBmw,7P]E_]z@>eYOsE"Z*SXM[ٴKIG-$9PӌMQU%t\iӓ拋xp>JN,eBɮ{X8t#pq®ztی*i&ޓwM]4ݸI5{kȯ}Qwn.:nT}COJ-NM{O'ӭ;K2mŝέ4\/ڬ5{$#5 ^HI{FS*JѢ_g:8pNVU'493>jYJ09s^׉|bP־"zO-7e*ԴXiS:|=ڭ7bKc5+YḒdܷսڎM[KӮ4;J]I-o/-n5R +Qf4gR#tm^턲 ].i+ yzuƓgXX뺄NkyyomqjiZ.<&G۬ysɵ?Az$qg] 3Vkh}徝iZ^ZjG25$q$NUFM*ZgMrPu ?kb.}=rwtxrHiRILvr,lL)So ^$z;;g= ?p2yc8@l=q1PXv&xKu ٟEq!~4DwյH>(m~]5kў{3 Igbm-;ޟӰ8x@:mbxH:fbg6I|e{ye?C`q0y>;g= oyizuƕgXX:%嵭ƱAj:f Zz.ߵAk\Xm%t?@ g#*zQ6|uᯆ^cua_ kk&tD y8Uv2|kg~Oi~=6>x]u K h;J:ZVx{\35qj ǮG'$b!%Ooc:ӂ2J8/~.>.74t ZOďw!y=nKMCRb!FgdJ)'~/kg]k_<3i \w<[OG{_X5?ȳ:U֝ iln#2ҝe8֨S|=pJ-Jލ,=_WtU*jGRX'C<*S>[H2 ]Ahn 1kG7L6A֞p2xL./a^xW T$+sAsBQ;{ikAa1ޖ#TOQ*\td/3|it~:8qՓMRm7}ZvL+ew~$iꕷ)a -zP݆6m_ZZn--num.T Ư^|#䄔%$$fќNr%i?g>JUJ~O8Jt)8ҩ($% F\?H0Qr1{ZM/doK6~ɿ)( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@P|H~*+m;UP^mgsucr嶓q76sӮE N'z`Dbi*TkV3a"ʤ%EN*-{8>gxOSþ*t-3([÷gj^0#,e |fR⫼.._Y T%J(UJ/֫*~ҥ,<'i1l.w,^Yse'eO Ѕj㽍jbW\«Fsq|.-[^(KV5]WU5]JĖ%>.}WǢkVukkk!jqF2yc4pcG1Q7ERaB51ZU<R̡O1-*8iSi㎣ 00r=<=8zT凕Lr:\Ɵ/?g j~ huO|KO:{4=c@Qhz]lMXxɫR5*NP <%*~5< EjUN,=<%nZT]gNխWWwV^oEcফ߇_ !͟–[Ӿx?>&/~44^5NKz?ZtwDz]/2]OMzuO>.u,~ڥmS| 'A'I]c{|2׵m 5t;9|4ZVMhjӷ'jUcgWt4y/P{;gZDci7Wzo&º_{gŭ'TΩ_qj(;.<o7Zqe5|SWKk-_H.RyȘ_oI(ks\x -ܿumCƾ(|+^:Ŭz߂~$Z4lX |x$'Oo]B~>;д:Ÿ#ެ5?{߉4 ޝ C jRm^z-.[7> |J?eǞ k m .Z} ?x_GǞ>aJݏ.!o}"w]h|X<a~? nExCG|I[z,|E7L-m]VuwZ~i xkP{7^,|'o=>w-5?PX;W^uI^ޣp˛q^G\nl%V_<}߃:.Skκ~>xo Y|&|+]^ 榗|oh^za{|̗]CY7mC3w_&y]o'*x?4kWv,[zZwZ.dM,; K@/ϠX6٧ſto־n5Ğ&7}6xխu薚=ŕ9ieh4fEw4 xGK񇃵^t+O4h;m>^ kx]k?CW]u:)߇Z#YS|;ձ_j.ߋ~#|-xt=4oIk_OYMZk4[-1)-7Z6<5-~Ɵfnnѭj.>*kbV? !Z-(1xKϋ$-,5'm+iG=Ǎ/#h6_k x×Z7KEUg>dg~|*6E [MoNI\#~+|>w7 tYx:?5/Vvp1iN|̗S|y ~Oٶ -xvoٿM5]OUMY|;sÞ>t[j>=t {\Kx#VA]8׼IψO xK(^H_nĞ347@OuxROΥ[iMAQlc~Gú[φ1EVF$קL߄{e/uk8~$mojSRma~'@x+>(Mxҭ mt j$X&:Ө|:;x)mkhɪOau{}n-m\ ; W݇# 3 gxB|wx}]inɬi&_ &f|CoK^sq6IW5W]N65|1{O|N|Oo]Λ7?>?wZ]6x㶒XGy YA-(^mߌx^}?Yi:t:öM|޸,|?GRE|]1|A+K7>|t9~k|6_ŏo5ˋ_k ~*G4H}/毣"=6J I.M~G_x+NC }#Zo ω$ѭ4_Gqsjﴵ19G~<_V'Of<}]o~}s4?:ׂn|2fuxσ/u/.yg8>Xq/?ﵝOS5K~ʿO ^GQK[s| ޱg_oxYִMž.|K^ KkC`=dm1- JO#>$|AxZ7+mwKԼOA>OL햻}x-)^#IשƦ ZU$VӭZse pXQUJ;9eQfXlN*ETu9:X*|s9G QNXj<5k 4s O{rY6oT+0xL,uVڗ//gË>PΖe,*jPBT҅Z4bj:?X*RxLnw9fܲuye9.<&BS5SN\>s :NHox.|{}[? u]W\u+[X^jXGzݭnc1qhqэFJQ`hƛuiTRJO2<V|O:(úQp9SU2{^hUxWM忝 {\<j&'lfӭfO2c$jFJP9Up}J_9ZSAҢ:ta:kU\+kG5#( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@P8nq0R97u+ʓnM=wIs;׍5}\wOHNP(M籹u +Vi-e[MNCqk4vf'p BKBr g$s FtP[Ist {doXӭD&f }h%ɸ""$|7ހzyyizuƕgi7ZNkyykkqjiZ6<kon&mt˒I]?CC ;[<g& r=׆x;=|-W_t ķn]r>5 Jm>9 im +[i犬D4luFFA82N[3 \-דHe<)Q`v@6V(jڶizv.Ʊ^^iyjwEamo<$Q #CjY֓/ Ķ/^0Юt_C5E Yjޡ7JT.K4-^vK2%9Z~H/7V-֡u_JxºmS4K+WG4hzv -d`@r o\m0-X-CG$ 1.GaFQӧ9r8@N {c FW8o+ o+hy8St4=G<<FppFqvN浮_ VVvJwjnmkT՚OE cҵy%tM'hjjvKYuG,o|OKaxKN}c?) M[iݧZK}znEn׾.7oE:{w㌑Oһg[iV-WEe[k!P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠 (ξ$x?Uv bx/U63 oxIk9wi"ц?ޓ=0O_14\R\ӵC0kVrTeRudpՒlGUέ]CKX9IF<'U2մ/^wM<5-6}V 6"4 YqV8}fQ<<+WUjTFSUhNt%%ZBu=|6yfgqX-zyOѝJj&x=Jhb1QZl5Z׵իsXR3'vm"B\~꺮V/>$.s^*Ֆ=Z [_Y4CUa f#45|Qy8SS3[@x^w|9 |7aY Ěڏ]-/=Cm_h:햫ezoMZRra-SYb+RZuia+rҢ:t^jjz-5_:Uulόޝ6{GW}ώo>tj^#Ь>bӣ:'Ԣ|yzo??=w[#|!vc^,j|sM:-i:uO+MoPG `6q/h|9V;)kgº]YlZH2Ar;DzNS /)/QCX6S^nus$wSj5F]*A-ek"ѥ;b N-Cǁ%?~zޅI־c u #I^Z hPj-eYlpݏ1kUK,8<Ư ,jW|t|O7utCcCLJ) wW_4o+~8gஓw/@>O drjO(ߴ]4{7m}V>-xFuᯌ$7 ];ŏ6msxO}wD,5o+O'4>/ǿF7+|Va[w>h_<|PDŽkiZ~CFui[[5YgB'kl|x$&Oo\D~>:дiŸ#ޭPyw[ká>}jMZ}gjYpݏoMoٺ\I?ߍ01wm෎46 sq_^m|,S|%s|74y-t5_ ]7xlh|?_|/'>#@ksZg-\h7})/SX8^k_ /?>.|wĶ7|9ƍkPޑqS],}?QWa>d+@a/+ >< /S[~-NwM|C o[O=i2kzwOGC^>9fzƿ 9xBNRwK.d_k/~Ͷ8moxó~k%)zow[ݍTԟDΓOhv2]v(xƽO|@>:xS^u?|/BE w~$񞑠|#C—u-bNlvz Zc%4>վ"|0񏌿g}/6>(> 4Z[?|Q >:g'(Q{Y&+k{R4o+ 8g~9?^Gm&ƕmoƾ'+kKT"_44~9>C~ Km^FMR}? xvKqi4t8 %k4xϏo:/~~>8xr:ݗ_M?D;t{7EFO׼ ]xY'mnXiJ?E__!c?XxC{hKsuMcH7WPi0C6x^9m>c)kiJM{oMĺ_| {>$t=i|ACޓ\Z-2 STѢNuu-D'enǘQ|{yž?hB,W|G᷍m;O4\BoGZzn}Z |7U K ᖫ_x.LFtA̻,ÿ Íso? 5WRS=ëz7_þ-솿ԭ|;Y#Zw'g/7t(!xZŷH'Ƃ9c!|QXKjU<-i9SZOa)*3e.x];,=|5dK+RUrFxNR`Rk`G*SۭxQڶJy_x~3IVv ysM#R-GVqxtSIWvv:VצJXzrb9>)^#IשƦ ZU$VӭZse pg:O"ռEℹo\5u]sUԭ^|Io{{]cڼUy,z&aQv6iq3a7ƍ7=TaMTP FiVE({)O.jmO2<V|O:(úQp9SU2{^hUxWM忝 {\<j&'lfӭfO2c$jFJP9Up}J_9ZSAҢ:ta:kU\+kG5#( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (c/G3^W_7(o?|Pc#B>: VEwU~Q}$nK]YnoXEmI{iwg{Ys_Xi0{;K_Iy q>j24j%V0ŤݚISRPQ3QvMIhTtV4*4Nj勚+rrP(27hI?|C8o%𽞃>p>!]J/-7.K-.Kl%'zTMOVi5xԌ#+(AsJ6S'Jru(ק$;Tԡ88U(rPb)NS >j|J]))txA9\ -}$^OfYY((*J6\WMtk}y_ޑѫ:g }<E6䞪FW{/Ů]ZV{^N!oRIY]&ҺZɽ5I6Sj~a}ꗶzfsiPäV4K3l]FNue t)NrP"& o')]Yr.*B rZrP)R)r0)I&J.q/z2A94։G}ҵ[MRWkK]K+At) `r3I''4~ZyX1:`ӓߦyցl?9;ր:~slwǯ׭c#=Go+OoПu8: Z{a{gm^iVwV%- wQ7R۹Y[;Q4FZ6FQsj2NTu '.WFQ}4(SJs (TΩԌ]N~tSiNkw<_^SQW%xS&mW]mg40z}ȺuVVJG^tT W8QXjp9ڌѨ$sC%R)JsUq+֥gRцOR{JJ*xQ'(JLƯ#>WViOx^M&t/xvk^SCķ0cD74=EӯnFU¾"5qg|#0p 1T/d5zu:*UdLV65JU0X9(`N𩃡BQ,EEҥsxa\ejpr ]R~ ~^|+h|uGWz[ (\x)iZT}XYZhiKf4fS)xK V)(Q*`nRu*sԔ ,U\9DG1*:1{F]lE)N5>iVu爪iN[Sxk3ux~5 /}]C_@Cq=wNm4]#H񥤒kn':|wjZrⱵ4K1ƽ+BX,,!NFS8{J%y|Tp~,~gV8(¯O8)b؈TfPUT*aZL$ѕG P_ xZx÷/v?  GÛ dP3_KYp[^YP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(=F;8PrnrH it2v:4.j-DkT}6KF6ݔ,"ҤTd=BO&;[7M5ƭ<HRI 5~eMu$Ҕ!)rRn)>XʤaRQWj2vRkEF:*g5ϚpNT)!Qҷ1ۦ>RK`tڄQFRtHҵ$e^۵_R2N?{sNnw{kW|y^IID1u5 BHuKM/Lu KR.`Ӭ,Pt XYfuHㄳB*ʝ*pIQ!'6NU?yY.[dʣJj*!J2u*NmF1+G6RrQvN;zOoL[srd[^vwwvOm"nm&wֺە'tz{cА0zgnأvmlDnz DP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(?ڃj:]o5MVgsuF Y[`E[)\7ٕibf Z/Ui&kOLVBʲӳm HG⹒'ϖQsj29]B؟#UE.3*ҜB)2:ܕ#*s9SSjΤ'oq:ox64A]#U_ \;7KRZtiyJ~2[ 5-7ன7¿| 7_|%wj%͇҅<i:Ko diZ*+G1^; t;hJX99*r9ԝJ%'f ,U\9DG1*:1{F]lE)N5>iVu爪iN[Sxk3ux~5 /}]C_@Cq=wNm4]#H񥤒kn':|wjZrⱵ4K1ƽ+BX,,!NFS8{J%y|Tp~,~gV8(¯O8)b؈TfPUT*aZL$ѕG P_ xZx÷/v?  GÛ dP3_KYp[^YP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠1 ` { 6 AF $ ahoct8`qOkinUeֲvi˺}vmmvycq jOM*ѻuJO٨iJI.\xhƥaBmw,7P]E_]z@>eYOsE"Z*SXM[ٴKIG-$9PӌMQU%t\iӓ拋xp>JN,eBɮ{X8t#pq®ztی*i&ޓwM]4ݸI5{kȯ}Qwn.:nT}COJ-NM{O'ӭ;K2mŝέ4\/ڬ5{$#5 ^HI{FS*JѢ_g:8pNVU'493>jYJ09s^׉|bP־"zO-7e*ԴXiS:|=ڭ7bKc5+YḒdܷսڎM[KӮ4;J]I-o/-n5R +Qf4gR#tm^턲 ].i+ yzuƓgXX뺄NkyyomqjiZ.<&G۬ysɵ?Az$qg] 3Vkh}徝iZ^ZjG25$q$NUFM*ZgMrPu ?kb.}=rwtxrHiRILvr,lL)So ^$z;;g= ?p2yc8@l=q1PXv&xKu ٟEq!~4DwյH>(m~]5kў{3 Igbm-;ޟӰ8x@:mbxH:fbg6I|e{ye?C`q0y>;g= oyizuƕgXX:%嵭ƱAj:f Zz.ߵAk\Xm%t?@ g#*zQ6|uᯆ^cua_ kk&tD y8UvZ[@cFϋ6\x_c/ğxUo o|'=0<#OAz]}gJ5{//Q5 ;*ASZ[Di|]c'F34x^ex7W 4&4$.)4 i6k+GK;H./.LV .9I(VQnJm%̴Wrݬ׳N馣!AǤzBzGյԬ\YjY*Nt47cxWOcWS7 |QtTsRN EC{ѿDpVabST:a:yaRFq7%(hzf kGuoY3^?l:fy ?V͹s"du{ s¼NPr4' [ ۚ ߋM+\X ¥J~ҌԩTrS&=9|qNOKkЎHqx Eemmfji蕢ծtӲfVi](KMi'OTM oto/-/׺/4mJпqiksip2dFV5zs$$)$#6(֌ s!+I9\*U=SP8|U} M'yƕIA'y(MJ4"A\@t둌%-W?i~h${|ZYMHA@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P߱d+sLi\7qƾx ( ۛ/-nIvr-jsm>#%~O Uʥ%;Z9# l<'%FU!/z*qVKW Y*ʱp5\ѵT1tkSʤe8N<}%`x>/CW.ï|4K=n wAFEJt; |g_ ԴԮ4r-*]llǤum9=\ .#^8Ԝ˟ukEJe3Z.L5jՕ 5f8 ZTjԪWuRYTѯJq8i׫Ԕƶ'jȲMu_ᦝ|!|=iQo~K=s7|MVx&iE[=;Rqg(UcK2]qr5(R*UBU1~εYPS)a.+P_kWZ|4*^20u;3'vm"B\~꺮V/>$.s^*Ֆ=Z [_Y4CU6b1&Ѧ㇣?Y)* э6Ҩe)%R/>e y/W8iQJ OuQ%t%Ʉr,45|Qy8SS3[@x^w|9 |7aY Ěڏ]-/=Cm_h:햫ezoMZRra-SYb+RZuia+rҢ:t^jjz-5_:Uulόޝ6{GW}ώo>tj^#Ь>bӣ:'Ԣ|yzo??=w[#|!vc^,j|sM:-i:uO+MoPG `6q/h|9V;)kgº]YlZH2Ar;DzNS /)/QCX6S^nus$wSj5F]*A-ek"ѥ;b N-Cǁ%?~zޅI־c u #I^Z hPj-eYlpݏ1kUK,8<Ư ,jW|t|O7utCcCLJ) wW_4o+~8gஓw/@>O drjO(ߴ]4{7m}V>-xFuᯌ$7 ];ŏ6msxO}wD,5o+O'4>/ǿF7+|Va[w>h_<|PDŽkiZ~CFui[[5YgB'kl|x$&Oo\D~>:дiŸ#ޭPyw[ká>}jMZ}gjYpݏoMoٺ\I?ߍ01wm෎46 sq_^m|,S|%s|74y-t5_ ]7xlh|?_|/'>#@ksZg-\h7})/SX8^k_ /?>.|wĶ7|9ƍkPޑqS],}?QWa>d+@a/+ >< /S[~-NwM|C o[O=i2kzwOGC^>9fzƿ 9xBNRwK.d_k/~Ͷ8moxó~k%)zow[ݍTԟDΓOhv2]v(xƽO|@>:xS^u?|/BE w~$񞑠|#C—u-bNlvz Zc%4>վ"|0񏌿g}/6>(> 4Z[?|Q >:g'(Q{Y&+k{R4o+ 8g~9?^Gm&ƕmoƾ'+kKT"_44~9>C~ Km^FMR}? xvKqi4t8 %k4xϏo:/~~>8xr:ݗ_M?D;t{7EFO׼ ]xY'mnXiJ?E__!c?XxC{hKsuMcH7WPi0C6x^9m>c)kiJMwߴ?}ϖGO>-i%+/k +M$(u!G~cǎ 6߉)h^7 k9xO"x~9kZ>Ӯn~)9au >5<0W]v:;se/")Yu?_u|]^3xk5_ ~:54}+Gح% u> Ov>#6t:Mg ?M熼m>)>9@to^&o4Ku^kz5Ks%#߃"߳u_m4YWzJJohZpkUtB]Mm heþ嶥6'H+jY'Ծ$|"䵱.>P>-njů[Rǟ ~rk_^͎xqr'<Ÿ-7} կRҞ_? xG-/> ƅ^ӌ^~Z=eh]wKǎ$|AO ֡o YN{| <1FZj w [nDSҼ-s{˽F7$⼏NO6<9]z7lt#\ehhzJ&=-'ǺxH7Z.X޾Ǘ> Hu_ AռxZdY5C:Vh/<=sۀRnt4'_Ul<'%Fu!%8[ar^UV9k5 (BS)ӝxQ^M+I|<ϴ'M[[ x/Λk4KL[ ZLJ-uM&j _Z ;^U)acqxUxDe'^ox*uiTXwNjM]+QVU*PN DX;bB\sJGC6x9ƶ#Lƿx3-)7WRKo UhkgմOik$v"ԒLqtp-FU~.\F_jtbGVRNt!(X嗼uEaӕ8biR rb .#FE9a8_ ? 4ӵτ6~?o>*-{8>gxOSþ*t-3([÷gj^0#,B#:YxˋiB RJhҩuʄbK ⦱1<#r˱8RxFT:|]VN;֯,=L?):phU*d`w7.>g:O"ռEℹo\5u]sUԭ^|Io{{]cڼUy,z&aQv6il!c'M1MF4~StU)F!QmեQJSK*^}<^:qYң<80KK G+QNXyTg/SyVv _7wh6s< hFLS%}NV<ɌU*JaBTU?*M|ϞVjK OJ/խUs?K=t׬p ( ( ( ( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.}_jvSIn2I'^)FEԧR: ƝES\jқQ|)N Qpg$c6_RYǩZ__s ]isi?=ƭC4ț RQsQj)E~N\ӌ48AR䨵Lj^JU98ʔeQBU=S:79Is玼5#:֡i|E5+I2VKMO:[[jRGߛXo馺oYJϕ+ZZM+]&h[D5]+NҬ->]Io/-mg5H4G\LҡT}KPEѵ-4ۆ  yzuƓgXXkNky{αEZf 򬚖6on(ү.64g4t/z1{pmsߍGrrpOR3G ~zkah-徝iZfZG25$qNUA#I{-3&-~3-~ o}SZ]ƍAx/.M:qrYg8m)՝B5J|/ck@G|YƝo|tOiy0KV|cI|1j߇c=6['i|r>y[}6=Bms$Zm.%[Ky{|ޜ#$pIntЎN qqp3] :t*A®Ѹq@$xkg][_ͧ%G_,P;xx-{AF{,{yd%wkhU_s’< /:{ -ܒ pH'o+OGppcAn9'1ހzytPF8<8^9 c$t'yX^Z淩i.ij~ize]pi6QK4SPJic?š%>ߊ? Gh#+ScNmĚIcx%_&XmǶ@zHbGCyF6}步Uk:%孭ƱAj:f zz._E.HXm%t?@ RztN$l~a|ox^exW7u]~M5iTwˤ_L5)H77wZK7Sm*N׎?9A88l̂_8pp~lO^O `cu GՁۆyXj^Z滩h&cwkƭ{ozV[=u4HG/$1~gsgZO'xxBjST4w?<%ez* RStK, е{,t˻ i#9 ZW:։w*uoº}StO+WHogo&G` o;cVWS(* y9'ԡʽxq.,2wERu't9i(¥9S忴*nJQ_B5_D:ޑjF|Ah{idX. ц; Z֫I)Ҝ˒N*pn1rx蜓Mk\F# b>V'VJ=QJTpS,!ҤT^ҧӭK2mťέ4Hd-VFV0)>HIBRIFmNQ(BVs蹈T{9T*q_GNN<*NPiE΀$?ۅ]^Ti]Z$I:{`\qF˗'wD_mkZz~}q{NRjK?5$EkZmmz{~ :Lɥqz:ZZ%ϺZZ+%`>0hM[律EOmj6.~v.~:{w㌑Owӵ6]]mP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>: (:WTVv-Tψ69Km&[nmݧ\FzO>߀UrIsN֭g9E[ QHKފyՒVJu W:mw ].o`%h:Nue a*.xZ.u߂V,|9 _t{gVxVм9{7@׊:~kcKheZK0؈)fJX%sGF]URN)U9Дj xaSb0u>Fu)x*aFjհkS^V5c =J>\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u V#1d 4iƏb n(„0j4cM)CJx sTK:y)8u\G *x<-<2kzwOGC^>9fzƿ 9xBNR1KXxoriڎŏ{Tm|=ρ4_#i> louoZ&g/&Ju-mZv~h|]5O:)~&jֵ xnM>Y~=ޚ5i輎 uۿ _'Ե_V~ʵllnƟ>sٞ|s/Hm&M$ӛyXW\Kl9< =6B5gp%ǀqMQX,u w-ek"[o6>-:uO*LoPG `NtKxEΠYMxυtX[UċF|;|9)O߇z'ZS|;Տ&!~^x3|[&{ӡh~@jM|"Sekv>FV/G쳯 ᭿%_sxW @xַ_4]۱~%#>X뽍|?{⟂5]#w[_<ѼcosែM߈soho> x/\i_t*U.NV7qvx [4.[Z׆3x&t<ڵΡ=Gռ_g4>@cL~}[>.o|Hyi~0vBw񮙥i& ֻgm^xKmoK֍g~밝G]x?sB<}k? |z;д:Ÿ#ެ5?{߉4 ޝ C jRm^z--Ue5~wĚ奆;dO7 m?%t Ms>orFch=_J,{|̗]SUoïW_fHK||ɭ <_k|q{\n/'G= -:;x~=I/zo<}ԼI6x᷅!7_𦫩? K/|{xsQg^nv4mSR}Ƿ:N>|uء|Jӿc? 2 tG>\kkoEGălEgĘJjZM4ljú/#ֿhxG U୮-S_ċ|`1@zu_tO%-MzM5I ./oŧl|7>>Ѿ!iE`v^-6ڷ^G~;/ ?^&ssaug ˑaa*۰$}xaZO|Y /5#+c]A |m x;O牴ž+Ѯn>٦I*淑74?~=7 sPܞ Oj~?QҾ# kzNsj_ &ҵMSF};]oGDcG¾! H׾=^i^)k߆7մ> |s 񖗨ΩqAt;sH562]l|?UZ3h5.< MCdZn]aXiV߅#; >1[ЪyX\u x'~!9/-2,o]k_5?^.e6ҵ\x[֦:1>".5 -sx/ Zm`|o+W  ̞ 3_\ ~=c6^hi4_ m?]j_/"si 厏MFa.w`qTN/-h9$y_Cd OHrΉ5PP@錎 xpp03M[k5W]' EkTtIm-ue)a%ZݝgIuaOWlY.5}&)D1ӢspXRvi&qNIJ*MBmFE7%QӕXӜӜa9K.jNB|ʜU&4Xj>%,἖z Bxu+ý=F1Бg;_S-B~t9 'ր:{tq#O( cnqՉot8GC\2swim,=p6-qL'֍`ׯO(OI#=OlS]oozMu;ZKK4Zy~vq{NRjK?5$EkZmm`0 '0wVO4拏wi4^{z_l$c4]vk WEe[k!P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P߱d+sLi\7qƾx (k5흶y\]YZ|z|d41GKwnfolgDMh%E9S&؟#_UEhҔN)Ό+œ)R:R1w9)ӟ%M8MqizOuMG_JNk>[]OOę_v=xYOvE"qYZFM+AyR1_WEa9 j3FkHG(:UįZoJ?3FLK<JsK )*G*3t^^Z5aҩ½Lts㾱,,aZ0KsKFՃ)IԩRRC4*gԱUsx.X 0ĪTG u8tYמ"u:o>iMٮ4ρ<u_gƞ4Vuj~}{Lj;]t#ƖI30Ej-iˊXhӞm,jЭ b:ON*NRN~X A<0b!S1B!VkPj0RFU%B|)kIܿ_,t+/&_l'7eC .Agn'P4b(T&Zu*tlLWCjC/j(Э_3cajk9QUIʼiQ2+Z.QS3l_|2]N4iW~z$$׿ 0.Tҵ G^w%ePumV7c\gIgMcqrZQZi5! n8ƩF1Ӡ(ygP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠OL@9 1ӰÌt3צ@3к+ݨOuRVi-|ZvSPJ=RM ?LOmlA4֖ ,e }J$(7MPJPUJ4c*IE_IIƓt:q[g>i.I8GS(pG+J8tnI-jF[KMKK"?;J׌m{{nN2>}H;H:9N;J{EVWM^?{'m'-o[85 "U/43K5-KPNKB+KX-cie#˃UT*tU')FW4۵:qT'dn**S֓p(ԩ9KSgIE=:8du=2n[ Qm{)ߕ?܉nmPM׳ZnTOEB@1ݻbn}{s5=W[7e{)P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠j1fav6ѿ=2}[G5 /*NYH&"KkGKH>YE.Tu ?b|}TeH0Js (TsrT_4TNM{:= {kiMٮ4ρ<u_gƞ4Vuj~}{Lj;]t#ƖI30Ej-iˊXhӞm,jЭ b:ON*NRN~X A<0b!S1B!VkPj0RFU%B|)kIܿ_,t+/&_l'7eC .Agn'P4b(T&Zu*tlLWCjC/j(Э_3cajk9QUIʼiQ2+Z.QS3l_|2]N4iW~z$$׿ 0.Tҵ G^w%ePumV7c\gIgMcqrZQZi5! n8ƩF1Ӡ(ygP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>:8`11(6$2rp =y AattA=UZ˯OAEK.e۷rw=ărXd;=7wkhFk%[+9>k7fމ+$]r;U: iwj|Auz}w2I0mgwgs>dBjpO5a5ofi-%xtB[N2M7FFVg qNO..§+*ӕ:f &cqЌ n0IdZzM5tҖ/v'tSwM"iF-ݸQ >]*K8;K=5K?LNm,v: Ңypj2|#p'-y#7'9N(BVsF}sTU9ZTSpphPSQe(l5xk^%GuBZ>Xi?xR/x@:mbxH:fbg6I|e{ye/|_I|/ށi׀~ xK~xi[}6-Ams$Zm.%[Ky{|ޟA8t<'o~C?t@AcOQp^/x/e3f|Y#V|?} x-w@լ?F{,6$k#zN󎣃1KoA`x g8I o+OAFNN=xO08)fy(;kc?š%>߉ Gh#+ƍ}M#Θ %.a-& A8t<-F}步Uacwˤ薷ַƩiT3ʏj6~Yyra4:{<8aF r>=׆x;>|+nMҢSH𾙨j3iq-%VIۢ]6: ẑO$dg'8HalH8?@Pu]/@-s\ԴDѬ/5]_Xo-/Ji.OP(4 kh'kHweTb|X׆:XKuVw}}Zm%W._u{^VIY}2 zN.l[YmJiZY埈yEҮo|OKcq7t5}:>qQSp{eEAo'5$[9W.%եXzXT:N-%y{XT*|eMJ7R/ZZ\3̩5;E^%>G (JI>Hͩ1s9œJ~}1JOg:%N2+p_hSIǞqRPIJqR9H>`:c8 KeO_"+m%RP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>: (:WTVv-Tψ69Km&[nmݧ\FzO>߀UrIsN֭g9E[ QHKފyՒVJu W:mw ].o`%h:Nue a*.PU5Ft[ChdҴ]ºLJ5-95+'\Jt->?+[NOW/C WgN5'*r,ƝZ~ҥG:}b3cVS Z5eBG<+4C>V.Zy*]TVU19etkҪNu5%:,]h~iklZ|T[߄p|=\x'ź|U^ ZgQcoVNԼa}g{?GY<q̥Wx>\\MJJzPFL_VT'GJXxO5ͧc3Z\Yv:OʞW ){凩'V WN 'I][yP>=kjω-o|K|\W5eD֬#=nB 7Px㍄1d 4iƏb n(„0j4cM)CJx sTKϧBcU+>ZTpҧFa a ra(j8zx{p*Lt? u^~>/>.$t_ %x"+oizǃ#&6KKF}We^VjTFxKjT?gVk6xԫVZXzx(JܴΝ;E/q׫Z<ދCMWſ]|C?"-&|$||M^?>x;isikÚ+;8~(_#dO']g/|]//n߲M9uϋZNS o(/QCX6w \x 7-nkY{b[-oV- \o1|-ޓxT/¨u |P D.4[~\ Ŕڇ|QWJuYlZHiaرcI|O⟇޻}wh>?u>GýXkG<ſh>:Ὲ8ڽ'+ Yl=V[!h\6co |o=$~:<]Q:u|~7<}ku>K-]=+>&ෂXo _ZwNҭ5Rc{ǎ$|AO ֡o YN{| <1FZj w [nDSҼ-s{˽F7$⼎xK୧ xGÿh?t]Oui>)W+|0mռ?O~"%ѭ|%xk7Ѿkv}焼=? hЇ [$u< 7S[>-Fwcƾ >]3ÿGZz hojֿy66Yꚴii6c<[cSx[n>mwG5? xk][a?-?ͧ~xWn4}'Ɵ <9/A_ \_ yoK]. j9s~ƅc8=/>}~"<%x⇏| Iψ:>Ήv76E7JK;T/me-?;ך?|6CKO>-|]--_|?Z<7\|TtŻ O~CxO_?/<G-Aд? @Ơ5&?>XZgB!)| ֵ]/[tj^#Ь>bӣ:'ԝ .Kğm>x[<߳~ uo j#IJwǺ<|ocF5'.|{sZ,̗],Gĭ;;Gqxş4o+_ P{z߉%HM6 /tPuwD_RDׯѓTOZy > ~ᎋ_Nem|O%~?^|k76^!| [vG~g^7ş*}X&7߉5D}?S.x? 5A+N|= :lsiz /C?5a)?٧^7<+ C7uk=_ÿ o> H[4cQ4[R5; d5W<_V'Of<}]o~}s4?:ׂn|2fuxσ/u/.yg8>XZq1xFWƟ | ,.{ƞ 9>>Z/> 6<5kae=Xǩ5 id& O-[G __OUxI|Pou߇>5׋t xUվ$K$\5 gKGM[MMk_ |'࿅i.$.JЭ[wwq^V>j?Ǐ4/sŚ]Nj.;Oxx*-.3WѴOM X<7qہRՔI9֭ g%ȫa)*3 .hǝrW [uW:yn9g5 %_ NSR YsҌwOGմWĚWφ(x;i:N_˝6/il?[äM|+ҵv4R/yNNHߞ50Udҩ'Gjԛ(VU$IJbvTW *1ZZlslFj|?$gZo_]Ro:>=>;MΪkvϫigJHt)`U/R*^*ѥSU ?iRMbcsy\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u W8CO0cFhf*0R(CF4۫J=5JTy)8u\G *x<-<41Y@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P߱d+sLi\7qƾx (;u瞣F)+Y]%~T[wZiK&lvVѫ䓺zEjv:@mww=p\ZIo#*jvvW0+YtZ*ڧ:r#dԗ:|즓TeN2SRN *uc*:3Rgե7̢jR*A(.Il"ҥS^u+KF6{K[[Ji6A&h;Q)KRڜ9[3ipQk:*sq(ʣ䄪{ISg9ug{˖n r5xk>%GuBZkzW-7eZ{ ?JIu#UoI6);̱OM,KٵMtMY*WڴVMOMojVqYZ}ֻ.^ZϬjiZ>C<kon%Zi  IY''U&Ru t kcTIԵɴ&Y5-A4mW^Qk^]mmhiv_۠ c3qfוWU +R5JF4[ _W5[};JҴ: . KS/$  kXekHHݶ2FM.ZgMr CZ%ׄ,gZA=oP^]<,u墳q$b2S);. k_^'7 O?4:% 3_A%.`=+oӱ'_:ƓcSuzmOӾ|5մxl6zYm HK\Kfv9FI g7@88g @$ot0=0T]q8Hc x?|;.OK!Y/<:v%KZd.YV$JЫ[Kj##;7ό%A<ym_t:[=$OV0rO#%۠c $1#<#Am`鎠p2y`q r@HO6CUtKԵoR]F5m_X/-/Ji//=KPxӬmlin8rӶ5J|/ck@G|Wcx@:Ɲo|'44LKA,Mża$l6Ď&h(~xJWiTΧvXAjXwQ-AF|uῈoP״m[@K]W_iڟW麮E,fIf.Buzs1 ^O&-@ӷ< m^u`?Nbp < H:q)|9S|}p9䞠z4 `鎣q`>qvNޫJӲ\_UM^ҿ-{jI zV6& `jԬK?u?ht: :D49GuTk)즪TPto)E˝J.哩 &XlLRì=_W 5Jn)΃%vtN5#*nJK kziڷ/,6ekzqCaZ vcSC4[&hIЭR:sN|$ySn\)b(.#qURГ%REJm^=%$j_RYyi}gu Xiwu{if_͞UaTˆMJH$ JJԥ%~Hɩ0s9œ6sJEF^O8NpԚ9AE% r' rjoa9 }A~R\{%uOXݧ/vݯ}-w]oVދwet#9wӵ[mOK B ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@P|H~*+m;UP^mgsucr嶓q76sӮE N'z`Dbi*TkV3a"ʤ%ENgu/:c=}3O{ OAӼeh^xkvZm?YG~olDizG%Nq9գY֣VxxVԩRМJJGNzg:O"ռEℹo\5u]sUԭ^|Io{{]cڼUy,z&aQv6iF2yc4pcG1Q7ERaB51ZU<R<^:qYң<80KK G+QNXyTg/S|ikqgwT'Ľ~Ts,>o+[xL<4 Z_z6u-V%<#R 0[R!k:_9VZBVEt){Zuq|e6Z6> j-u )o5;'#mr!aK3M|=e_ԼGY|ŧGptOǩE%ߊ<?z9xG_CGQᎽY6>/tyj67-{VГ]CEj6;yqS̿h 4 Y.~ t|5mZO7&{ssK?ۿ \\\xMt^G}⿅:?/\ZheZ66~ ~7OO9 ~uGG6uwiͼ+%G|ZuOW~F⏊m^o&suwS_:υtk~ hd-|lS:~G_S_x↰l:%ǀqPH,_?/<G-Aд? @Ơ5&?>XZg Bᵻ#xcī}O#Yyφzů9мo |?qy[_XԮm?G,n|uƇŏxS>=Ovh;-h1W q]&^974}7ğxBSxZ4/ۺuޕi_p'+8#nq2Seÿpj wZ$k]7 'uv_m;A>&IOg]c&%7kM¾ujiqVZKǦ|u5~6?c?|Q;j5ϋhh/oE"x⧉sIOƺniżGŧxծMD#Pߴ|MBk|k|[F-ލk-V_IowhwmZP^iYXj/V N1h|&_?fCFoW>xG>$|<4x;Pyu; LҴA]/<%භkF΄?uuNZ#IM⟇޹}uh>5>Gý[5?=ׇC|7FxԛVɱTզKOᵻ ߃"߳u_m4YWzJJohZpkUtA̻o4/ |3- <{_ZO|Gց]д [ѵB)o~R_ڦ{k(qo9־ _~*|]kloh>2s֡"⦻-oX~SŚ|oV1Û^V}WC>|d/߂>8⽇x .|s7A~sRggs۝'@Zg>dPg>%i߱?Í{Ğ,|t𧄼Z_ċ߅I=#AtG[.4 ZŶ4&;/gJiO|;|Ea~_im#|P|Ai~5ֵ|}A}zt OǶQxVLV5-piVqcso~_#⏄MǍ*ߍ|OVK/Eiis|=:ï'&}&P엷ip6>Khǟh t_40|pu/ěm[ĉ/#w?ox]OȰӰ}m>C< ~-'z,WƑo1Ю`m>6s|QSO7lms[țu߈t|9xOG'?w|I_zZ'Dҵ/ZdiZE>7{h"[|Nݏ1_Ꮛ|?$Yko/4]5owğh> xKntgTo~ ֹ\yXә.>e*-xK[&x~?~!_Y?_-VKOv7s®4K ]o׆ _aUt.dgQDž<??&.~)> =N~z7_ .⯊Ɵ~З2iZliST_ڟ_qJᇇE_-6>xB,Oxo|^A࿆7^HP4M#H^ןRWN+?ڏfO|U?1\_4}?/6M/8oc{z-WJ]|wG 4/[Ot;cO_4oPxCA5 o</M~|B񧌵ox—Xx^Z|9O xrGCNF;8xZryK# uVRTg8\vXzjW^WRTԎU Z\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u WHa f#ͧ[+x̞dIe*ԍJ% 0 ˪rZ&sgb+ByUEt×u֪%VW֏kG8P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(GN}{1@l81{c4q3=O'=>K2[lolrFwp3Ƣh&mqՄdKO\4iJPRFNQOSʜϒp&8xS'K%m'V5 'LگxAi<`,'t;M"Et븬|#c&VcJp՜sգQJI懵J#RIWK7ΥKEӣ &%ʞ [ Y\UJNPJ_o:/ F}cƞ$-M^27ּ;⧰ÉnaƉn#hz>^,|Djh )a0Ga&*c4^W.jFuT͜lk攪`/rPTSBX K2AQF<52W/>&_ķ>:P|S?'^Ҵm ,-6+[_0iTa^:9RX0S%PBUjSIΤT)?3Xsl,bU*tcK º؊Rj:|ҬU:ҝn4lg?/3O j_+5?=zhFKI$^Ot"5cj,4i6cz5hVNXXBYT'Fp'JK'?fFXάpQ^ qRakOF5ԨTʵjIa)a*Eξ G $n_/~:[/t62ɍޡ |g buxO~iathGCF_k::j&KFOҫF5!ϗ5hVg10LƵ5*gpkc^ ڴq^ҭW(Ž /߾x U.?=lxkk߆yNiZm/S׻{}2z(:6q{[xH3VҞ:J2Q:jR,EtoLux[3&9IB(׭F47VXcTt}z ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PPzw'p:d uh][U~g֩+4lm-ee)ɨXEkI}gyi}esz֟Mjwo kKk[Ky2>jkP(Iӥ(BRj*DS|H¤e$֊GNucIT8j-34$#ΩՔSmBוo:cL |c [evejmf}kIZ6˷k'uu^d{s'r]%vO形+MvQwy67یbja}֚^\eaXY%{p4 ge*U:T*B+Nmڝ8~\|kkFZ)UkI8BdTڌc%)Vmr~pv2:ޙh-ɨunDwJ7z6M(&k٭u*N駢ǡ `OnݱG7}v9ۿ+ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PP3|0th2kjϩDG2RFppRo*N)~? n.4_MS֟>eiggg,$F%յݥԏ s$Oq,Մe*rE>G r>2$\gU9҅xSe*u9*F/T*rUB=HN tm?=5i:ĺFuxm kčĩūI[w=[;Jm7D_V*8գPP.w m'W *reOu)89K]UGi?ka*|uižJ(e|8wZndΔe_ jZo]SZo+υ-nJ\Ks# x7>#>u+J +M 2bb3U3XWcw%+F{nw a(ЕZrrT7Rs:9JO3Xsl,bU*tcK º؊Rj:|ҬU:ҝn4lg?/3O j_+5?=zhFKI$^Ot"5cj,4i6cz5hVNXXBYT'Fp'JK'?fFXάpQ^ qRakOF5ԨTʵjIa)a*Eξ G $n_/~:[/t62ɍޡ |g buxO~iathGCF_k::j&KFOҫF5!ϗ5hVg10LƵ5*gpkc^ ڴq^ҭW(Ž /߾x U.?=lxkk߆yNiZm/S׻{}2z(:6q{[xH3VҞ:J2Q:jR,EtoLux[3&9IB(׭F47VXcTt}z ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@cl|`H98< Ӱ::pܪ}eקw{ӥt9SvA9,2gtԞUwܕ5QqoDҕ].yIj*эKBR4;x5 >Yn=CHk$}T2 EY!p8Ti4IOFjT!+I9E uU9qt*S)8Ni8s4g(}Բa6s玼5#:Ρk|E ,ZoWGڽ[ d]WE U&Ru tX`ҵ-v]3IyMKPEѵY%[M*gk+F~K/H888~MA?89gP( 5]WK4K\5+ Dt[W5[};JҴ6 n KR/$ kHekHH 'ƚU瀴ψ御-~4-~ omWZ\ >u]o@4]k?<%i|r sm4- Km6B-R٭CkzO`}:vzA?L!dqƀ:{cc ǧ8/ |L𗋼 qò>,?Big|k>>|P&Lt~mom0o~O`}:vzA#>V*RotK[k[cTJuٴ*GԵ]Wk{a,K`J~=PGAT0om9_ x¾V 7@MqQHwˤx_L58Mim i$ .wOAFNN=xA'`z23 Qs?$m(jizjZ~hjvizuw^K}5$qE 1gcd>"{m[wm|3|:a>"5ukkvu 赛XtfkM/ݏ]; :N3H-BKnjuMdq_>/x'eυ\|\oon5=SI50Czܖ%C%Ůy-,RN_ x?|;.xfx/ƶ(xH \5{쁣,6M$ vw=:c' ӜFp$NۧN;r@9ozt@$6_Ċ[[4V/{%hN6nwz7tQoLk #V[RV|9syg!jft:9G1e[5}:>qQSp{eEAo'5$[9W.%եXzXT:N-%y{XT*|eMJ7R/ZZ\3̩5;E^%>G (JI>Hͩ1s9œJ~}1JOg:%N2+p_hSIǞqRPIJqR9H>`:c8 KeO_"+m%RP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>: (:WTVv-Tψ69Km&[nmݧ\FzO>߀UrIsN֭g9E[ QHKފyՒVJu W:mw ].o`%h:Nue a*.PU5Ft[ChdҴ]ºLJ5-95+'\Jt->?+[NOW/C WgN5'*r,ƝZ~ҥG:}b3cVS Z5eBG<+4C>V.Zy*]TVU19etkҪNu5%:,]h~iklZ|T[߄p|=\x'ź|U^ ZgQcoVNԼa}g{?GY<q̥Wx>\\MJJzPFL_VT'GJXxO5ͧc3Z\Yv:OʞW ){凩'V WN 'I][yP>=kjω-o|K|\W5eD֬#=nB 7Px㍄1d 4iƏb n(„0j4cM)CJx sTKϧBcU+>ZTpҧFa a ra(j8zx{p*Lt? u^~>/>.$t_ %x"+oizǃ#&6KKF}We^VjTFxKjT?gVk6xԫVZXzx(JܴΝ;E/q׫Z<ދCMWſ]|C?"-&|$||M^?>x;isikÚ+;8~(_#dO']g/|]//n߲M9uϋZNS o(/QCX6w \x 7-nkY{b[-oV- \o1|-ޓxT/¨u |P D.4[~\ Ŕڇ|QWJuYlZHiaرcI|O⟇޻}wh>?u>GýXkG<ſh>:Ὲ8ڽ'+ Yl=V[!h\6co |o=$~:<]Q:u|~7<}ku>K-]=+>&ෂXo _ZwNҭ5Rc{ǎ$|AO ֡o YN{| <1FZj w [nDSҼ-s{˽F7$⼎xK୧ xGÿh?t]Oui>)W+|0mռ?O~"%ѭ|%xk7Ѿkv}焼=? hЇ [$u< 7S[>-Fwcƾ >]3ÿGZz hojֿy66Yꚴii6c<[cSx[n>mwG5? xk][a?-?ͧ~xWn4}'Ɵ <9/A_ \_ yoK]. j9s~ƅc8=/>}~"<%x⇏| Iψ:>Ήv76E7JK;T/me-?;ך?|6CKO>-|]--_|?Z<7\|TtŻ O~CxO_?/<G-Aд? @Ơ5&?>XZgB!)| ֵ]/[tj^#Ь>bӣ:'ԝ .Kğm>x[<߳~ uo j#IJwǺ<|ocF5'.|{sZ,̗],Gĭ;;Gqxş4o+_ P{z߉%HM6 /tPuwD_RDׯѓTOZy > ~ᎋ_Nem|O%~?^|k76^!| [vG~g^7ş*}XxJž4Ѣ7j^_|!>MwWֶuo*qS_>GNw}O?$~.t/?t?--mt_?_qY<',~ ΋.bVwo|=|r"׾Pm࿋/ k-" /ď:VG#]O#<9jN2`Kum/U&᤾ gΓ<MF[-ƏbÖ:d5_tEU*1hr}^S*F"2R7L<Y:IƬ;Z&.+fyI(a'RR"XjU!U® k֥V<[~ڦGO_D<W~ΫGm7S{msZݵYҧ;OjIqM 88J^JqNr.#/h|R{E)':wr:̰}iʜT1 49IbrtpTbr#Rxk/Zi?7֟=n.+P_kWZ|4*^20u;3'vm"B\~꺮V/>$.s^*Ֆ=Z [_Y4CU6b1&Ѧ㇣?Y)* э6Ҩe)%R/>e y/W8iQJ OuQ%t%Ʉr,ͧ[+x̞dIe*ԍJ% 0 ˪rZ&sgb+ByUEt×u֪%VW֏kG8P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>: (qA9asoJWIk5{ߕ'ezwI+{ݵj$mڝP=ƛ{gocs)}:Vd[ʷvݕ$h%DxJ6Μ55%Ο5;)7Fiԣ"SBJXʅNԩY5iM(Z i(J3i1to-/׹o4]Jџ͞VҡdMIZdij( R䔢v.iVs )rTZ&5N%JeJ2!*T)NjY^囂js^ω|bP־"ޕ$@ xw@VOҧt@wHo[)oo͊Ns,SzK7mt]7VEgʕv&wS}[\iVwcuˤvVOiP*>GۉdeVI t:IԬ,nB]'CX"u-rm3IyVMKPMEWZW[|i3]v==86~Cg#9b8'uc?UtJԵsRѴMWVNҴ3NKCR #gy8'wm̠K\Pu ?b.F[ Oo&u8h峜I6،jˡš%>׉ Eh#,ƏcN:'t4L~mom<|%{yE+tp> >]o1i9 |um<- x[m6B-R٭ݼ=CkzN~x8$~:G 'o88 :qL Wh@8 {53^.-_˯fn#/Ɩ~(K΁m V*RntX ҵvm3KyQ=B=F/M"w$,6~O\)=r:@' vx?0^MA<{ |2w<{+ [O:n4*D|/j}r-%VY'nitqd fA/8?6[''p 1:NxS`m Pյm/@-s]4]EյcV+LӭRf : yHF}J?3'^l<_xaa\~ 5Ok;Cns]hZ:eKor_xo?|[+BE״m[@uK]SoiڗW麮i KZv#x ۀ`[0:/[|1H#=c\ۧNspA:q&VV;z.p88h zy'y<\4dեk]ok[;%^D+׶;4!Aǥj:kzJOյK?:Ye]ifx(:}nxaJΓ?oӋΤc'QJMJ78lKK~_*4)R(M9S7DFK kziڷ/,6ekzqCaZ vcSC4[&hIЭR:sN|$ySn\)b(.#qURГ%REJm^=%$j_RYyi}gu Xiwu{if_͞UaTˆMJH$ JJԥ%~Hɩ0s9œ6sJEF^O8NpԚ9AE% r' rjoa9 }A~R\{%uOXݧ/vݯ}-w]oVދwet#9wӵ[mOK B ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@P|H~*+m;UP^mgsucr嶓q76sӮE N'z`Dbi*TkV3a"ʤ%ENgu/:c=}3O{ OAӼeh^xkvZm?YG~olDizG%Nq9գY֣VxxVԩRМJJGNzg:O"ռEℹo\5u]sUԭ^|Io{{]cڼUy,z&aQv6iF2yc4pcG1Q7ERaB51ZU<R<^:qYң<80KK G+QNXyTg/S|ikqgwT'Ľ~Ts,>o+[xL<4 Z_z6u-V%<#R 0[R!k:_9VZBVEt){Zuq|e6Z6> j-u )o5;'#mr!aK3M|=e_ԼGY|ŧGptOǩE%ߊ<?z9xG_CGQᎽY6>/tyj67-{VГ]CEj6;yqS̿h 4 Y.~ t|5mZO7&{ssK?ۿ \\\xMt^G}⿅:?/\ZheZ66~ ~7OO9 ~uGG6uwiͼ+%G|ZuOW~F⏊m^o&suwS_:υtk~ hd-|lS:~G_S_x↰l:%ǀqPH,_?/<G-Aд? @Ơ5&?>XZg Bᵻ#xcī}O#Yyφzů9мo |?qy[_XԮm?G,n|uƇŏxS>=Ovh;-h1W q]&^974}7ğxBSxZ4/ۺuޕi_p'+8#nq2Seÿpj wZ$k]7 'uv_m;A>&IOg]c&%7kM¾ujiqVZKǦ|u5~6?c?|Q;j5ϋhh/oE"x⧉sIOƺniżGŧxծMD#Pߴ|MBk|k|[F-ލk-V_IowhwmZP^iYXj/V N1h|&_?fCFoW>xG>$|<4x;Pyu; LҴA]/<%භkF΄?uuNZ#IM⟇޹}uh>5>Gý[5?=ׇC|7FxԛVɱTզKOᵻ ߃"߳u_m4YWzJJohZpkUtA̻o4/ |3- <{_ZO|Gց]д [ѵB)o~R_ڦ{k(qo9־ _~*|]kloh>2s֡"⦻-oX~SŚ|oV1Û^V}WC>|d/߂>8⽇x .|s7A~sRggs۝'@Zg>dPg>%i߱?Í{Ğ,|t𧄼Z_ċ߅I=#AtG[.4 ZŶ4&;/gJiO|;|Ea~_im#|P|Ai~5ֵ|}A}zt OǶQxVLV5-piVqcso~_#⏄MǍ*ߍ|OVK/Eiis|=:ï'&}&P엷ip6>Khǟh t_40|pu/ěm[ĉ/#w?ox]OȰӰ}m>C< ~-'z,WƑo1Ю`m>6s|QSO7lms[țu߈t|9xOG'?w|I_zZ'Dҵ/ZdiZE>7{h"[|Nݏ1_Ꮛ|?$Yko/4]5owğh> xKntgTo~ ֹ\yXә.>e*-xK[&x~?~!_Y?_-VKOv7s®4K ]o׆ _aUt.dgQDž<??&.~)> =N~z7_ .⯊Ɵ~З2iZliST_ڟ_qJᇇE_-6>xB,Oxo|^A࿆7^HP4M#H^ןRWN+?ڏfO|U?1\_4}?/6M/8oc{z-WJ]|wG 4/[Ot;cO_4oPxCA5 o</M~|B񧌵ox—Xx^Z|9O xrGCNF;8xZryK# uVRTg8\vXzjW^WRTԎU Z\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u WHa f#ͧ[+x̞dIe*ԍJ% 0 ˪rZ&sgb+ByUEt×u֪%VW֏kG8P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(GN}{1@l81{c4q3=O'=>K2[lolrFwp3Ƣh&mqՄdKO\4iJPRFNQOSʜϒp&8xS'K%m'V5 'LگxAi<`,'t;M"Et븬|#c&VcJp՜sգQJI懵J#RIWK7ΥKEӣ &%ʞ [ Y\UJNPJ_o:/ F}cƞ$-M^27ּ;⧰ÉnaƉn#hz>^,|Djh )a0Ga&*c4^W.jFuT͜lk攪`/rPTSBX K2AQF<52W/>&_ķ>:P|S?'^Ҵm ,-6+[_0iTa^:9RX0S%PBUjSIΤT)?3Xsl,bU*tcK º؊Rj:|ҬU:ҝn4lg?/3O j_+5?=zhFKI$^Ot"5cj,4i6cz5hVNXXBYT'Fp'JK'?fFXάpQ^ qRakOF5ԨTʵjIa)a*Eξ G $n_/~:[/t62ɍޡ |g buxO~iathGCF_k::j&KFOҫF5!ϗ5hVg10LƵ5*gpkc^ ڴq^ҭW(Ž /߾x U.?=lxkk߆yNiZm/S׻{}2z(:6q{[xH3VҞ:J2Q:jR,EtoLux[3&9IB(׭F47VXcTt}z ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PPzw'p:d uh][U~g֩+4lm-ee)ɨXEkI}gyi}esz֟Mjwo kKk[Ky2>jkP(Iӥ(BRj*DS|H¤e$֊GNucIT8j-34$#ΩՔSmBוo:cL |c [evejmf}kIZ6˷k'uu^d{s'r]%vO形+MvQwy67یbja}֚^\eaXY%{p4 ge*U:T*B+Nmڝ8~\|kkFZ)UkI8BdTڌc%)Vmr~pv2:ޙh-ɨunDwJ7z6M(&k٭u*N駢ǡ `OnݱG7}v9ۿ+ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PP3|0th2kjϩDG2RFppRo*N)~? n.4_MS֟>eiggg,$F%յݥԏ s$Oq,Մe*rE>G r>2$\gU9҅xSe*u9*F/T*rUB=HN tm?=5i:ĺFuxm kčĩūI[w=[;Jm7D_V*8գPP.w m'W *reOu)89K]UGi?ka*|uižJ(e|8wZndΔe_ jZo]SZo+υ-nJ\Ks# x7>#>u+J +M 2bb3U3XWcw%+F{nw a(ЕZrrT7Rs:9JO3Xsl,bU*tcK º؊Rj:|ҬU:ҝn4lg?/3O j_+5?=zhFKI$^Ot"5cj,4i6cz5hVNXXBYT'Fp'JK'?fFXάpQ^ qRakOF5ԨTʵjIa)a*Eξ G $n_/~:[/t62ɍޡ |g buxO~iathGCF_k::j&KFOҫF5!ϗ5hVg10LƵ5*gpkc^ ڴq^ҭW(Ž /߾x U.?=lxkk߆yNiZm/S׻{}2z(:6q{[xH3VҞ:J2Q:jR,EtoLux[3&9IB(׭F47VXcTt}z ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@cl|`H98< Ӱ::pܪ}eקw{ӥt9SvA9,2gtԞUwܕ5QqoDҕ].yIj*эKBR4;x5 >Yn=CHk$}T2 EY!p8Ti4IOFjT!+I9E uU9qt*S)8Ni8s4g(}Բa6s玼5#:Ρk|E ,ZoWGڽ[ d]WE U&Ru tX`ҵ-v]3IyMKPEѵY%[M*gk+F~K/H888~MA?89gP( 5]WK4K\5+ Dt[W5[};JҴ6 n KR/$ kHekHH 'ƚU瀴ψ御-~4-~ omWZ\ >u]o@4]k?<%i|r sm4- Km6B-R٭CkzO`}:vzA?L!dqƀ:{cc ǧ8/ |L𗋼 qò>,?Big|k>>|P&Lt~mom0o~O`}:vzA#>V*RotK[k[cTJuٴ*GԵ]Wk{a,K`J~=PGAT0om9_ x¾V 7@MqQHwˤx_L58Mim i$ .wOAFNN=xA'`z23 Qs?$m(jizjZ~hjvizuw^K}5$qE 1gcȵ [v7×l _>,񍮪_Oz xhq%Ɓ"o%{|t=7(мcX×ooRIkyZ^iZiz:}_[ZOKo+=q1PXv"gs{{ `q0y>;g=  qHH8x ե⯋I/(YQn&M_GGEf Y i~ѕm|1_n{qj|yvu g[|K9) Vv\^\m7k;v]C;}z r9#06QYww$vݬKh{;Y[gtMGCxAI5푫k-Y +K>5?ȳ:U֝ iln#2ƯGN5*n_=Ol-検sR*~q.,2wERu't9i(¥9S忴*nJQ_@_D֏ fhCztoNd~i {s+DےRyPJiN44%~V4[ b=JRȧFMzr7ʝ6Lӡ'AY=&w+E]d̬һVQnqWOV[rRYǪ^Z__u _isiڕ6VeMɩ,HIBRIFmNQ(BVs蹊T{9T*q_GNN<*NPiE΃޸#Ǿ J[.e4I^oW/- ( ( ( ( ( ( ( ( ( ( ( ( ( ( (c/G3^W_7(o?|P@yď.Ңӵ\Al]|A7V7-^[i7ݞsmg=:Z0|G K&JKvhsl<9iɩ\i:ZUiI_rz~*\F:3:q9S>!f4֋*9g]Zj%9*j?w()qY"RrͩT J-^WpӯW)ՍlO0եeCM;\Cgz'ܖz獼o>-<;wJL"<=mU;zzv ? ?pIPΖe,*jPBT҅Z4bj:?X*RxLnm<#r˱8RxFT:|]VN;֯,=L?):phU*d`w7.>g:O"ռEℹo\5u]sUԭ^|Io{{]cڼUy,z&aQv6il!c'M1MF4~StU)F!QmեQJSK*^}<^:qYң<80KK G+QNXyTg/S|ikqgwT'Ľ~Ts,>o+[xL<4 Z_z6u-V%<#R 0[R!k:_9VZBVEt){Zuq|e6Z6> j-u )o5;'#mr!aK3M|=e_ԼGY|ŧGptOǩE%ߊ<?z9xG_CGQᎽY6>/tyj67-{VГ]CEj6;yqS̿h 4 Y.~ t|5mZO7&{ssK?ۿ \\\xMt^G}⿅:?/\ZheZ66~ ~7OO9 ~uGG6uwiͼ+%G|ZuOW~F⏊m^o&suwS_:υtk~ hd-|lS:~G_S_x↰l:%ǀqPH,_?/<G-Aд? @Ơ5&?>XZg Bᵻ#xcī}O#Yyφzů9мo |?qy[_XԮm?G,n|uƇŏxS>=Ovh;-h1W q]&^974}7ğxBSxZ4/ۺuޕi_p'+8#nq2Seÿpj wZ$k]7 'uv_m;A>&IOg]c&%7kM¾ujiqVZKǦ|u5~6?c?|Q;j5ϋhh/oE"x⧉sIOƺniżGŧxծMD#Pߴ|MBk|k|[F-ލk-V_IowhwmZP^iYXj/V N1h|&_?fCFoW>xG>$|<4x;Pyu; LҴA]/<%භkF΄?uuNZ#IM⟇޹}uh>5>Gý[5?=ׇC|7FxԛVɱTզKOᵻ ߃"߳u_m4YWzJJohZpkUtA̻o4/ |3- <{_ZO|Gց]д [ѵB)o~R_ڦ{k(qo9־ _~*|]kloh>2s֡"⦻-oX~SŚ|oV1Û^V}WC>|d/߂>8⽇x .|s7A~sRggs۝'@Zg>dPg>%i߱?Í{Ğ,|t𧄼Z_ċ߅I=#AtG[.4 ZŶ4&;/gJiO|;|Ea~_im#|P|Ai~5ֵ|}A}zt OǶQxVLV5-piVqcso~_#⏄MǍ*ߍ|OVK/Eiis|=:ï'&}&P엷ip6>Khǟh t_40|pu/ěm[ĉ/#w?ox]OȰӰ}m>C< ~-'z,WƑo1Ю`m>6s|QSO7lms[țuh Ş!-< a4׾'Os=G: "|>[Ӿ-Osk!K,V^VI PCJ-8ǩg>mxS4мu5$uv_m;A>&IOg]c&%7kM¾ujiqVZKǦ9뱡|K=?|Gmuπϋg6~= x|S|s񥖁MiZ7BkZ`KGſ%71u{;iھki˛V=BN1y>[o٫_/<5w ~Z_|,|A >-;? 3д{]PtMW~]/[w?gk Iž/|C<k9xU>(xcAc\={=։&oxZ×z.oIyxK..?Z']ol|mm˯xsƺ/oi/}GѤl,M>{{;?@ZOto /ҵ\}_.|.{}VxP1I,#]մWĚWφ(x;i:N_˝6/il?[äM|+ҵvTR/yNNHߞ50Udҩ'GjԛhVU$IJbvTW *1ZZlslFj|?$gZo_]Ro:>=>;MΪkvϫigJHt)`U/R*^*ѥSU ?iRMbcsy\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u W8CO0cFhf*0R(CF4۫J=5JTy)8u\G *x<-<41Y@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P߱d+sLi\7qƾx (;u瞣F)+Y]%~T[wZiK&lvVѫ䓺zEjv:@mww=p\ZIo#*jvvW0+YtZ*ڧ:r#dԗ:|즓TeN2SRN *uc*:3Rgե7̢jR*A(.Il"ҥS^u+KF6{K[[Ji6A&h;Q)KRڜ9[3ipQk:*sq(ʣ䄪{ISg9ug{˖n r5xk>%GuBZkzW-7eZ{ ?JIu#UoI6);̱OM,KٵMtMY*WڴVMOMojVqYZ}ֻ.^ZϬjiZ>C<kon%Zi  IY''U&Ru t kcTIԵɴ&Y5-A4mW^Qk^]mmhiv_۠ c3qfוWU +R5JF4[ _W5[};JҴ: . KS/$  kXekHHݶ2FM.ZgMr CZ%ׄ,gZA=oP^]<,u墳q$b2S);. k_^'7 O?4:% 3_A%.`=+oӱ'_:ƓcSuzmOӾ|5մxl6zYm HK\Kfv9FI g7@88g @$ot0=0T]q8Hc x?|;.OK!Y/<:v%KZd.YV$JЫ[Kj##;7ό%A<ym_t:[=$OV0rO#%۠c $1#<#Am`鎠p2y`q r@HO6CUtKԵoR]F5m_X/-/Ji//=KPxӬmlin8rӶ5J|/ck@G|Wcx@:Ɲo|'44LKA,Mża$l6Ď&h(~xJWiTΧvXAjXwQ-AF|sᯈou _Tu;]So麟W麮a UM%#ˌv00xp[rH4Am1`ddrǎNn `t^N`ą> 1FӃ0T#:yX[yX?}pxy~lt8( 8b0U<].oh>\|d/߂>8⽇x .|s7A~sRggܚ}/n3{qsgqq{[4kypOkھk+U?>|?K?<(>ö_%֑!MޛI6򰮹kI(kKxE֢YM_>|?Ze*E m&/|[uOU~N}⏊Ƌw/ëA#P5 Wk-_,716vlr< /S[~-NwM|C[K?xB.班n5gicRcKG}H{<=O??k".G'yx_3?tx|^ Mk4 [nzU]}oxxֿo<)w><^v5_ @c⏁]dž?hMO?x-׳hjvWo|9w2W7ۛ/ |O|hw΋'>*u(_h 4 WCׇtH<9LJ_Z>i.^!%c~ P"|Gx>. x{m[?&'?麕{ Vk˙4EK7ƽ ᮣ|"' :io]ķ7z5[ |f' hM,ykkCz{%qeaxiZ|8ǩ~= 5 _ⷀ|] DaCׄ<']3JM7ívO?g:a;]kdǁ&7~z#օxH~lx˿xxw߈ ^A RmZ&k=SV->Cx'~ xo ~ֺ=M.h k l5qYoۛJݍƃog5_:++ᾡ-kU2ooѾ8о xeG?u>GýXkG<ſh>:Ὲ8ڽ'+ Yl=V[(Z6c5z֫z'> j++5K cvoOq}6K'|"#֍RDzoY.ফ߇_ !͟–[Ӿx?>&/~44^5NKz?ZtwDz_!s%x_=xmo xCǞo_.mMWS|~x_\瀣π-hڦ%Ϗnt}kEW3CŞ~~h5xž>?k/~o['M<o3kpcbth)?$o|Qᇌ|e;|?|CZ< ⏉<7?GeZ1[[ڔԵyXi;=χt_ G~ >i74o~5?[]/ZĿ>: c΁+J[h2j]Cò^[O!ۡo/_٣|}|1|C×)Zmo$wc_/6~Mv:ρ>kw"NUaH/ 7д&E_x@_ÛkGWBI>viGOK j.JԾiMjvC=mm!;+v<⏅|Cφ>- Ge|{0ҼS=w׿ oi|A*|9-/Q5RIwwZsk mcNdīfk]/oxĚ'd Y->! Gҭ,-u xGw^|5~cцU򰹒뱟GOXCs^ZxS\O:׀|eY|(׊+Xk~B\ӼeE[‘xKZ\xNm"=CTn-4#zxZ^}I^V%8+j><Wf{omr|">"&ht7~Ծ_XxὍ<Z]_ ]*vdZ6;e`e> O?>>kψ9Oᯊ›_x^#Ѿ5nd5?~okΎ0G(м=m=Ӹﭏ<8|ѼA S5H47EA ƞ2-I _Max|EiA?4$/mO:]PkIʝ.^Zrx5[ IQ/ysa%]zY^: J7/Rr+Ҕc_ R8yTR&hҌ،gOGմWĚWφ(x;i:N_˝6/il?[äM|+ҵv4R/yNNHߞ50Udҩ'Gjԛ(VU$IJbvTW *1ZZlslFjů x?T~fN>^ όN[tƑh/ӾoNm wio--*1PteO(UVZVpQTf)N\𢙒 G3թ ~i~/%Ϗ|ox᮫뚮oK{>Yc5ຏ[񵐃M5]#1d 4iƏb n(„0j4cM)CJx sTKjy)8u\G *x<-<41Y@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠: `fj-O֩%mvw|ZvS}B+m*KH;;BM{LOԯل\jLSΈc;UEkT*N-&M✒Tڍn2KEJ*9Q8sP,\ԝ89[3YE69BMdi&|K Yy-υME# WQxynwOYivz^{a,Q<>+.3~ `q=߯^{c^=~h x9?@yX:{~[ǹtǧ@coN1?td OS'`69 u9otd OS'`6w@Sx9ד6 ~8rOm`r1;PyOlcې01 c i`=yma?9s~'6tL{zG<=yOn:dv? yOl q@G r2ѥ)BJSW9FR>uNb*sS>JMp_ Oꚏ,[_|&ğ3hj={k9ԟM4EӮⲵV=Z(biŠVs gVE)'K*!JPu'^&,:,gN0*x,l-fRUqU+Ž!9BU*f5}(7kOxĶz>"i6+x#Z/%$!A.{p 2<4!Qҧ9{'7\ושR#'6rbRļ CRwL ib*,7,Cg +WSFxoWT֛_x C>5HBǞ OG{JҥGLXحl}|1S5z9K}bXX´aNpB U'%NSu':S7hTϩbͲ'<\aTю#5,&& b)JqJ" "ÏviG-$]{w XV^MoT>Oo&7z%"\ㅊO ֡ ?iQ M}Tؙ/mM?gJ-ԇ>_7QZg<a3s'x/jңexWOJ]\$g 8*ؿ~e-WlhҮII~xa]8>jO^K_ w %Ƭouo)#ZJxbj)(FiJ0RҦ:xz?7tq*U1n|,% ^ӄjBYbqRcAQ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@Ol1ܞ€s;s@cK`铷/Iʓzt=p8{zeܷӛ&4S+ mu(i4fܩ;c==v}۶klkgnz$nwR$( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@9cQmxxɯxBm>m0V J-J̫M;gإ7|o"%|oI5OkZ|zdj^UhMDV׶vRV2XIAB0m\$rq?kUԧS,uuWw̪ͭէ xY*Iji^ҔjՓO:Sbߗ|.QiuO i >g>O+sW-τ.l" "ÏviG-$]{w XV^MoT>Oo&7z%"\ㅊO ֡ ?iQ M}Tؙ/mM?gJ-ԇ>_7QZg<a3s'x/jңexWOJ]\$g 8*ؿ~e-WlhҮII~xa]8>jO^K_ w %Ƭouo)#ZJxbj)(FiJ0RҦ:xz?7tq*U1n|,% ^ӄjBYbqRcAQ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.}_oYn,num.D}Ndd7$%>FNZFoN0s2!RP'g*9TӄrN29(ќRQs߷c[v2{MnEuчLv9#0IU 9_ xPtյCLI9>wq[Cqupcx;tCK]q;`ӓ^9IX?ބ==l-v8==s |2{? <-gwMCR|3k?>"\ꚦkbAz_g %Ů\[,wjB_m'c×oүqזvnjVomfg}j:Nki_鷖W]MKo;|w0y#~08IOBJO3jr\hp9BERSΪ?e S%W:pqiTwTNR.t$(N=ARu{-S戲Jť~dynP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p( |]Emjػ M> nnn[Þ4n5=&{]uȴaL M<%W*4jx3UTǝY,=\5d[*QsFqPjQO*;\P^_?ykgL,.t? M+E,||+xsRӓRuȴ wI!մrT0uxftRr.|Biխ*Ts#8ϖ5h0ժKsVT(~QSLD3 iR嫋R^JeS[F*^RS؟aK"5ևv EOg۹,xI7|[xwZ\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u W8CO0cFhf*0R(CF4۫J=5JTy)8u\G *x<-<="hukXmv0jMjRsa/wNU\#/**]ZuJ,5<_/-*.N|Z}~\eUU/-~'/{ Km-f#|b/XYi 5OԵ Gt+_%|3>-W;c̗sBY]U ,7_DƋwU^#MGR׾miji:l7 C/UuW 3(>~.}ix/ki+|mKO+en_C.|]׆IE]G5+>Z_xkPvh7VMm/S4hJ Ek]#FV|̾3 gσ|O֙ix \,m~Zo?AaJ߇ x>Ohfk-3N̗]SUoïW_fHK||ɭ <_k|q{\n/'G= -:;x~=I/zo<}ԼI6x᷅!7_𦫩? K/|{xsQg^nv4mSR}Ƿ:N>|uء|Jӿc? 2 tG>\kkoEGălEgĘJjZM4ljú/#ֿhxG U୮-S_ċ|`1@zu_tO%-MzM5I ./oŧl|7>>Ѿ!iE`v^-6ڷ^G~;/ ?^&ssaug ˑaa*۰$z쟳ϊ?ٟ)xST_k֑x6y'Þ&ÚG[HW.Fѭ+?-mKkiZ_.dtZ'<_K7߉~7a|1K|AwX:(ςo|O擫bΡږ}AEhץaEox6mg4i2?xRtv獵_,)vKi_4OI//!r_۬5}V~K<=O??k".G'yx_3?tx|^ Mk4 [nzUW_p'+8#nq2Seÿpj wZ$k]7 '_O{W쯦7?t>xHF:=s_>>ω^xYO"7ndԞtz&a~^ Kǧ|S xO5GXŸj>,Ck|=ƋxGAu-Z}o&n.+P_kWZ|4*^20u;3'vm"B\~꺮V/>$.s^*Ֆ=Z [_Y4CU6b1&Ѧ㇣?Y)* э6Ҩe)%R/>e y/W8iQJ OuQ%t%Ʉr,ͧ[+x̞dIe*ԍJ% 0 ˪rZ&sgb+ByUEt×u֪%VW֏kG8P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠 (cP{y8l`)RZ^Iŷ{YEokwmemI;[vciqw`w _i:ΕY42ݦawes !⸵ Q'ERMs(B6MI{Ni5MQi$+5(ȺPV2xӨ5*kpZS|/)¤J.L}F+]*[8;K=5a[ .m>RggմfdjVxõ.j-B%(ݩ˚qc9F;J\ISR?gR8HJ8JvqSVpWf)+P8is^xR|3iO~"jz\kwbZ䖗ضZȂK]TYb$Vu>~U7k[>Nt[J=L,KHԭ/: Yv o8pxuleG#h8!N;s98< Agt^~`tf3bv]H=eer2zbyXyzmƓgizFiyyoksjPiZ.<&}%ZWE|I^3{c g Qjr䓊\+^:'$Zb(}W{jXթIա7ROe':5R)4*>T$mo/m/׺/2m>S/\\S̨R5;E{9J|P|SbkFs9.bU*uU)(Je>W*ӄ'kZifKO/ָN={ׯoӊ[[z?YgY$zk_Oo@鞼4N.}WǢkVukkk!jl!c'M1MF4~StU)F!QmեQJSK*Y̡O1-*8iSi㎣ 00r=<=8zT凕Lr:^uA-Ե1|}=sZ7=y<|7džo|6⏇xj~tox>MZRsa-Su\#VZ6xSZuia.^Z4]WN*Ir:k|%}&Ϳ,f?_;[j_n|¾<Œx3>.>m+fu:7WKus%=S>G40Z\^x}3Kd?Ox >j6s',tkn.|t'>I)?eM&ic07qovs|6.l_ĿV_٧/&'o+C[,ikC(Gs5s|._;6CczaTwvgahƗ|/~%ڕK.a`XS> }?ǚNSI:jZm^㿄?%ѧ|!mKm4OxWǞ6|?Oxgù/?m|LҮ='\n[u.|X<a~? nExCG|I[z,|E7L-m]V]}oxxֿo<)w><^v5_ @c⏁]dž?hMO?x-׳hjvWo|9w2W?_.MxF]# |Jo{/>%xg g[? RxGńvVE{|.O'^ όN[tƑh/ӾoNm wio--*1PteO(UVZVpQTf)N\𢙒 G3թ ~i~/%Ϗ|ox᮫뚮oK{>Yc5ຏ[񵐃M5]#1d 4iƏb n(„0j4cM)CJx sTKjy)8u\G *x<-<41Y@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(GN}{: Z{a{gm^iVwV%- wQ7R۹Y[;Q4FZ6FQsj2NTu '.WFQ}4(SJs (TΩԌ]N~tSiNkw<_^SQW%xS&mW]mg40z}ȺuVVJG^tT W8QXjp9ڌѨ$sC%R)JsUq+֥gRцOR{JJ*xQ'(JLƯ#>WViOx^M&t/xvk^SCķ0cD74=EӯnFU¾"5qg|#0p 1T/d5zu:*UdLV65JU0X9(`N𩃡BQ,EEҥsxa\ejpr ]R~ ~^|+h|uGWz[ (\x)iZT}XYZhiKf4fS)xK V)(Q*`nRu*sԔ ,U\9DG1*:1{F]lE)N5>iVu爪iN[Sxk3ux~5 /}]C_@Cq=wNm4]#H񥤒kn':|wjZrⱵ4K1ƽ+BX,,!NFS8{J%y|Tp~,~gV8(¯O8)b؈TfPUT*aZL$ѕG P_ xZx÷/v?  GÛ dP3_KYp[^YP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(=F;8PrnrH it2v:4.j-DkT}6KF6ݔ,"ҤTd=BO&;[7M5ƭ<HRI 5~eMu$Ҕ!)rRn)>XʤaRQWj2vRkEF:*g5ϚpNT)!Qҷ1ۦ>RK`tڄQFRtHҵ$e^۵_R2N?{sNnw{kW|y^IID1u5 BHuKM/Lu KR.`Ӭ,Pt XYfuHㄳB*ʝ*pIQ!'6NU?yY.[dʣJj*!J2u*NmF1+G6RrQvN;zOoL[srd[^vwwvOm"nm&wֺە'tz{cА0zgnأvmlDnz DP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~_fο2oP9p(?ڃj:]o5MVgsuF Y[`E[)\7ٕibf Z/Ui&kOLVBʲӳm HG⹒'ϖQsj29]B؟#UE.3*ҜB)2:ܕ#*s9SSjΤ'oq:ox64A]#U_ \;7KRZtiyJ~2[ 5-7ன7¿| 7_|%wj%͇҅<i:Ko diZ*+G1^; t;hJX99*r9ԝJ%'f ,U\9DG1*:1{F]lE)N5>iVu爪iN[Sxk3ux~5 /}]C_@Cq=wNm4]#H񥤒kn':|wjZrⱵ4K1ƽ+BX,,!NFS8{J%y|Tp~,~gV8(¯O8)b؈TfPUT*aZL$ѕG P_ xZx÷/v?  GÛ dP3_KYp[^YP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠1 ` { 6 AF $ ahoct8`qOkinUeֲvi˺}vmmvycq jOM*ѻuJO٨iJI.\xhƥaBmw,7P]E_]z@>eYOsE"Z*SXM[ٴKIG-$9PӌMQU%t\iӓ拋xp>JN,eBɮ{X8t#pq®ztی*i&ޓwM]4ݸI5{kȯ}Qwn.:nT}COJ-NM{O'ӭ;K2mŝέ4\/ڬ5{$#5 ^HI{FS*JѢ_g:8pNVU'493>jYJ0.{ c<ַutî_>UiՕn^Z0ޝA' q *A~#xkᗃA\i|Mqp:-֥.+PԤӭbfHnZ;[I݇nwO1霑= #̾"|^O» YQҼ/k?"\jk`5%ȊI}R._Rz:.agƓyj^]i<Bʲ#`J^j^qVzVmyyomqjiZ&A4څh6~lm^\041ۦ:#8$q<ϼt:IԬ,.4 Z"ҵ-vM;Ji 4m^VJ*ah;y>:{בH{q1C#c̾"|^W» YxPԴ ZOďikwjzߍRcXeGI/^E_;g=  1C#`펽px9H8㌛yX6BU*PԬ,.B]'Cu8tK]NҡD{]W{{uEүn~3pAC33NIYֻ|)Z:ַ]ZvKkz&v_ֽIYY ='P_[C?Fd4,rCP<"WZu7s٥cʺ{ҝe8֨S|=pJ-Jލ,=_WtU*jGRX'C<*S>[H2 ]Ahn 1kG7L6A֞p2xL./a^xW T$+sAsBQ;{ikAa1ޖ#TOQ*\td/3|it~:8qՓMRm7}ZvL+ew~$iꕷ)a -zP݆6m_ZZn--num.T Ư^|#䄔%$$fќNr%i?g>JUJ~O8Jt)8ҩ($% F\?H0Qr1{ZM/doK6~ɿ)( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PP@_<JNqw<x|XܷYZGa_@kMSBӾj/'j6qVJf=z}ZwǙ/Nx ;O>1Yo_ AF<%#š}EtnA{?A]s#>;:ůٿZ^:AgQ'5c+ǚ=}]?bӢ_d.> V2w7FVhݍhU\|G7SVk]i[S4ַNDHGH%[5o ǁJ-sկ//|HZMixG!7+&oi^#V!+`h"# j?ϋ<_&Da.TaEX-t~'m_A𿆵![x5SGj7'#?/k~6ּG/w^L1&)i|e"-[xYo5,:4xWWX{=;Pw;ǎ$|AO ֡o YN{| <1FZj w [nDSҼ-s{˽FoIy~ 燠g~+ey>}-u 6 L?xsY|#Ociea]4WSC? D4Nj6~/>Ԭ5:_u;> Z9U.>m+fu:7WKuۯ|u<LJ) wW_4o+~8gஓw/@>O ok:߉^̞>D,#/u< ҿfSGŗb-|=㯇8u:n[/ ~$Lj˩|-棨Yiqޗ|u0?hx3//<ᦣ< kLK[UO| & kRuo#X4&x&nm? h:ơG|1/^4mOxRk +O6 1!{oXtiڇ~O ZNT)rփaxJJ{˞N_ Y*԰U\Q}*X6X:ʢ6G.e`:x=$Ҿ|4CL9tյ𝇂\鶱xHԴeU\xr'Tl]ݎR7XOwWFRuF񩇂'VI:8ՇtVB%l5X% $JPKS**xUQw:Ԫt3gkb0Tk'<]="z“|(/&ުomuV[}[K:TGi-I.4ɱ1{yGG RaWN)ReO*Tyoh$B5Y{TY/9S!fU G),NN. ,\Q=jTS eCM;\Cgz'ܖz獼o>-<;wJL"<=mU;zzv ? ?pϮy*1K|}f)P*V*gZN)J*ki1l.w,^Yse'eO Ѕj㽍jbW\«Fsq|.-[^(KV5]WU5]JĖ%>.}WǢkVukkk!jqF2yc4pcG1Q7ERaB51ZU<R̡O1-*8iSi㎣ 00r=<=8zT凕Lr:^闚>kj Syoyo3ڡƉi2['ٴ|o̘,EZRF!?uNU\#kRlEhV*P|Ν>XrZW3乪Mz ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (c/G3^W_7(o?|P@va=G# #S W\KY8k(t3Mx[NW'tnu ?R4;x{.K'Pҵk&FU.a$ieEqHo-Yt9~->.k_ZOm<)O[OTtnLX~K\QbIqk,Z](e×oүNזvw}njVomxLլGImm/: Yv o= y 3F :`~1s造asU611p;yqӾ1`펽`>R:1o+/5]/Nl J]IЭ//-n5N +ReӴf=EduүnB-et?A6hnjb7mnt3( O!:vF1Ѓ郀#=x3NIYֻ|)Z:ַ]ZvKW{Wtk^ڤz,ϭ iCV[R^埈ƣ3\^f9WS(* y9'ԡʽxq.,2wERu't9i(¥9S忴*nJQ\fj'Αؾ :nB ѥZw'~f.gdJoZ֫I)Ҝ˒N*pn1rx蜓Mk\F# b>V'VJ=QJTpSl"徳T^ɴmNпsgksqO2HdFSHIBRIFmNQ(BVs蹉T{9T*q_GNN<*NPiE΃v1;rZM/{ۛK;%t1ӌcH[ ?h'@l&8lqq׽t? c1qp;G OOI#=OlS]oozM읭ki-<;Z8^N)mn5gedBӢ뵭~==LGzOd8w_~-wsDdvz-Ud^u4&խEէ5[hi;~kmK=H'hg[iVmOK B ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PP@_<JNqw<x|XܷVλJŞχXjAOEv'YѼ{m(nAm5jFIaMp]Zj/VNiեyhu]:|%կW^TIwC64|"g|Yiqy⏃mχ.?~/|BG'짮x@4}]Eپ# Gm_xU?<i3Jrw+Wom>d cOc|S]+/~Kk7m?n|3WI|?M M'oM3KwZjuwZ~i xkP{7^,|'o=>w-5?PX;W^uI^ޣp˛q^Eo~c|7Omljtk?<%+75o/xfO IGk[dzw?i__u)vK?1jht7Rխ |Q?jveԾ Q,K>d _ o eQ|_a~5ׂ&KvZ *k>_Vj5 :~𷑬j?QHwc|sFN4NPo#>/xX')}5ÛI@,t}?:o4 tmC'*wykA0Ul<%%Fs euexX*U(IXJQ|,HQJpuJ2b0U<]V_i_>K}:jxN.tX jZe\h*.<9oi6CUJJPYҩKSXV';«b#):#~xUJjúuRnrZgTu%(b%ƩR_<*ƻjUj:5w&SUs {V ;T{x.Ox:>1:?]oIgFNi:%U[PB3+Qѕ=kjω-o|K|\W5eD֬#=nB 7Pt6b1&Ѧ㇣?Y)* э6Ҩe)%R-Psϖ4QBXwB\J9Z=G*rʦS9}b/ttJӵj)w^Ao|˙PcD4b-ubLd_"HԩRP *Rk6|"+WZXjx(>^ZT]gN,9}^j\eyhdsP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠: `fj-O֩%mvw|ZvS}B+m*KH;;BM{LOԯل\jLSΈc;UEkT*N-&M✒Tڍn2KEJ*9Q8sP,\ԝ89[3YE69BMdi&|K Yy-υME# WQxynwOYivz^{a,Q<>+.[]OOę_v=xYOvE"qYZFM+AyR1_WEa9 j3FkHG(:UįZoJ?3FLK<JsK )*G*3t^^Z5aҩ½Lts㾱,,aZ0KsKFՃ)IԩRRC4*gԱUsx.X 0ĪTG u8tYמ"u:o>iMٮ4ρ<u_gƞ4Vuj~}{Lj;]t#ƖI30Ej-iˊXhӞm,jЭ b:ON*NRN~X A<0b!S1B!VkPj0RFU%B|)kIܿ_,t+/&_l'7eC .Agn'P4b(T&Zu*tlLWCjC/j(Э_3cajk9QUIʼiQ2+Z.QS3l_|2]N4iW~z$$׿ 0.Tҵ G^w%ePumV7c\gIgMcqrZQZi5! n8ƩF1Ӡ(ygP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠OL@9 1ӰÌt3צ@3к+ݨOuRVi-|ZvSPJ=RM ?LOmlA4֖ ,e }J$(7MPJPUJ4c*IE_IIƓt:q[g>i.I8GS(pG+J8tnI-jF[KMKK"?;J׌m{{nPfN2>}H;H:9N;J{EVWM^?{'m'-o[85 "U/43K5-KPNKB+KX-cie#˃UT*tU')FW4۵:qT'dn**S֓p(ԩ9KSgIE=:8du=2n[ Qm{)ߕ?܉nmPM׳ZnTOEB@1ݻbn}{s5=W[7e{)P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠j1fav6ѿ=2}[G5 /*NYH&"KkGKH>YE.Tu ?b|}TeH0Js (TsrT_4TNM{:= {kiMٮ4ρ<u_gƞ4Vuj~}{Lj;]t#ƖI30Ej-iˊXhӞm,jЭ b:ON*NRN~X A<0b!S1B!VkPj0RFU%B|)kIܿ_,t+/&_l'7eC .Agn'P4b(T&Zu*tlLWCjC/j(Э_3cajk9QUIʼiQ2+Z.QS3l_|2]N4iW~z$$׿ 0.Tҵ G^w%ePumV7c\gIgMcqrZQZi5! n8ƩF1Ӡ(ygP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g>:8`11(6$2rp =y AattA=UZ˯OAEK.e۷rw=ărXd;=7wkhFk%[+9>k7fމ+$]r;U: iwj|Auz}w2I0mgwgs>dBjpO5a5ofi-%xtB[N2M7FFVg qNO..§+*ӕ:f &cqЌ n0IdZzM5tҖ/v'tSwM"iF-ݸQ >]*K8;K=5K?LNm,v: Ңypj2|#p'-y#7'9N(BVsF}sTU9ZTSpphPSQe(lۀ1drFs犋Zu׭|WVWݺyhæ;zudTg$|uᯆ^esXxWº{h!5ƛZG4CRN9#hm''k;vlu?ǦrGld$GǾ:/x1lU 7A4. .7|/:}1ⴴᶵH[w[H2 ]Ahn 1kG7L6A֞p2xL./a^xW T$+sAsBQ;{ikAa1ޖ#TOQ*\td/3|it~:8qՓMRm7}ZvL+ew~$iꕷ)a -zP݆6m_ZZn--num.T Ư^|#䄔%$$fќNr%i?g>JUJ~O8Jt)8ҩ($% F\?H0Qr1{ZM/doK6~ɿ)( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.}_b0U<]xkׇ> x%·;߁^ #wɥh3/ujZrjWNa6Z|6cW:_ ΜjNTψY:Ju2gƭVI{Njʅ3 xVi}AT*\qtjU+ºRbrbץU84jJuc[L5idY|&/N> {w%o>&OuO]Ҽ[xUoޭx8y*1K|}f)P*V*gZN)J*kO9fܲuye9.<&BS5SN\>s :NHox.|{}[? u]W\u+[X^jXGzݭnc1qhqэFJQ`hƛuiTRJO2<V|O:(úQp9SU2sjZŞjoxkA^ƾ&ּx#^ǂ| r|9ǺWk WZɭR5jNp <%.ʫteVZqVNiUEt׫^/Ky}ʼe)Ѽ xJRYĚϏW=CAaoǾk^#zsJӮts6}{_#o>dӾxٻ;^fx?][Aq4ֱW~ี]R zJŵu.OxKO.PO~Ͼ3'I75d?go_.g_@ /=Bֿi?"z <:1mf|eQ1x@JGGBВu/lO|Dw;<3Ob|[_Agw?;[m5]cAԵ -#:AF^o>x_ZnH?k_us_h(~^3~w^(լMK\i8gzkUe5~wĚ奆;dO7 m?%t Ms>orFch=_J,{|̗]?|kԼ- >.|/ >.x_mf~[v_h_ZW>/xWi6&xm_byX\|? D4Nj6~/>Ԭ5:_u;> Z9U.>m+fu:7WKuۯ|u<LJ) wW_4o+~8gஓw/@>O ok:߉^̞>D,#/u< ҿfSGŗb-|=㯇8u:n[/ ~$Lj˩|-棨Yiqޗ|u0?hx3//<ᦣ< kLK[UO| & kRuo#X4&x&nm? h:ơG|1/^4mOxRk +O6 1!{oXtiڇ~O ZNT)rփaxJJ{˞N_ Y*԰U\Q}*X6X:ʢ6G.e`:x=$Ҿ|4CL9tյ𝇂\鶱xHԴeU\xr'Tl]ݎR7XOwWFRuF񩇂'VI:8ՇtVB%l5X% $JPKS**xUQw:Ԫt3gkb0Tk'<]="z“|(/&ުomuV[}[K:TGi-I.4ɱ1{yGG RaWN)ReO*Tyoh$B5Y{TY/9S!fU G),NN. ,\Q=jTS eCM;\Cgz'ܖz獼o>-<;wJL"<=mU;zzv ? ?pϮy*1K|}f)P*V*gZN)J*ki1l.w,^Yse'eO Ѕj㽍jbW\«Fsq|.-[^(KV5]WU5]JĖ%>.}WǢkVukkk!jqF2yc4pcG1Q7ERaB51ZU<R̡O1-*8iSi㎣ 00r=<=8zT凕Lr:^闚>kj Syoyo3ڡƉi2['ٴ|o̘,EZRF!?uNU\#kRlEhV*P|Ν>XrZW3乪Mz ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@P8nq0R97u+ʓnM=wIs;׍5}\wOHNP(M籹u +Vi-e[MNCqk4n,wgZƛ{Y^/X[4WO՝C_ .|-gfŸ $EM'JŇ-/lF+{>qnťIoI| ?1:9}*$R=ƙ[_iZ_mݤE;|`zWlm`/i<GPy |]נFgc}( Sž`U$qMyXyZuƕgizNkyymiqjpi:6C4ɢ۬-{uAk+hn;v(ڣԊhcW'8郎H#8푚vNޫJӲ\_UM^ҿ-{jI[ zN& eumdVKi^xj:e]i7i[KcN-%y{XT*|eMJ7<=o:&zFb3_?la4}X.I<0fu<=jQR)Ϛ$(J׎4־XVb0ږ#jRuhMԡSI΍EJM' 8I;n[}B+]*K8;;=6KRJOo-٭-.u}.DTYYZ5xs$$)$#6(֌ s!+I9\*U=SP8|U} M'yƕIA'y(MJ4"q?.:dL}ޤRu{-S戲Jť~T/- $sH[Olzq; =p6 ?h'@l&8lqq׽t? c1qp;G OOI#=OlS]oozM읭ki-<;Z8^N)mn5gedBӢ뵭~==LGzOd8w_~-wsDdvz-Ud^u4&խEէ5[hi;~kmK=H'hg[iVmOK B ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.}_b0U<].oh>\_!].C迉~zGSIgeL< d\}>\ş KxsÛ.ş/զoiɠIoZxPN=ͮ'i,s^izn,1E8ǩZ'<_K7߉~7a|1K|AwX:(ςo|O擫bΡږ}AEh׮Eox6mg4i2?xRtv獵_,)vKi_4OI//!r_۬5}V~K<=O??k".G'yx_3?tx|^ Mk4 [nzUW_p'+8#nq2Seÿpj wZ$k]7 '_O{W쯦7?t>xHF:=s_>>ω^xYO"7ndԞtz&a~^ Kǧ|S xO5GXŸj>,Ck|=ƋxGAu-Z}o&n\|?ςtEx s&k櫩[ĺϵxVXMj8.m|md u WHa f#ͧ[+x̞dIe*ԍJ% 0 ˪rZ&sgb+ByUEt×u֪%VW֏kG8P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P߱d+sLi\7qƾx (1ӡ^ONim~f߻DjVn7wͥe7,"ҤS,׹/4]J͘K=Ư<8SYtVNnJTn$n))EIMߙ&$T:rss'5 rIӃc9(USoS$Fk Gľ!Ŝ7^ATR8H[ut%gwcsǪ&'O F%{qL"rOUdw+.+='7)ri]-dޚzK\5 &U/4/K5GPM..$Hmaie#Hݜ2:s(S Jsa'94y9J˗vtURhSzӒ!N29K4)JMEA7.fQv=A>y';ә7i7kՓ[MRWkK]K+ߧN= ot=yӶ=z=Lt:q@1c'4|dh =8 V8zqqh Aav6cs8Nހ:~188?$')鮏^{歷}vNֵ4_pz^ӧtiI!iiֿ[[v12@LI =fM'mfnoޗ[;ç~8_}]m;ZjmYiuAHA@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|qѯ8WduC2k&eK9p-xׁo/. -YwQo/˿z׶vڅޑquguko_@ec{e-ݻ55A4ej\Ƭ#$N\KO\4iJTԥ9ӅxSe*SU:N*m7 m|3{K>{j> +[_|&ğShj={{9Ԥt;M"Et븬|#c&^,C(-i\;QZ5.j~*!JPu'WG7ΥKцORsUqU+B!JPJ_o:/ xFmcƞ$-M^27ּ;⧰É$!}NYy|D*chK Qa\!*c4.w[^Tn^Jˊ***appS'BQ,EEYS9xLj2zQF<52W/>&ڗ_ķ^6Y ׁ|WSPYX|Elt**8\FWtE*rFp^|˩Ƒ*ζ | -o <4CQ𶗩kcuu=kNp͢\j^l~6+0VҖ:&IK\gMSJQM%YPt~s7̩c=SBy26,% F47VXd(8t<%]:_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_, (AeaE? . -YwQo/˿‹y~][ 0_TY5 ~!մX?j4n5GBqjRirBoMw9&FHM0\vQW*jI+1I6N^G[W r.w߲5g|SO3|??OuY5W>?OuY5W>?OuY5W>?Ou?(#7/#vI'/c/wO~I-^v`įӳ~X)X#/ߦ11pX$I>y7e׮ Gx޾?xӟG^䜀hYX̷>K⽖+krŦlVVm?xw#9{}fѓQ^얊aVySZF{OdO7dt9{8bSI+[j۫;hu\ѵn]/ddmy,Cov?۷lQkYOvmvfI]lޛwEğ gW9q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=?g|SO3_?q=PKHPPPK.AOEBPS/img/shortest_a_to_d.jpgJFIF``LEAD Technologies Inc. V1.01  }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyzw!1AQaq"2B #3Rbr $4%&'()*56789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz ? ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( k5<ƏxOYO/k6~jpYKI鮯et"|CxGZ'_~>ck7F4.o42W֪ Ҭnb-ajW3ޗ=Wh|H>>xW|Q-牾-xZ2MH|?lUZ{ujKh Y?oe/^ ~&i>)WRGҼux[E]1ZHHd%x԰-/[OAi}7bYi Kc_,D+@ )6Ov;|kQڃ&4{ VcyGYkt &a#udpZmy&Fſ*~Ah~[SZ/4\iozNJ4rM:&q& 9eȞ@ x['gOT|*O]?j7~SL!ᛍEWm7c{K]HNt rom9?  |bao%4\>)&|4Ҽ;b~XIsnnc}KF/-Y6m?C__w}WC.g xRR_ڗş?ok4]f9Xf?Z*O_5"  ZxXdo|wSCzn4?Ť7HEo!MkU[k>{{OOhdִ-cc~Yk?4ԴkDxXZBcUXN;J( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( Sx m|?k1FdzW/Tp\Z2C%rQl~5Mg/⦖~"7<n+OixFռIE55ZkfhnLl$?C1@_O~$^ljoti5ľ%(4<}n6 ߌ'w/ٔ٧?ɠm=O𯈴mM97^ U6טL+;M'xPԵU߉j|C{e}?A>Kk CӼE^k QmwXNllf~)|S??e_xֶoxB==HaMlb"I3!P7guo~+xoRW7F?hoTzx'?F|Ey]J-BAkQ'm1Ѱlv'젵Wx3-ɧxfwм=ynk=Ӽusq]V(mt+'k{6[[#`_ 7/߁ir_ 跚oC~$F 4Bo ,eY\)lzwៃ_<Xj>6o/hZNL/|!-BMtK6֒_ɼ<4 `5,<7<<2@'Qr@ RD_!Q#f!o^^o ˸e<ÑOz?jQC317 |[#zЇPmsl70<q@( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ()j]g̷w0[D=dq݅x_jٟRm{kKmِxږ>szXO&9_lo't_%]'-,G hMU+q֞W_oJ7v:)gko'[3_oSPo]¸s§'_bG|o-??i/ڧT?kYh~_SGTom{Gxz2k_3Cx4S芣@1W_>?6ky"\矛<րb/T3:{ U:}KY;6 rO??mظ?.ZrFmsw@Ҁ_U9sI7S{{ퟶ.ʺ;esEݏA5Mco})l~zo<|߽??hj8v$#k phAi>x{eY|g牥@:xjy`}0/x($ u?EFa~ڿ;mg⍧.]f.9O9G/h)$ hܼ+玣Y]/|Q2gz}pqAvP@P@P@P@P@P@P@P@P@P@P@P@tPrEO$ooy&u. ڻd@u s g >?|[Vah~C¤~+4YS8&XMp$c >!~$߆g/'Qk῅JJEcjKXDΜpVdsTah;TSr\ w+ǯ['͢!Ǯ[0Ώ\s\s_AWXW+˰_c|5t9)8G sh֒M_uj?<G^ \%ϋ1gs&OxgQ9Ni=:*yv?9?(ſ -r ( <7" >I ~?l_^<|IO\C"]7#->ԃMr$Q>~ ƝQ1߇="ºq.s_hq;mt{}[QfUt̏ ?x*ǿ Qe]7I0j#Y&*l\iJL3B~|`xxK7 FZ710θ=qfڈqfo3i`p󧄫*qNfݤߒ?;$p? N;3|./=QUB?cJUS%?iFs劲wz^xxUJ_gđ?`Ny?YQ+3~ɇIafy' ?~đ?`Ny?YQ+3~ɇIafy' ?~đ?`Ny?YQ+3~ɇIafy' ?~đ?`Ny?YQ+3~ɇIafy' ?~đ?cs."27sM.I-m. ,v12y!Y b2Sq2:pT牣 nZ}m}?A?pSsFjjԧJmBU-~[s8bע|pP@P@P@P@P@P@P@r,ς|[>04 Ni>/3?GkDxg/gi۸=*D_KFX o-ڇş/f+so&q ',Ya+Ϫ0# ˜Gϛߴ;9[WNG*j0+ YH܊ڠ ?coZRşK`ytBYq1  |o Ai߆~@6t R1g!ga#31$$6 E ( ( ( ( ( (s ~x\Og-ڭ| + g}# KH ^8%PҸ(h_֔:'jL{:&.Vz"uqMMxJO=mA'6%;O4y>x2Ri[?ؾ[lQ9e|Ii>XgV[ÿ-#ftgEY=ֱe˥4im :+ʯ;t/.Xiwhw-~`6mׇ4ZE$0JI,Ēk3,&qfYI;bRcZ(+HJ h’_$w52)FP+T*TA`MŦ+kkv)nƩ|9#%[f_e# -#6B~`Sd'AE?\{]Ļ ?e[Z xѤDQfdžk"{WHg*ղ*:id3Z8;%,FMYB&3 *J?J_ܫEw_!_I<3=exz/<oDOߨFtMkX$-`TC0[BM7n's@Zor,f ux7> O:./#e[-qlt( ( e7""cIGwyt( ( ( ( ( (8|LsO<k uh`[R9#hH6A$ k<㌍Cޫ@D xoeXke AZ?!|c o_> }uϊ)[[ CѡKrYI~3ď|j?Gо(b%esⶌZ|oxOP| kKK]GJU*_<\]'%~;a!lVdѰl}"G,qƪUDU*: ( ( (kό(g`~kς."IeZt0YHdP$Я7j|M=շ+Qi>%uCZ,U"MfX㶖YSʊ7 c_~2LIxZQңB<9u͵iom5Nڎoor(H=Jg6O~-1՝|+kwu;xln'-VI=6WEi5 f12#λ{mwFUxv R]wM9.!u+"9՝wiı$r1|{Cxo:i^~2|#,.-5G,M]'XckyM%O-YdGsLA??5ψB agUҴh^o$vJ*FÒxjfԦ_6վ!o,P,>G"f[돞*HЀp9zg>xߍ u߈aiNJpQk]{]b?jPZ1Y);|7M3/ğ!c<|_K9Ko ;Kmڶddokr *k|Qx_`lhܞ5 U}l^aRR-Z15*^^ΜmR`r]=ï |-𽗄|-6I%{_\.K:ٟYs{i%vQ減-9'(ӧNTL5%C<4-N)uhT.XVM;P (/4Rl1xB*ȶ%m'u,`'5:iZx[Sjswj3ZT^jWVHT$tQ ԡV4tFti(ϟ(=ބ?aV} K;UȘiW\=C,f͍ثP.K- Zv5c,x>'L$6/Z`-&c,\(~:,4owýFHS^&}*8|5ԼIʏKs!V o#o^?bë[ Z\_|Gw|J5_YԵU4SqrL~oᯍ3v6^o7Շw":-)OֺSy3L%%megK4v6>=ga /xw°dT~`xz~ſP Oiz?o^kPG<ֺU5?:zZʚ1;:Z0I-퓈Ao Ֆm G\վx&O>w?-W?|1:uV4{4R:oP@wZދci=^IiWW 7i2ʳ_%34(/EoԠcֵÉ&ɠĘ/ :tkP8%D$1,`,x'H26 (F~nOi'rz['o$whi"\Vu'٘Mhl7@MjVAҜeN-:VսNQ棻8iymZYmjX=J9/'1z}VJVͅWgzU{mXroXkZ;t2L *8ҍx*4<5ʕ(:NԜ_WbBp7gcjxG0t::Qx|2EVpİVJ1:pxn=kH5]@!ݩn\nⶩ8L2cG1U]iQµY髳;jyO!+᜿d1̔AkG GBUsra״jjM#YVgXgb܅eY-#$eG0ʰ'cG(F9FJtSNqn3OtӳM/*8/dVVVSpGkؚ-zU FrDJ_t__1$Q`#WN;g׺~RP@P@P@P@P@]?k^1?%A7V6:d[F_0@PO.8~=­'GǛݣas:߈u%Q !`2Jl/υ ~+GoUO4!4Q_gO>*.~ӿ:ON6Xt Aٵåtܶ#ZZFeNnGlik%qžo|3m/I>j?t/Ɩ˨pxos-Z4K彑IfxͼyXҾI|K>iz6 ?f+ZӮF׾ \O5?ÒEoY UFI o#_c?٧_*"kڬ:~&ͮo񵷏Z\ۼ7Zz8WH o#诊k?>'a7u'oKxOק";?>%h_چc>4O.{.6Z*#N [|YᏃ'ßڿ9-v9΍iWZ[K$w?Ǧkសq/G?+<->x&?|Nӯs2l5kio,?SY/=.魥6  xGOĿO&u >%|<[- kߴZcd.hn-tջ"TV^sn:Yl~P˿kM<KҼax]m<'隟TVKz::`ّfDcd|4?d/ oupR4Ѽas?5Vm/N,,^[f7~h'4Wiz9xm7~2#1_;2$ɵoursWR_J})j*ywCdҏsv>ʎ~*u־|[7CZ˟jW.4ֱNda[Xdd[J$7 #{P7go7_+ȼ_x>uxbO%| &Yv2n,֚qk}Kwj|o`5[}6J:<7cz4>e{i%CjW }@'~4w@|,_x= |UHӾ$O_oy9mt?|Cu"j ~D;-=d ab(Tɳ\ Qk>!+$ 6q`$Nq]HSպ.iGWig_oک[a;m6NxG~i0#?j#KC;գON7gppRSw*pGm>,9 C ?TWCs5N9ò1~!?eک11ur" ֿiZFoO5.Mkm"S[ OF.5  rKh$z|)jg/wMO}IxwT>#ZV`$:ηvIo)v@lK@Z~a=2g/l11z?zrؿ8G-(O>%(b_>G6ezh~0Rz3|D0fV~㯟:a_i6?:lqg٠LsNї|¨?&Ls w^~ґ8? &P3n1x,U#`v6mu/ /:?|NqᏆWVVOIu\I/ bk'|,^4BL>BS"~R;Mx6i:ͥJž hˆo Y<1nvLo~A6:7>~&↉o>%fFF]WEuWpnmoee#nPN+T9if2vmt]ٿw~B ,*YUSaO֩IFUiGFvUcqv YvDcILlkjʏOEA{ :Ji/g|RwͩN]h}1q3)b$l>άx+ 8BOG(*8iPCޙ#K]xN$-|5i25 "*uIP𡆣p9^ x,<NjSk{NznIZM1Qx0q5>-Jy tpp}l<,&chb` E'*T)VJKӯ|A}p>Ӯo:on4_gyw5N$1kX| p2օoju)EKg 4b`)?\Aȳn5γMn)Yxe,>U/eQ^ ̳ i>\DR:L̓Lx~,ľ*Mug YO_H5S= ![/)ԏ+K4}{*Q"+j%EkWપGqVb KrYmTI!BI8I*sTIk˙Y᷇|hBIƷ6۲IpޭVװ _xǺݟͣjZo̜AuuiU#eLI <~*owhi.IZPSWJ =2¹iawb|3|frFisRS Qrq`e7""cIGwyt( ( ( (!kᵶT"Kn׿k߀:F?O7v5ϋ!@CM`N=.ZIm9C6oO /R,kOV3j#7)_~3)~ze7& t-xgS|9ӼeP*g bv|6on$*F\QGq i PQ"QHE"*@ ( ( ( ( (YU2U]BA:Sܬ29q@<| 4߀ڇ+g|S_I1/O$p/glD@r#߁2bUd0.;+i-dwlIVqj;Bs3g^xPh.3>QӜTռe 7!H9[ELt^8Go[,m"3`\w@xOXJ Wo|{8 =qqj 0:ٟ֧Mه7 ( >0>?i ~M|?Ѷ稄Sy\sgA`?!||['ŭHO /5-;Lռ/֟[>|9JX閍ww<5 匃|6Tuߏ'Y?3OYxǗ1ZȤE22Kk2+$Q?Yg~-_|'-ts7Ωs]Jvڥˆ(5SšZCúKv Gn 5?-p~uyc)Ur _q<6//쒄g9RO7Ztf.ʿ7;>x?W__|ghkiy%s'x{O~-tv1vsEiӗ&eyu1xxA)><_>=i@ /ROϤxľ8Θ!mG7I|W,a-:lv>6% !m߉$5[mikw[Ew"q2W?@#VFQDќ6w1l܁ckM:P{˩R kKKX{!!(Q݈UT$kBlUz8l5)b' TST%S"&m$ڊmUvމ%f;C2m="k4kgWຎ@ >U16ޭ_^6ȸG V52*%*SWg>I(ͱ*mN8x/RVxߒNvs ( ( ( (Mo @oHKΣ$izǜt=,m2ziEYzm)oeְYX1ȑ[£(cֵÉ&ɠĘ/ OԭRfVkkhn!܌Ό a8E[ ^SJt*sIҔdZ5}O|&SsܧeӔg,.? ChJT1I2q^@4m!nViZj-Xڋ Z-oPm>Oں~֧羼S<-#G2Q1eE G`Tp._w?cuikfG^ @RE WZaT\O%.{+dNU< -$pmw?1a zT_7xK—mu 3Ó ZO6we-كmkتCoY@K;~sQqXq``SѧV4VԱ5Ss9El~cំCqaƹ bqcׯc0ˁRx F_aib֌Vs]3MY&Oi,I,asf$, %| |1xP:xjN1%NX}=.x3_0V)acZlZ]b*IӔ,%,FMj<1`[ۭ,܈\<E{BM3*L@ruj{81|ϑNJ1oʥ%I+ ڹ0t#ףK SSXzR QFz)R)εYF*U&SkÒK4xDy5ϥX44iXvG5!Re:JЂUQ;Ey$ _Vଆ3%f~g$\L:gnsS! YT1^2\\9{*شJP։ɟ'^ Qxa9KSª0ʱO֊?}TJ.JR~^Y_~3'Ikį~g [ڛ3lH}dmCXga8S@6?io%<|Yn5keo;S@xK2٘N+\~8|ߓ#}O?>'- (x|bJ ~ Fxo'㯏Z$s׍t` hd!`MAG@燼+_AѼ7ڌ[i:cb :`a@tmy4P@P@P|L_D5MC./[՚[Mva{N$*SnZj]/t18'y67 ZbJtL#J-T(Jq(5P@]o`|Vw!|c6O/qD{)#7}W{_gg.COK2(DG]`0ÃMa>7u^ó}xuŒ6#3xJA>5+x{T%Xl.oYrI$NM|o:nhg~fk[X[&-cD\8ZР (  }Puk B>f":σ|a;OI2=o'. U89sXjY ,.9}[2q)CrbՏ0tq4Vƽקۄԡ5Bki<ƛ_z8-.Ti!W{[P"k]KdV[LSӞVG{t[~˪׫pTxf=Rgjw8 >iN/TRiK (⚚⒥[%T&?z=20;B ( Zh}櫫iZ^o-]Acaai&!I#.pqxLN VZv:TNsiF1z$)F MF1Wm$V>J/ڶ-BҿfK7o-ti:ʞqCuXδt<). c q~*֧(p ԸIT3ˈ#?.K,XcXa󜥏jN8zN׷؂z/cݟ_pG0Ɛ ,qEqj# JNR)NSrۓwmmճJ->h3uXxz_ M3F4֕-i}kgax\"q4Smx1/o y,uk?ei[eo+$Hvyl ,`SO>x×~'<x{}gK} 4E孙Mමֵ Zvy1hʱ*7CX|;Wk}c/skO/A5ΩmKOcqUW~? ;V³ixYGM*ho 7KDc 3>"kgI4O X|@7/_4]{2kZ˝:߇,uM}|J f;=0OXkĖ፜ƒϴLk;✻ KT(J6Jk8vM7k ?2vmX ˱S^eUQYOᰘU(Tb8Ԍ99+)C FuA:{Buɯ9mG.M씝(z+FupB~6.1ȰF8BRmWmݮj~%k𞧬Rޡ0Eaa{ H LUvu^'6*儩Z*3w'-y$y-ՓOFG^ $|7̺cǞtN3J2Lϻ ( (?/{M/(j+'_k?)>ɼWx$?C;D̿!C_P@P@wgxs*]:~Mq]vDV.Oyug]m 2Šcaݻ\F"eׇoQqֲަ.?~<#є<.SqJTqRR&$QRk #~:}8YO`P}uO~&U]KO 8g d _. 7/oxQv]TAEf<z }:OV#$? 5KǺ:hޛg;=wKSPf6me}K+_5-:Ua/Sm[QXϩ>^^ED!dC~x}_ƚëo:~smuJ-I[Z4k66gGV_FЫ@=Z ( 3wc5 įocbJ?+( ( (?gL"o >dE5nm$?~GB/ 74kKsOǗmklA>.~~˿[O|e?tL19HRk& +éi*7obUp`~?d_7okoYĖ!>e`JIvv cԼwc  \A["Yh&߁my@P@P@P@/_5_1GĶ1xƞ'r t Pl\_i$xW2i_((VykSjTШ¥jSt3Fh:u`ϿF鮍4FtQC$t3ƋonYx{T.~R{ݒX!_fk]Kׄ,hhu?ueNÝ{9^*p&=(gT._Z2ݷp'uV>t*%Heɫ<=:ֺ)ѻ+ O p#w fTTq J w^8-<➛|QK_לxk<(>MӠ/6<[>\ƼrYFM,VY슴uy% /wGS҅U1KoߴG_~IƚEF:< WZƎN>i0!$:CZŁLQJ_C,J+J8j_fj?TWIW_N;~Qg:?:{oKCo-%{_67 $.#M"ïtrl5|~z*ٔkKĸR\L)NR磔aKXL 5,eWqBܔb֦kc("$X5 q EPWR)Nrrrmۻm[oVޭyX}HP?7Oi=J xV>_|gox#÷ڳV="-th-/#Y~Ԫ̊a?@ho?<_qE'1>/?>.w/Nh O=gAIJi4ibmM,;Dz3񿆾#*xWSc ~#x<t=;~״7Q/ih]ckcjwPZj%e6 A'fj:G|69Nwijzώ$w]ok%G}'Ý +2Ŗf֕/tjsw e8$ZEG6 Zg'8xO_xIf\9>xo(oϪv;N^%kkx$OZ㷈U#;⿃>Feh?7n;ER}]e%1=-4bYyX6mIvH/ |Jͤx+@ja_v*~^--u(tXc-sv#gOZw=|oksPLJ/OCxA/[T[Yџ3e(˼1QIs{*8hTP80PMSϏKc(j<)nn|-/՝{ǭ$.ul%°X?eե,F^ OQQ)F$ʥYFiM<N \; g<ϋ3gթV^X)ԡ[:VTrЕg)*dOzwďVV#=0Axnq;A Ed΢h~eʞcORvqou*˾ټ')8/150%*Xv: nWSASόOD~9}^6wkB;\}Hˣde m s?a~?eN?V3Nxiv_(~> m s?a~?eN?V3Nxiv_(~> m s?a~?eN?V3Nxiv_(~> m s?a~?s27om sMѮmHn.-! ,0K \+*~ৗe O F(UIk{?$ĸ~1~(& 8e*5\eV:nxsۙEt$㏄_o?=xgDj!?s4*ON3//~P@P@}?IJ?h?zs?ʠc?Y 7'>'~?4Ǐhj9ׇ$𞡡hP/.`9#]^}_S0fz6 3~0CF⏏ Ӽ]zo#ogy~z}ƚ`oo5G>rjN256 ?|kB7eм%?^%¿xgþ.Gs}e=0\i7zMwwB cX@Wwi,7 鶳;\ߊM&x;w}j-Η{>k6j*-yeF(vo`ɾ9:O7aĕZ-c_.%xvmx"\*xQk-<9<k:Eޘy'u~ᖭom/[A xKV? M-$bMΑ6i5\jO<[>,tl"& SVk1^u&|vsJ","Ou=%seu_ OIC/Nhnų;xbN?oߊx/:t r7&𭯈z&hn;qKmV6މmhĥGwO#5֚t^/?uA4}QlwMk -)0+wǽ'XQYxu/^ovZmׅ|YjxZk&7sf: ,4гf'ljt.?gώ+ox>=bᮋ> gxruOȆ8mnRhY .ߴ.lj>-xRį:Kh|>&CM/daXoek=r=Ehޓ!}sN/M猚|*9Mq=g? j0t gK@ךv ْw\v-2 o# /Ow q_+X7Vg 3|}ۥLB}A_x;ៅ úo1Ba**˱ygM{q1y/.dYiWgcom:( ( ( ֬.uM#S,O÷wvvL= ZZk; [>ܼk[`>迳wai_Ş/t;?\xFjz׾~4Ƨ{y ZItl@P@)|=[kxź7mY㿹PiM{1Zé ~WKt[)mv[>\?ťRT)Bj7q/ ڕcX4<U{5F\Qtxc)ߕqKяFIREDQ#iR2ٮIfx4&+RUjq튫dR// x#IB i1-m4o /4s˳M} R v8/VWowogn{9cfCƯ_^z"J\{B9ii:*<` ( ( (_o?=xgDj!?s4*ON3//~xzߊ:rj1'mDGiȷoܓs)23+ ]q,&~xYbkcq[4i^ҕь) $q>&}#s"<,Fdt|3b_Tǹb խJKRqbNT]?j>'Q,~(;PGi6׆7t{ID^ 6+j6x J*U%SJNrrF1++~2LZ|?ŴibpK˩P"</RT,EZq.PFz7aoZ !+hvex-IwV{PVfg2lVг׎<,=<Sui9hT T]FI${|hEqf3_qL 408N7>0xu*8$iӥUPS^T:86*H&a G.n>Vݭ\ʤicpxw(Vu Nx*RPV\UQRU*/[׹/ixm'#qI&z ?X.v"|4G(=|/U)U#ыj4)nx ?7IShǂJs*E 6".hӬ8R$fo x<_qejf(l+;61W:I>V#76R9zWU䔫gVK#YJ ~ܩ?ya抿" c/~7*\$~r iO`)Ɩ/&kZ?h 9V"Y^!9s_PO)W񧌆te𾳥YغO0~KGsokK6乯1QJ\pM?A xo2yYХCƝGǕr뮪$k ]8EM4"m| J+ _In%TD~?|lZ?ƿ !m};&x/ß5]^FMԬ$/ΰytT}.!gxzɯ7cI[#>6cjTKqRLfmNMI-[ poC?u!5 SoJm[}WJ{# ^0--5KPFH+I׋G*vq^4_/<1cٵwѦoSP֭k |FVA=Ny{s,elD.]1_&??N_]r?9T3h/b׆*MuK>-u񆕧xvx : i'Ҵ?!p1uʧ1tNYkV^¯QSռuٯ|O5VWZ9h!b'Rѫ  LV??N_]r=?eSw??o<J}?x)'k/ x~$iSAoӭd1Z\>bL`ĝxrgh`?6z/-Awn-\,Qߏ,eu=KEzܛO,u,B `ĝxkU?p?-':|5O XGz&[x,,8l4$,$B9,\OT3~_V":G*a [S׈t \xĚXz=ivMJKy Rْ?ɪaΔJ^+Tw7nY4gklS#"xβ|nwG#d*RN*Sru(FT1q "4k6"@±s-Է gX>Xb<Ɣul*_9ӛ?\GOx]x;Sb)ڞQB5vh xCo=>}ns2]q<6rXDUK 80\AУW^~ii8Ӓ*EZ4a/\&8X )2qYԅnzxqJRwWIiUIė>}f 'NzP<ֲe[iр:1P6b'8쮖o'wJ>ǞYK4Փnq q?ÌƎ!Woy,uG}_VK ,.)bT9Э_/2=ZU!KZ:H"Ӥ𞏦^YG,>-,-_*$xU 9t\1RUk®qN zczACN2圤ܒJ0>7ax, W_/$ʱuq~|6%Zs*4QUӅ%R!Onto?i?hx͵ZycI%FYUbr<UibdIBT\S:6>nEFSpsvM7Ĕrrxib)fqP<b,?WĪ^41u:t+ji,uk(>W]е[?TƩ;K풦mKoeX1H^+&a>^Yy~?V W3Jt'F3Tܔs]Y9|*2evmS*Ͱl68?eSN|M_eT!)NKcl^kxCIT#MOVI+}&Rs{KMSx=8 3旕\DEwtdfxxwy/9=|RN+pz)`rl6iVvW3:2m]':  搗~"iz,kJlt2Hغm y=eYg n}PƼ>+k>j*ЕSyרN,Ϝikpτo3~"2x| ҆cW/=:%< a%}Ò _D4|'kwZnajb[jdNY!utI?"׏χps1m^ 7]S59R2J4\zp-hd(G$]8a8_ qcU\N{WSF 8RjBX҄zHe3cտ4__%ؒ ) (oۯ׎-~~4Nj.nu:e6Xw'gh#mĨ|\;_ŚOu z>#IJxO-:W:kkulG$fz?㞜N"Ֆcω|GkqSKim.|`t` :_f[nb DSj#&ռ1e þ g5}ŭʚfh@I6 g\ܮo<bmOY OxWݽ֕7<2h2[(AV t -{Sgy>٧\-)rp 8u>-~ Y^ 6 |G7|-u{Y-<%km%Ix _#ҼQ Wz36q}< Up~"6~ӿ'Y^ ?aoxO~ {űT_Z:6$+6넅Dtw>~_ |xg/xK?,}x¿SH_ oxᖗeۡYɆOG2=lu 07wk~Ͽ!x[?h|&w HK|;x{h$2|"?c߃ϡ2׈u|->]R'OBIKY>F7 [IxnlOVNSGkV_|U SCi:NJ.m-LYom]&X cCKgÿϊ <3_Px㇌=xoTGu?;m3▃zևK zMk֧-Z|o/AsHփn5k^Ԭ'OHkGP+{X nE gr_3ş韲qi)1 :> is\MiBTepnl$;wk0)@'?f6Ķ1ō=G|McoaJvK;jkPe> rlLP@P@P@P@P@iσ_D־)]޿!<%К߉QaQ-Of,Ox'V)JvG;B=F~>x+[!*a ǬZ\*?i`bg>y]|OZ| ?ŷm-6{Bt*c翴8?-!Kb1:ה{7WZSjsC'ƼGWŎ,dShV_cUF~kjY/|2 WkgOft_ķ{ܯWAx592W;a}k1L\SsjM5VUmqocl~$GӍ BO t wjPCki6Ѥ6$[(aU"TW1T!$I%I-]?VR"׭':*JS'yJsrܛm[&fP@P@P@P@ DzիBhU꟡f_^!T&uԗ8]Z3m4(r7j\ij0Jv owF8~MaW[ {gR+J6٥*qI~v_b|-k/fpl=%W'ԛ˱9e*͡J.L*uU7FP4Zuu]xJL!/Y)ΠMt98UX3 ga'jkN3/ܯkN|{)++ə8=E*?!,f9nF Ry~/ Rs).ZTMZ'I&ulX.}ysx Lf@K_I⧞}<\ʜHԭiQ MR^UIAY6j-a|' xKƘ ap$e9Y<^J*3.]J+˨ʮ+ \1j'<]yN!Ӎpte֚mٳ"be\/`D|NUO3c3 U WU*~ʥu8EzU9q\EQ++RN39 u_m޵_:Kl|~TvISp VOS'ibc0TIv|ti84멗 |7<;Ù 8LX¶N.vu'K T*i8Pr@vE(f(cOZSon.e骊5VQUpe:n1u&⬜N8o|Mjߝ3g&yo?w3 ( ( ( YxWľ7M`y)TkH8s[K_W&v,Sɱ'?ޫZsJ'+.OW;|_o7_nhMZ+Mi-R«!t%NF߬Ĵhf9.``ㄍ7_PXFսɸx!g =8xu*9%Fb05)yիO uVW*S悆:Y5?_6o"MխR$z}Ο͕^3ӱF%e8unK_QӖju %%*i)pVo>ؚAo>#p^%d<8C.Heج+:kJTsؙ6%sU|3/%׈hݢ]ZVGCLZe\< ncY% kmg嘅G<-GN***V8RpŸѷ9?8W `# ereʤe )xbEӕ#ҭ7wgwi5gT5=HiI+ iSNYdS+ɸLҍoP`0hСFu\άayi{GBJ1*WoYc -e+m,ZUSʰ6:SӐE[1.vNnbDp4c 4=|Qp*\sx܇rz+:+SYuW?s% ljN9a57ui[iv)&%}='Q[ FK))e> S, [ ',+89Oޣ[iY-]+ ?dr .ӫƶ#-CgXg D)U;sqXkƣ Eke`}$K}w%F?y*X,Ε,pO:SNeZңB2e6.x@kM J"ģYxZU8TnΪRj]bn4y⪘_ ZUJَ8ƗY9ҫ4䟹R[wO@~Ǿ)~|)kٷ"-GPKnLr u eϛ;\]b ˌ>GuuHzc}#8O"Yl,e:^΢4M ZN߻ԏQ.|>`煢G.CGxŽ-+֛n'ʸc&9gª֩Vkq|t'5W gdVS)`rS7WX؎_ˢ>? ( ( ( ( (/(^|7_OʡS;tLK5aP@P@d~~2>0>[=W:xWm/uKg)/*1@X~j-aj;h54VǭFtiTT$9B2rPOY826 YYU#AReYH0AW @P@P@P@4gǫ0j_-%驟B?-Ĕ?~0WgPP@P@~ [G? xͤ|G0Gwf/c嬐 H<5naa+M@5[pw*~ 3yكƥwcxN:i%G7ǻwee ړ^V?~67N$ Sܭnb6$_f a~)h>qg%~XGp@`~D`nRWVcq\{p#Քmu)h;vG ٰ h^ue-so h ( ( ( ( '|5*}sz~Rֵ m:[Bkd&s@186 Q(>JjZ\:<}, qyc[`TUj98Ra9ЂRI\]~sO٧ᯈ~/'n,oXn.Do&M&1>p~6VX^j㣪*xhvrr_tbMyWѲh5_S%jXҧKҍK;eV6j̇ah?o;H ԣLVEz!\I>DpՔxg8ĹX,#䤺9[g.z&qN4 BѬw6f~N]4ډ>q&⾻/r쪟a)ᕬU+=I^sy?.=}wxԌӫS Ap OD@ ( ( ( ( (BC t U/ƞk-O84sme UK2('_ǔ+2g:X/%9Y)ݕ־g35#ͳ.WOUxXj^ (T8CJ2j7QvZ3g'><C?b ?e w!Wл^,hU د'Ydv@¬DǟHxe.=_@?׾#ög'><C/3v+ ?Y>'9 }y _OWY;Ϗ?+]z|EGl4*ωN|y?_G^gBW{,?;`??V|Ns$XJI?F[[⟋ c"x?N犯LFT\SYE odm]/ލ7Kz|{4! Үk(srCQ{O .O{Ǟp%u+2ʯ嵾]L @P@P@Pwg^#'CZצâwڤ>"K4V0JF\m L6#`#R%Si{)ڊm.?Mq3lSNKҞ3G ;a*e$qn*ωN|y?__e.=_@D?׾#ög'><C/3v+ ?Y>'9 }y _OWY;Ϗ?+]z|EGl4*ωN|y?_G^gBW{,?;`??V|Ns$9t7֞֙b[IVbkIm T!j8V41Ү=ʍ*q匭y&%Ij }288Or|\ׅe3W3ћԫV8ӥ8TZ|f$*9_a@P/.<] ;-NŴ趚m\ŪY[;]yIJF3D_͊D 7(V*%zJ0@! +)J*G_Z񟆼;xCZ]sǚSNK;;Cedq򏈳>ʝ 3ģ mko/nׄF8xuE)7)V9%uK+AI]~epuwUߴGI(|ӯ/o4HֶvL$'\d= )׫_g(aז1MI>ӃL󯤾/)UK.!~QGN*jiΤqx;8cn}xAmxkF vz5an0,4'$.Ikhaa)F0aœcH$6fkcfyesC\N2\EykJsitM-Hح8( ( ( ( ( ( ( ( (<[G{׋e e~=>!'uk?[aP*O?C[_ >?_hm<~s|vADxoaǙsm Ck-gR}^|sY,K}a˖ K_ VPҞ_i[k& fZޯU75 2h%e?,@P@P@P@P@P|a>jK/1}ƾhZgi6T"+ &&;ǁUlMeÈhV[^I54)[O9|3䙟r)7\KՌJRSw|0UXG~?P[-DŽDm>>jb@w.ch/)qg i(]=t໯zr򮬾(-ct~~5>|=gwSχqz<ҴU˚M/~Կ>/y6Zc|*<+ނ ~|Se-ׇѭlum*{ s=Yſ_Li'_-<3sVӭ;)-|?ssFkɤhgR|f?_1G KX\[Ѿ0xNNxQF}?B_EZifjs?@]O@O~? nbC56O9CC];^@=Bۆ`+'WŸO:}{=ψrm!/33-$vye cXT6<25&_0k8:]2w>v7v-ƥ#^\[¯s,cbI);Y=@VZ|Q i?MY%ώo4]z̶ZĚoeżz`lki7ߵΎ/oRcVQk:ĘS2Qn ꖺƝᥙ&亥PZ-qlS8ڿ|Y|ul<7|B&~/=7?uޫyxv/O+jnvUn9* ؚܐ8_ꔣhFOY+| dX%n2cOO_XR0^\ҧR,#)/*~?>.o+/5#uv[K[e  3+N*ɲT}gZ(BrE>%q:xMIfÖ%SbaʛU0*QMZuaۿ?xwGo~ Yme+IJM)K@=gadžʗ*QZRhUW-sƷ<_< hՌP>jvNxL߱/EK{6Quy⟈3m;9{E}ꮒ ʂ5Lec[l[uM s3#aZ8 )Eңd-:j8g ;RßY$HƋq"(DD@Q@ _XI.UZZi)Nrrrmۻm[oVޭ ( ( ( ( ( ( ( ( ( k*:FYXVVee<A @6L+j>(އ$/> kKL~ /ĥɮ麜n 8o wkφq/#ðr4?p[$[E򥵯sX12]K`r֞>P qOM_aCm|MbG|+n7¾2uþ/Ǒx_Zu.RT6#gi3Ƞ ( ω׎<[עkkK-Mdžs^-OgtaY@;ʃ,BHQ*.Or;@ePBPFX)+Q?JZ}gjz6Z~iq}}yp8-,!{{_~GӾ16ט< ,Z vzvqwG!Lz%qfb}e\Yȣ1qlfR[FkUv}Yi韷4? <{sXfu~)ėqZ[M YE<+֗R43ܯzsJEw(mvEׇ_gXjwxEs{ XX,ƒzƝZR[ OI?c?Sx##Ǟa[Zn4kӌMo}F 73uMq?|ss83,v>'{~MO W_Z};ͯd~#>u~2mo?{+j/Z=ޙ4stQ$L- |fU'o^~/𕟁aus\xzvwZD^}3ķUzgu~no g?emw*_Z'.Qmӟ/{ ?<)hsN|k$m[WΓYkv'|HuUMCM٥oج`=K |o~[ x9ĺtƾZeϊle>^j}{Wt&m8imo%yD>|?7g֓j|?v},z?5M'^X|5iڥGŸ@2Ͳ0'~+~_2:֨~4y(G #}I ^[M?pCmC vх5ee@\~|(xƳxf_-<W?fMo54_Myujuv!my?e~lj|x2þ+mR>:֤wmZφ]x3Fd k20ҠYmaxcGķ24ټ]{JߊeǢ@p?Dſ`+3na`ϩo-;yO庿OT( ( /ٷ瀾43>~wM/ZIo{SzYȐۋw3,IIus_Xl{U RSV,dҏ73k_çVnXneE!)\QVݧN_\SkcOa?(-I֫~jO_ u OĎ) U\&tM>WG .>'r?n(~XD'5p5NJ푆vVA%Yw(ɒ*V~XPOxO>c\7keu}Ԑ}EGao|TSA |1Ah~3"~.|+Ѧk9l?D~W`G>CZ<\[ŘGVaNW,XI?-4X>+З_P?>|U~M9Zaum!մy:k8QY.#-%sa,T-u<;*<];kφV>gEE B~Ƣ.Q]=\( ( ( ( (ARR*@ 0A;P~*V>o5- M_VW>V=բF o#eOxo:T-h?@EĿ#AE#hQ&/|o~Mſ ?H4t_D6 ߁i_)WW!?{ᑱgӵY,@7oz(kBNSjqMdڇo%ݼSGγg|m<)+Y|Sk,,Q ~BzmKj(iv}츱m ?5oEEJTӫN5)GJ/-4hqv&3/b;ӯ:JQ_d獿br=~6Լ /rPY]_ p9p_^IҷkCޤ^8g5 8yxf qjbI5Okt٣|(xXCpqFÃm"o,ǟDMRPGisj (t}F8 MMj<5]`ozW?mq/~||=>*]_\r<6>5dt3:qxJ4J^BQ\W$ߊYkj 7ʖb|Yu4?4Y#+`:2= ~X: C3d~ҍmu3*E.$a,Gq *e^K18hbam9x];⿆|+ďy6G?͟Ķ13RFRz^8woeу[6γ3peoeʎj_iG-]m(FKG&߈= ]n7Fl58XHzsp؅|>"uޜ5ų$ri,(U&3 _ +NV6s ( (?"Lp-?< %)9W P@P@Y_t vw,D?(ObO()q|u#GIV"Gz0 0y>3sQqU?d?~|Ukfo ^xgYuMė,;kk iI&~]C 7.Ko0hZ[=1o FAo WkXNڦy%4t h|g', <=Oú{Nj>"cy㙮n-OZW,uRb''x䌤 }q4ַW65{KTIJ, _"/dU#ԍN#`gB0N- jN^Y%W?Վ-Z?S6|u"Dj""DDUUTL)IMRmoVzéP@P@?_ωVޭ]^7ğKZڟ`>y{a(fxx)U`4 <Δq61^`+ޜ*?úoZU9a(jRX5 *GK[u/gh}?"|j$NJ9[|O`U0閪PmR捃a>y⦧>M7}#Wω,ƙo jm%E-v>3|B|I&Þ8C|/DO$z3wkOxKƭ.%|~3^95K i~{m+^#<;c KNM6ZK%A 6#13ou- W u,5S▥m+]OXH$nl[E1lm[|+}ix?{ |@ռUxG՞[[K.qrCtkmAmp[Z 1Dlwg'|2Tyw|CB,OM𽿉_Ci3i%-c!/D#ŝe7&=-*j繳> x_Вj:>ZجcpZ"6 ~/ i>-|*OZ!񗎴Jkyj6&ľ%{B=4g1lfxIy~(Z7,V.$1xUo4 :lV 6qFՕqYoᛛƾG^.Mwyݐ54Je6 ? < xHүfv+W~(M7t[;BXf]yuw$KOs;\%l@P@G40DOs"Y"t=UpUE'$$jeөRVIRM3eqZqki?_o"yrelEIʰoyFi/$Sǹ T5qձXxpعWT`?P4xBumFXj=00:+ĭ\=Qޕ*9-o,=? (f9VFq]&9.w6|=>1b?Yjח ~{KMkI|tϑjN0mՏL f!֗[ F_[ƭ\aYw'ß|G }yahΕ0Y_ݪߘOem?T k6:YaS- 8~/FD<0[,&._:2զ?c>D|ȼm,foO!t/a|"o Wo+)>˖yY_E l8)_?iGva ?׊T2,΅_:abOI !~xgH?#៊>:Oz\[,#4@cL6:YJ?ܳ{}[5Jm3?~)[ ﯬD𿋴H"[[8 {|j7N-y v+s*9*Qt+K*H%'şY'Uhs<9:R<+Ʈ#Jtթ䏳X$ ]k>&i$"֡{ZZAۥs$c#NHb 8Vatj`150aZNr())og/pmَ\m*0VTN2JWɇ_M~:G?$ϸwZ )zy^|\,Qr9Tv![2XӟqDkOӊ2<|ZyIF1K1mGQ_Ͼ&a<`"[6N0Kԫ|?/cTiazeQPV9UkUR5itlo-uh&kDwKd ! nbg~bcԯy-\8TCu]JkSGQ~Vo/bQa2|&89Jiκ〧φaiKW ju*VP=YA++),<'moy";RԮ9diQG 6,K&O/QV:q+PaIIWqJ){T^OÞ&K-)fpنc3 ` .xqU(KΕ;櫪?Zơx @KUP 55͗o4ˈ Mbr(RըͶ%$q.}QqFyd2:Ng˳%WBJRRd93cX>^'C:CҤdaҒbp>GB9Z]8%SV[Mj_;RKU>(83|/IG ў 8pgQRPpL'b 2ϸz b. qwS\{_:vG72V׌(RhER]j:}I浝J#[v)d5jEuEZJ7*s`er9@!)9I⧁do~Mx{P|vԿf+g>f߃&4dIլ{Hȵ{{x^Nimb* fk c3UY5mMOjZA\mwkPGo,40Yܚu~*O_ n5> GKtW𽵽tҢȷy l|0GW<[3HJ~?uß? X/|e}Z)5d/-o;]5ƾuK,AW>*tCƳ|]xN:G7?5ZI:.{/Y#I6=?xo(_.͵MGfכZ֑j5MZo%xZZ F)Z6<:\xM?oko/.~G6oca Ai]+NO -Bؼrhׅ/5o_{c~  xgS~qx麈5N)_td6$g࿇w^ռxoľa>07m𥗌Gw:&Dڭa|# @|o_ ~xex?œj(6}k\QֺP,R Xn Cp[ f?~h_P*s|*{M}tMNmWM8,7Ymڀ6mc|ּS'旨tۭ^0Xvial|ou~R|A|8=A|C+7׉t_k֚5-Ykt&,.ռ5{-7R{;BFE]ou_k,xS0zUi+xgt7NZvE+Ymtఘ~3 SmjXz0[έiƝ8NI|)(FRo1M$C7wW' m;yhx ʼn?e4/ Z$&6y^H< ?9<.qj7exxrҒfV u*,,&ЍPn֦[iN''>ʾ:x]o>/~"ÏDPO~557IR)&9_k+ei_/Uyw{ǟ|1_BOxVVxn ĺ]WWnȀnyTU⍷|~6i97DԾ'k .uK}K=FjGdn]%wJx kzόu85o}cZWlnu-3TFJEsuq Cb[eG?zh^,5/- ?_dwͮmftQFƜ3C;ˉ<>:Hѿj\G _ <0Zz.7@L'K[˹u-ҼMvC').g{~.?|Y}|]mu iWk{a__ gx6g|c/_㇃u'+}EÚމox:A_ F;IZHb(bchW? ~Vmw:uΕjZ+r.TU8x捒H%H՗|ˇLuav:RH9Bq%(Na*U" &}!C\xw?|U*/MGᮥ>GZ=x%N2oxzKw7}k^YKa+PVԯ .kG /g[KlRFS 7I(NOQ{~pvP@P@P@P@P@P@2H7XXVI"Utaс  (|[0Ξ9(&>_zƐX蜜yX6‹|$5χ ~ Ĭ: / x_T>mGv?vX|o_~;?.WB<p9ĀIy_迻?g^. <^:UQ2G@ _K7TIl T3 h\Y5W,n{.mfg TVN|I)O]Gҽ3xz}G>c<6ڣ 7yR3r/b!u'lgO ftҨE^UPTSt-%Y~ļϸ0Q9X)'5/VO~az{O>̟&?![w,CVѴdYnuRnģ+, O=Agҧ(ij_ju/&R6sg>X,ڗF8_4CbM?c)ʓXFX#(JU')7%h(KbZēV[MBbc^i1KAnad沬׫pUO9T˱Υ£SWST'N4ܝE8F1MG?p,>=f xV8< ‡>aܷ/x,d4#ॅ֫Zt&!< yqxzK!XҒQldBi[0q*yfg4)`aUkujjQ0Tc_/Ϊc //7YS8 0ꛣzx(C3}{+*x x|Vqo.McDiڭ1ghkCy3`1FiZed0ӡ}W따e/m*:*œRJbIپ+3Ljupaf4ifr/c]_ V3 SZ΅J< E|/L&@[I8 =|vI;sx-#)QfR ~ HT`WW Hp2OIeي@O;9`~0cxxhO&7~_oC>s@:ݻn"7g;>B,gV?8j'W$I߳ۜy~bb\XΟ 8<'F pƪ?@  ?|6CX#UAa0}ρ>z|fbL}ς\L!Օw{<J?@Y_HZ]dw)[Fp I/]  ,ݺ; vg Yo'>_xw~[ xž)Hm3OsXYϬk$:ⶍ3pdž?|q⟈2G  +n=˧_7KկpϬR'̵Ôel&1SB39V jQNPMԦ8N|єZMLԜ$dQvGgt;Sğ4 e#M*]Cw_ xDsŏ4}Bܫ$D8?+yGqUT<^>a+bO0˱T FFoY4i)h;r0PYܜ_kJ-_uKC׼M?-k$G/gAѭn xkR~ۮj666&[UQ\X`5{? -?g+ۓ?/M_"<\y4rm<9;?f/᜿ nY8~~ak~ 7J\x*1ZK"-EqNddd)c`?k+ᎱKX캲ǂmt>mt]$짒XgLmg+̓cپ #G_ >&Mτ>2ԵZxC,t27l4}~K;7XյBX 2#x!ǔm DZ/iZG>קNT־%hv>&M ~2xŞt!&'g⎓pj?_˧"Y|A |UmGPŤ>|OUq7Փى|,T-]&m ߆,uSM{]5x_2TlE1~9 ٣qԳo¯]dɠb7s~<684;cz@G 2L_{BI$h3c,X㎜P? 8 d|.͸?eO ?|qnNqX z?boىH#ʐF|ex=!?e$vzE@1|g.=7:v{6 /?# ߳ʜڼ;ovtd21zPŷu'6c? !{o_i2e6? " m>x،c0 `~T Mc?wû w~ ծ>Fo ,4=IK0,4K,8jP@P᥿*GO :yle;[-bXZ% ^?28aI71?"Lp-?< %)9W P@P@Y_t vw,D?(ObO( ( ( ( wZ~ ZBY47>ȏklI*Ey{k}[B !݆[<#㌏SRܚ2y%T*MΪF2,eكSYmJiVT*ajKB)Z~+okMuSI5-}7)'t)Me{ΙNh-+xCVU.Z:zrJ'RXNrGUԫT58w]5ZMv;:B  |3<7x$2xY?XĖ(? ^1.-tyM:wV9-bEE6 n (I1EHI$qwwb"$ 5Q(rcm$նIjmV[.|Ԯi_+Qq b1aB ,RI(SN95NRQIGI w!Wo^gBW{,?;`??V|Ns$<C/3v+ ?Y>'9 }y _OWY;Ϗ?+]z|EGl4*ωN|y?_G^gBW{,?;`?,ok vYޣxoYYXMՒEx!RePL\z4i΍HF7֊W{g_In-L>3M񸇔*X|6c^{UJiT*s.XXBSv[_UPP@P@P@P@x'g 7>f$:.|w l||xfCՒAW5[{|$fYK3J|Nq#cΖajM+]WNSuiJXz ZpoG9iOƟt/z m<-oSᏎuHlfu4\nm+5Tz|,od\O,.J51j䡞4'8/Ҵl牓NNLu *ю* j*+EOwi..WvH0)zQ~e^2$.`AW =o8Ӟ'%ʨ +b8kϢΆ[^2qaOGƳ *j5&Zѕ/iE''CƁiGYtTNA|c9=GK6q]X8bOhAc)_?ɳ0 z|uwmck2\^OzҵB&hԯV_ )SSoڊj)uz+o5~ շ]Ú~0+~Z$5g$fԾ҆ $3f/ dSx'r_f/_ 9|p| mJ9Nȼ_qG+Ѧ2Ԡ\>:t}X d_r$O398Tt2:<3yJҗ-}jM0Ius[;*kԫ _K/57/n&  נ[I$m[Ov [JPp6^#d-GAKbjS8Y 7J]9'%̀{h+qEo]NwϡW^{ٵuu783MoO߽63 ߀Md jv#_bf87z/&yRT((&>c5w#lhQ(3%|twX3>6|J'硑(\cj6SVvT$^YX&#UˍvMf$VTI3)|`cBgmL%Ki6r[&Ryդ7qYL SZYu~{TsjKʴi:%S19B   0Ѐ<*0`j$ #EVio E@ ($``jH"9 @U@+h p_pY |"0A@ <Ѐ .@ &p(*0@'/)O0p @P E%B Xk  Ի' .@luߝ!W0'S !!\p .`8J2 w W+ 6Xr "7C D`!@ @ t+ ]ʦg,͊d@ TЮs/@% (0 p   P`*|/:+"N9D= U/̓N6 .^o$`3H a@#x vA  Jpl+*X]@- UȲ-mwALez@l@*18DF  D@`z@$J|萌;DjD,[@ hg;YO0D |ÛM<@}7Zv !!TE^ߪ@ c/D [0pvց,7 ".dUKf51XEP-])U k,#fqlaUJ~$mb@*P404 b 'm$2 uPXJW`JJ Lcv+n.*F*'1I[B$:,%KCpiYS-&R |q!hEMl^[d_T4XJ Yj)lfq?=g-{c͒FJZh@Ac[[ -VS'` %$21M[N=ul} ⃲Mi"ĹnkR(nbֽh Z,2eOd3Ķ(oX'2@q⣻[bL. nwzHd<&N0N!L~/)Oag1 X" >R{K_2B[G8(wAG[ ٳ7N?t̑'%r?{"ʞuف2̭qxΈF]kAtX/lG…-GMNӍ>4emI՗^G=&UxzϴYR+N5\3^evw}+^HO5)T+x9Q": u#` UU[qVUXVǙIiYьs^n ;Il>|x<S!d=))hR Ërweow9EiB6Q|yIQ6!mX[C9XcQ ]! 8HXr/JnZt,@"1aLcJv2n l&DgAMG-@f4{Y"֓ۏk'PuBH8t o @4 4 ozJn7*Ϋc@YpY}IN[^^%`#K$8_Ut9?#A.>cjږbwb%p& /V'MwcDd ESB;8 )BurjvP5FPPsԐpo0T   k?H T"xQ) ֈٸ' 1RvXx`zJ+:(+H 9- O0o*qjo͸ >-8-.(N D4$Bk*  DW%0/0MWa5#1#)90w5^S -3 Pu1x@bF;H #0gTt֓3;3;(v9IGnwrw5Vt="J4?M2WӖh4U9ЅB ) `55 4CHS 20;G|s{}hVe(`/gH+4!;(z@@`=7{Vy=;7@>:o99p1@8*`N(83Fb % Gw$t4D@>`? Cmq&`1-ɜOtsd( @(?C/` `?Dy )041'w ESA)2;Y +TZJxI @A)ܥA=%pHVsj$Bն &J%7qQ1pT&$""z&Qp[2r?gv!n8U {"%~;jbTse"devenZ'w]W1"O^һ4]aYkl!]һ^ŵK k[E\J[ڋ̣W𛼮5qpcb ]1F5a`ſ_&vA^ <\23&T`^/Jf4!TR\YN\4A. \JVGhW<^g+c[m Y[lbX򆻉Xǻ_ka O ALlij' ѺC$!3QȞܰrK2-wd0XGO !Ws.ĜaQ;pRRh<~fbȷ|VL½&+tB+ai,͞A͉"v[͓ il2s e; i!4@sVi\  Hbˋ ЧR\Ec'wY 8"F) \ xvZ.<.1`PCS4G?-F4 $`E}3@]y 4*2HԊx5y>zlnϼp#=:S:Ssm:-64|7(Q U 8~|JAyջSѺab:m,;ad"]?I> p==NRT))0*`vuDȝp.v;bxH--BJ$G;=]gFną}u*hl-U܆+80RL0 P:p%K`=㙘HȈ䈐-}-AܫDvt+iEga| ٻ04 R![mHS~Ę$ief,+x I1zQ4vS\̷K{"Q!a犀Qː %N,Rel_Uc̵<ǽ?{)̬$.릾 bPdc+*Ԉ#zNVUq h ~ 꾀u&!n{;fzƵW!ܓkN #MST-X~qϸdY5L Au0o<`` w= ѡ'\ll&_}fY uf+` A҇-(aBbHrVI bS1Ccwy @_d) PÓ0xpM>5)@Ft3@d/tm oиP@7jݩ{7z#hC;'K#<g< j=ӨBs!ޯ㜤v\}f@5p㠗tIF @d9A0Hb9:BbEa)(C@6=L,PE'±ׯإJrqNoS[hя=+ //:<7"!*+0%"5%0$5ۻ܌6/)1"$:/ X!N("0 :ح˗+Vm֚adcRo`B}>sBelݯHj~NWQ{-L{QZAi.6(GV-ǣ֏gʴO@Uy?l{SVi`>h5M-j´]7,?{){;y|$CbOX-.z`GG~V=_?6}1K]! mQn%~.[ǏG#ϕhi.O@qBm;Iu>}1?jZ$Յi.z nX:Rv )5'HZ]r83O٤g{пillcڗ#kyB 'ZK\F?Z=*=]V6?ޅN]Xv뷠}c:~44I ]vܰ2;}tRkXO-͊?U`9"qfIZ@~Xǵ.G Iu 07 N?QG%?O"Uo=~{>Uz=amd߽ %o@KDtiihV)advK4ְi#Zo:~iw>r?D?fwB?j\.`n#o@~Kt~Ei.rG>6(tU﷠ |~%k=Kcc`Թk[-%\0Fހ18E?]W忼xr1V=~*vô],#񦩥MXV뷠c/go/җZ}h}lPo@,4Jz o=r>ַZK abq9-xc-%i{n?#TՇi.zXF?#MSKD%oAM#J^_!/浄I|ءV Kހ.X'i4{-L{R}o _Apqzr[+Iu_ GZ'ZK<GЩk]v|GOƚ5aZKނGoCJ_k ?C}\##N? i+Yt/[ZAi.6(GV-ǣ֏gʴO@Uy?l{SVi`>h5M-j´]7,?{){;y|$CbOX-.z`GG~V=_?6}1K]! mQn%~.[ǏG#ϕhi.O@qBm;Iu>}1?jZ$Յi.z nX:Rv )5'HZ]r83O٤g{пillcڗ#kyB 'ZK\F?Z=*=]V6?ޅN]Xv뷠}c:~44I ]vܰ2;}tRkXO-͊?U`9"qfIZ@~Xǵ.G Iu 07 N?QG%?O"Uo? E7┞,4? Oiz=x? >'|T~/5Fl/_^{[ihF?Jjω?gg<=>1t[hϭ|?4ω7osMwXxY7BNn.O ]?I׆o>ZϬZ.|)xtGshz'}Vᠺ/k;ᎅo.<<0.c9n<^OO9і~%vK3 [/ LֵY4Ceu?Fo)]R{}>M;%)-?yKK. A>-[f Ƨi~Q)k1|Q\xZ,WR_Ae`zխѧo!$}G?zv{-/gu׿uFZgmk6\t:߆.Eɫii׺uɿ%v#S4UO< >]xKwxO:|/xcú% 5>Wt_A:v? ١OԵkk>E}Z۾_-?ψ>Ӽc|;MӼExc⟀dOIh{Km;7_c\Vizޗw)54_e/֖ Am[x>*%v~%}{KƩcQgMRIֵ=7Oi"mmx/gx[W&UO fZ}'펊$>tX#۵C-nQ{ }t?o7<'>.=~ºee~Q=q \Zx6ZtLIvыxD4UG]ͪ^hpZ|֝aak7:i)е-$c](UW+ZzO_p_?Fʾƍ?}O*֟4U?G<xjٖHԕXYK"Ȍؔex^? nz)A8;4]u"+CU~Y:_vׇLZA2yb]XgY0r֒kw EA?ξ_Ы*tj2Rs'BhjqWܮOM{xN[YE mcg1Tc# uRxʑoRNS8S6Ux'AFqd6"U)E&kQTʛIlޟTqoƷr3Kb}F~e=樝"KGu\ztiM+[`KZp7*;cT/bHQܜc񣚢.O ](樺m u(1yX9#Qr@?T>ң~49Tkȗ]R?1{QQ;.DŷҎRVȗڵnTwsT_r%Շڐ 8G5E\+QQt!QcOrGx;?9)<}}GQhr+.y~cw\/o q4m/k]ܨ1R樿K F;rqƎj?/Wt鷐(%Ct#E`qEw):~4sTRyR_J2SW"]vJ3(1G5Do_pJ;#MJiZ"_jQcwQȗVj@0 vArx_G5Eo PKGp`F1Ƌ?⋕Rth!eGʧkXD'ڔgQcڎj!r$-.wUǮGOƚҵDխwr>K+.>Ԁ`?9.i_Ҏj@Q Ҏc=+ܤSIK~?C*8ʏϷCNֱ\uO(̣ǵBI|[]](\5)kl}[Gp?1}ʗ5E W"]X}*1ۓ~4sT]~ҿ˥MA.?/4K${.WHQ񣚧ʗTqoƇ*b뷘jQG1j9'{|ȒQW?jSJ9V ʎ~c.jDRTc'h樺 ?J9.y]D7J;;1~4^iH\r>G5O'/*?>ߍU;Zr%o1>ԣ?2csTN %mwt=r:~4Ԧr%o1*\1\uanN1Qt'Jx.sT]6nwvch,(^ }GOƎjO*_kiQT~}vKb}F~e=樝"KGu\ztiM+[`KZp7*;cT/bHQܜc񣚢.O ](樺m u(1yX9#Qr@?T>ң~49Tkȗ]R?1{QQ;.DŷҎRVȗڵnTwsT_r%Շڐ 8G5E\+QQt!QcOrGx;?9)<}}GQhr+.y~cw\/o q4m/k]ܨ1R樿K F;rqƎj?/Wt鷐(%Ct#E`qEw):~4sTRyR_J2SW"]vJ3(1G5Do_pR}GOƚMVخtt ao3^O YpBa[OCC s+h*3x,^=w h:bëX*ww+)eF9C|P~arS QtU*+A[}n>߃*-{{B:}<9Lj4 $2]ĺ}372Ioot(-GeE{2S\#TvW<VǠI|?t;V+oO_Yk5Aud;9dt)$&%d[~Ӿ.|cO<5c|ڷGaw=YxkP.ưjZWZeYEVj[|~'tzM3]5/O<#:>M}_xCǟ +|+.QqZS|6i^9ҵ/gO;M+N W쵿|=[?J|@τ-ᯈ^ ֿf~qW/(|M-G[ݼ;wVWsmjl;'3%4|9?~!|2< k{?ϫx~xi6ڷ5s u#T6 >|eCuxoki*϶³ky%w%_>j^>x[kN=~՗ikko-iڅ:{g6fVbh>_ois-- t"|7-n$CR1©}J“m.؎V j| Gÿ<+y/|)ߏ|l𖘞G_xᏏ|9@߇t۽#^?/trػ{ڇxXtMX>hv_qam|`Y]'>^;iD5}ZxŚĆfÖMG7㟏&wMGTm?|]mOZLԾ",k~!KPFƞ-o- /a Sޱ_ψ-+<6 &v~X:>^gkkЮ%-?\j&mo{/x*z/, |YᏌ,/6 _h7𼖗"gKiȩzqiA|{O^"u {~OMt~x_CK|3i,|;Þ9 0;>'oo]yM~ϣ \p+Ǻg'?Kͤ-[HPӡ%7~WmӇ؂Kwٰjﴨ9F;k[nm#%Z\co/#tadka߲Cc wrZ< 9㢱 <:1e\L J&NQR/(<eJ\4(ڢv|xOG\c?Wl[ڄĖ7Siv{not7$}9R7ua ?VST7,OUme̫a*F(.^˪wO_Ē͂$ޮ3&U/gVtlҽWW8XA{isb%kZޟy܉Æ"U_LM?G*f1w0v=,~_L]= Qe0ka1c(]/f8a{Qʻ~_/pbTkzr%9Wo{5170=÷OcڎU0kbno({51w01Eoٯ=v=_G*{4s9RVaȗ8no(]arߊ0ks=j9Woٯ鉹Ȣ˷a]/f/m+ i?ﮧy$e=7/p_1uqTWIyF0iEF%d;P0сASAsOE`Ӿ$91ԮO!w#$9z猜rI-k{$&#8'zP;?/?k~$Mƾ3-=4]~!@|/cmicli4K}1mPc8\w !{ c {rsགྷK?yFvײ m@ ԃqvt(o|yV @끇 ^:c3J=c~мm/cSߋ?[Zoگlt{NIMqowk 3[\C2yC.,?20}GNqzVp@ Qt_x;/!7 }l+ᠻ͎$}:uvV*1N7N:1捼Gی;=КO[;l?i)úri ]sTI&!Lb &hY#kd`?C\E[qaJawE|/Vo.NmS(*OݩsEҵ֏>1]OԤc|A1/u'?5_׼X@#8?qzI!xzt: rtsda;~ Dq#i ddy>*db)/utW y?S'֋ŌؕowGS1b?ǕP=|I!E4cmC*~<2x˷9j/_V5Nǟ |Cy?jv'OwGJNgq{1S'OXV؝?h?S?a??L2Oл@Լgm9 3g3#'?_c,_/?V7pV/0ǜg?d0X˱*4S 3綳d:?c-?.4?S0ǑdtE ?Ԭo+k<>%GE/??@_a?c.n'X ?X y?L21rb]h?Tb y|~2tEoO R3?]пQR><g2x˷?,e k18}?+~?1Ƴ|S'xE.?V7p}OVY?OX-]Rbp}OV/0Ǒ=@c,~#c'a 3g~2?g,[տGx ?@Oa?u?X?zl~V?ᆾ<gL1[.īӺ?/+'OV?G~.=|I!/o ?xX OVA2xˏ?r0^//AX y)"YBGCJN '0ǑY?'{܁Mx˱?.4_AQ2Ȼ@Ԝwʟ?~<~?vGE؟Gc'b y>o$vGE ?Ԭoʟ?~< 2tj>ܡJNO:Q y?cO"Z]Vt?ԬoǏ OqE؟CJNgg>Ğ2?PY:e^ COg(e?o :Ӻ? 18}<?3zC/g h+JTb y>?do?rվ_][?xԬj?S?a?>1S22N4?S0ǜg2x˷_2(RʟOac_#9e ܡa)~<g~2O"[BO R?ᆾ<2xǷ,e-??+A8?1|cOO?:Ӻ/+A8+~?1Xtmo]>lmd -aEib3m1“E|]v-t;Sݵ^sk>90AfVGKG_WU;?}Ɲtvm[F{Mgy{wwvm-5{kVxu6 qm{oqo4yId4][̫EGW oÓ1#sVKb~OOGtSqvd律^-oPwM0yr h~c@Ҳ_eAv׻1pFGO|%Z^g#?ښgoi+lߪeգz;xK 7ռoOO4tڠ;1KVDS;cc HK;@"I}Wzy<;mS#9 DMA?T?O}}{A:>ӂGR99<ݫy wMŪG3>;%`wMڦ1^8SLq8N3U+@O@<9jkc8 99zy.y+Gu %ŽսXKlU%3_" !S4bCo7tj=׻`si1+y/[B:WuitөjNVCT-2k. r-,[F M_ʿ\uy4O HwauF11wrFF9A/?/g}U?_G-Tc^ׇOL#Bp1S}'#h-PcQ1.9LW4+K[ {{;Ok7v1,Ū2\A,G"WGVS-nO'cOkAMw]hcizM_2mn wMŪq?3 'R;ե-nOOGt>mS?{?ݸ` ʲTWixvhp1WW3]^fuş쯴*TR>\xTTψ\_gZKfI)4Aw}ɷ?wM]9ΥG ֕es??wMA?s['#h-Sk2?02N@VZy <;?=Jzyj;]}A׵4V|ɵ7ڧOSVVk7ڧxĩZOo7"=c\c0kN/E]xú`-SRkujޟ飍ڦos_A?T1*:Z7ASk%KgO0?GjGOQkt_s1h wMk%MZ;]|uoOOGtj>?aBvnOOGtj{]q*V_s0?ӇQk%EoWkwMs4Z&o wMŪqQwĨ쭮飣j:J[ASkJ:F%QWW<;8?#M{][Z%mt wMwTt}KȀ:61kR{w:Z wM@u{JV^zz;|گ'R-M??xSc{\8ʅ Gռ?T׵Tgik41Qtt7o=^ wMŪqQlB~Z_A~Gd6z;{T*,ӳ#o6J/0Z wMAO1q*-䕼kujޟ;8Tݼ [4wĨ]ռ??qQwT+y/W;?}ƪy{6AS#%V* T]d'!?GLk%MiO0m飍ڦ=?R_evGtj{]q*>KO/"oOD?GjGOS~[- wMŪqQwĩ[}ȋW %VE^3ڧX_&o wMŪ\{A?Uoo*O}ڷnOOGtZ?%Nw{+X?yLc{J+@OBiN?=qO0w?=xnOOGtj>eE-D_G煮^W*ux?}ʝqu?}+/?/g}U??qQwĨk͡5[̟AI-EvC ,4p+o~Ǎ|ico xZ4ZK\_\Ekb/#$HGU~cTסz3A^|ǀaҼ;6mc,׉ii; &3ⵉmfw})=m?i?p<7hRzO-W\Z]h!"B5r:DvB@m/jA.atis[<>,n~-;Č4Qeǧӷ+k]v72n3o1FDc$#\e$nI.cԒQ+ ugis%:w8?^5?ßxN[G@֭HK Ԡ \2LrDdT$;9 㿅3> i:Sxu+ Ciƻ NЮ죝FeB$5;yZ]:>.Юm-4ou Sa#I}pq>Az],[itL=q_\\\\;K֛Y:$Mrdovv+ԃqlvI%kh|e~#?ឭ㟀_|Koڇ"!^~nTQm+=|l~ ֟𶇧>+|N~ 3_v#L:&㛽4IG-:-:-&&]5Vc׿kߌ1!xWWW7u7~xᗍbxαxڎasuOWL}x&Z?kkǏ?h~Η<'?_<5k[M#C>"j~ j^2[^iK\]xq4Xm5)w6 ݭbf_x;n?co ¿<##EEK<=ē'[)+6uuk{M%{lJ3m?(-w<=^9xw$ڇ9sڃxkLvm':h%ե/r}>??~;~'Eo5x*]WSgOs=1n"e~G5Su/S-E4 h|_ 8>ڦd~cewn-kV{'9WLjcJd{}S–? Cዻ9"oRY.M35]б7|^~([[t5(bJxJKx4fU)+#mO>--wWk:~ޏ<O/(~0STJf̒_kwڌ.|oLmx+J_/۾cFZ^g >֗o* [)nfoF Ԁ ?/ou?h.>axIN$p|X>ŏw>&3p ٛLjxz-|o!cs\hmbVey̡TT/)?C~3Oƿ'1oZօjz֔%ZX_˨(RK+4Ga↱ᛏà>kx|1~:sY7EKO< /;{ej?x- ATW~;>$/?OOٿA j>!M_ˤ\};:zȚ}/&Xc,01AӾ':~Gǎ4jo}[mKsW=C=K)m4ϊ'^(Jr^ ÏX}c+_kjv[x\N c,č@?G}bRѵz?_ZizFj"K}6=1",co/|EMi_=_75 +xQ𽗅O]xZ] Ii<;% 1l`xW񧉴ož 𽮓_ +6=gNE]CهUFQo%rƧw~)x?1;3j#ҭ_?Em5O7Cբouݨm5cY|L4 %ogkkIhծZ쮐gn*mbOj?4_x݅֓]vHc?,gSO6][7K$Gc3CzVGOC~ 7uWN5 vºHԭܿ:uiHtm@?Cw;xYumᵖ^ xFD4:>w5P{Ioo|A/D[ὤ> :k#ӵSmlE}^xu#{{khod-X̞#o׾+Sмaq~(R]~D7Lj_WZzK <]sLm Fͯ7/ŏt>8u+6=xc_j+.,`!Xa+K"G1ؠ ( ( ( ( ( ( ( ( ( ( ( ( ( ( (?(`+q;#qxϚ W#WžFCVbہRRo`p8F^D||׾|cco]5&[eh-!vbu|T=c̑h oflrB8qo$#19R +ÄΣ$t&>sEAZ嶓9eԵ+{*fE9zGݍfg9i ?K;_!r/thۭޕ^j\Y_[kqp۔2pTPC]ÒM?'»i<W-NeDA'uٙ;X"q-n>(u>3o5|Kn[K2ETj_01 P.[u*}m.1L:Y=­loΏieaY*o:\Z3[\C-{Ըz=ŝ_YuVEicڮo[xz Iow},ŜN^ |:i"Bp ;t^𾧢mk~%<[oLZflRiU^k'塷GeOZ~x&|$?i;L5N+F4=kU>çxQbԹvVC_&MPU~%04sow=[~Lj:Λ]ZmKM6tڍN쒷wg`wgϊ x_AƱ|IG!ؿBwc^[ ş|(6wC{7M/|Oq}ĭ6/j#uՒ[kM:+;}%.{t˲~BU>.ڇ>x?n<ORU(H!o/x\VPŲúh&/enm c߈ׅf㮁ua=7$%VxX>scv>ci捫=Κ|}7;߲+Q<~+E?𳯼c{<(ִkkV%WKIsfl&#»/׭7hTѮM]IgOiY\MCw5k[-Hh UNSHUSLѯ4m.~bYu=V};Y- I }yXGQGѮt{Y<oitK]_F5ݤOwg43,r#I0;GWQf>m&x:jKL4sGL !$K?iSom=_5τ>0C^{cK4&Oԭl4S隃]\__DbHi'6Ƈx6xc2.}qwm_~fexok?OscqhHPl|0|No~{ia-4O7VڄqjuOG/,h`&9G!m8xg/|E!>8ἰɥ?cH+jzιKQsox!AeqMW~׉hq.$04uխ7\ȾZm_|,}{Ri^kΖ> UŚ޵p,}ߋR;%6Vc?[G"Ԣ@ԭtֵԬmȶ%̷7-6/G| G '/O3ھf ?t:F4AI$+!6>O{~ !}_ڕƣ.?b4M>d!7*|1Q Qh^OWw_xQ|1!nj h(D~FD{ IFׂYѾh Ƴ>&kγi˦ȚHǍmof(]&G4to ǾGSDRlb~D oKǖH~ |eϷk9Qk?4kkh ]OWOOOmj\!cmwƚ}߄u/XkV]x*[kYL{K+]m օ{w43t6ǟ/u^ Nj>0x񯢺m$<9K}ysR#մ%D Jm|5ms,H#%1]ʼn4m FP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@5G?IJi_黁x9U^=+.2?x%}=?O$qۤv~>*}NrA{Hu榏wL=?k_4:g)=(]C54{ay_0\' jha ?OןI|-tq'> `0c]$Zi<Կz:ԏ @w_=$:i?O^SG&H/HK*nw/⚎Q^L&[$y[x}/M_5^>;=_/?4kƨ4?!j}/M_5G>kW~K'zo=/{Z_xdi? QiϨ?C#I^6jcK}C֯OT{_ P?%F@7mƗ=_/?4kƨ4?!j}/M_5G>kW~K'zp8ǷG4kWI+x?4mƩ*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~O/F?9+nb@ @Q_ FV+~/Or{b'8sBTtrZ߈{Z߽ cVÜ@>V9cp_Ks`i2$c%zenxgnxqRFkPW /Oحp|=3}2=>U}ycm9/YuHGW){*KOc Z*J/+~di? +osyO4rT_i/1F3O< +rF:q0z Z{w]/U?$o?40?}ӿ<Q0vbuj/{(GP^1y[@dgEʒO]y4m$dcǨt=;/*?^{~/O+o5OR4!j_G}ؑq*](B˯,WUkҿv2Nu8sÚ=.ae~j/y/E//O+o5G>i?C /O[g݈~{*mhem /O+o5G>i?Ej_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_F4<ح88!+}ndi? QϚ?/di? 1.Vb>ʟZ1jiy~di? QϚ{ZR4!j_#I^VjeK|?'#I x;iB?aZ^I5 4iӕ:~> ,/:UҵOT{*_^֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_bob3x:p# r޵A@P@P@P@P@P@P@P@P@P@P@Nן-?iKٳg/>)EOKW:>xwDUӴxFQӭ]5+,Kҩ nn^>/lapVX[ 3tI-ZV/짨~>-fwƿ|-;{O~']O i~Ѵ 2uoH]&5 ym'VqJQ6c WcM*T ѕ*]5W.Fzin?~3_O˭.xOW!𝎙kZ=SLT{c$v-75UQ+Zї-՝RhVE{NM8uDxW?ϱj$sy_Z[YisN|$R4m>Ko·Vq—6qiqyue=A}͏S:ra%'r|9O K-wпO |7=IQ|I*x^'լKx.nqi~q᫏JP Zܫl^{3 5e-̯kOIM+ώ~+(=keMK¾*a'ѼCM\]Xks%i|r^uJxvӂͧuOz7NG;o=_co,x+XC? |.-GGc>2s iZ$ąFXTx#n,4oe{g $z$+$mKYmo5ֶe0(F:9swkM< :{- sK䭯Fj?Eÿ uk'\|9 v:Vj~%uz?u+B+o&h;.Ac~}ybᎌ FkJTk+;ٟ]~о//PGh/GOY]2RUc@Ooo)$m#U˫{>N4V'B1"7n_<ϛ~84 ҕ'(F-{;n|27'<wnj|Ob~O N4:Q`层嶵xs:u(JU!wwoR傩r B.k[~ mO?|E Lj>hWaWF<*4پ)u m[uiapdTkB2iTRE^()ٻY{[QUգm8U$G{s-s?|+j|m?)uۏ ,ȋGյ 6k[L<.+&EOBtjU\Tyyo{v{~6*x ӥWVwG<OǍoHW+χΏm'R-;ÚF\5Io-/%;-㌄FZQ"2\K)|ռԼVT]j1!'):ە]zm+=_/_AVc.n `{\ǣE߼ֺCOes[ln3^Kf)UTGhMY_*:ƥRw̗M/|)~T y:|kOy'v5Se[] jz;;cuF.iU'GuS:.k3_~\%Z1-**R.:;}_#/cÚ< u_?ZY[WzmZ}Hio"Ki.>"T5蜔UIaq)jy_q! xN,e_z+Z  cQs i{K" {F8I's匚[]u"'V)GNg{l:۟O ⡫3⎉Z[мk CtCL𝄞&^]cQ^,sjZƆ[1Rv: ⢺m˗Kbt)ʄyG~Y9mgq%/zS>~o!AJxOJuh-9uM7OQ d{s#И-+-},F*CGWM:Z%wYhe#/؏/ox"#Dh_ }Y)xCP9mc<\Z75W^.SP~"{lߍ 8:-F~ז} Cǂ'ƽc[9 S|;3]^'׉5;9KYԟTחEK;)D{5S(uTk+Ur[XX Sknov[ I!j^ :V}'w7z'mmQ?>!?kHlQok~.~mo=\ur%ikN6\kcOo/ >oQA K^uOxQYm>#B [ VĭO>c%B^HWNyKYG{+INQkE_ᕯ~2l/?oٻS-__>"t;؞\όmgj_Y\ʐz&{ry{ER\}į]'E$5 IE%o-_¿l~(xGI4&𵦷lnZ>x?t#̳,UZTWyMuUiДv]tn| l]?dŜ ![;Ba̿}^W{N:y;ӷƷO?ooZ[ ~%<=֩x3i$@ eA4KS{6ea KxK\㱵)}_7MJ*z;;r]WOՂӟ=TddM4}yS~O|)"_-5i>ƣ?.|Mx|/%ޥswi4BxSΕHE˗QJ_ w忕9!0NRQd>%srkO+<?K/Z:|8m7R?3Jtφ֚n1ӵZ%3,RiݤMbΤiUj &uOFZHui&©-4v.ڷsᗉ& \|*?xV5gdKx %|Gc_jNlZvwZ~ n?{:5xwE&X5H^nㅄUHin+ZjYj-w4^x\$J-=l;ֹOCyυWZ{˿^Z-3HT0uY?i6-VЭbKXl#Y.-֍(;&qSޛ}'S*5ElW3џ ?j-kvex#w/o\j[B{Xk,;7,e5xGP`qnѦ!ļ4pT.w5ORuGHFQ޶ ~ޖ>~+?J>/׃|p f/kgVD!n|;[^mUkk.SA*m-JT~[oT<Μ'=MIk?{WgWxnƾ +U?Ko= ¼@ffR: ':tqJ)=۪ZܺFثukeefӷ"ӿ{-C_?$>E濤tOiGxR-[FsC- MHiE}yccka5*>kr{ǝS7-'hh~oC<_4_xWDGhiS1n1/+M7_X4xT3c oϗr{4'گeZE=>nG̯>HF 7_#kWvia.SKHgK0iNn}M^lNyTݚV+NW?F2I/ŚOo}yWNŏ-g:|iZLJE kZ[ ^:[\?6_ћ:?2\YtJ ZTJy9y{guψMைz+ mvI9uEԭGоK{O%F-0=<3jJ<ڥ7*q^7|M}KcX|!g k~>> KgI4}3Z.X}v6V7VfS褖ƛ8bhӫA…줜o˵[^+V*Tw>^mvz1e|u_'!O cм)}ZmSQaKo~.ֵ_|/kT MWᠸkb!Iø֯Nєi=e^-]:U*ThQnIKhR]ӷ?yT`O%A/մݬVмtoԶO|AO46OuVXتEFZZN6Zy{ַkah*ҕ Eſwnt+мy0_x4 ?b_80C}s×>X/4WwQZ}n/!ԧlhjR_s\µ6kHRQN#V=k7Ge# k:V/޿oQ[mc>.v@nKXT0WNuGO'UT,t\ks~?S]‹{x?]t?ͮ_ix?MKn^V., # o 8J:Wm+$kZvt* ^ֻ{.XVߵ?|/w> -> ѼOKkpumGzV+3_/yoYj>'K<1Ju(1\gJӗ;Uuc $7R?%~חwWcD+wK--O|CcOvs.]DIw&mneN&jUxEFj˖|VZ=5M|57 >3 Z>ӕJ״}۱~ÚKTRo.^|mQFkki7&41ʍ?JJkj˶xpˬ}O]JߗKs|0?ldc6cS푂2+V#1J8|ܮk~W2l>-ݏN~pA?c/-R實]0jM/ӱM(~31/X1vc?]٨“zYuG†'}}kiV)k#n?g/'I7Lr@I|qXcWJͻ{6'/kh[Tw}tcxQ/t XjlHhO6 pJNQmw#Fx|KUr^7,\yR߾~'NQ|.Mu<+?ivj>[V mEVK[hjsXzie$7gΗ-ݵO쯆R(SS0;rʤҺqwmWk{ k)]Xx^}*KNOwzot6|\i / }5 Yއ_>&4  Z_o^Iy=IJ4j9Z]|L RW9shࢯ{iM=ޗ8)OXwk׿u'Vej/5KiᰆK\,%::ʹ1ku椩iUgۤ}Ft-˧Gm 'φ|Eo[c;=_z//,O8J)bq R_|ŏipne)ÓU$y_n> stt~}GW '\4tuo{.gN.}3qˑetֵy.Ygm=O ( ( ( ( ( ( ( (؛>>"v?Z=Zsl+6Qou_^L͆VoWGޟ?Ƽ#??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@ 8`0qHbg$ӮPIwǚͷ٭|I.yv^8HO>Ў08nMߖcZmtge/3L w~O[uzkivE-=x䍬s$zp0 ZY%M+[[K`S2dӧ=rItZ/K&qk;T.lYxbxm$KktG_˩WUjU] ]r^W-C;RtjU7mԻ-ϫ~ x o,T5-k:uSWzvWֺz 5ŦŷˏBJ4?j~iJܚәVr՜gVU)اov)ꢯwzqg'ss8I-9TRvW|kgv-~?> #qA㎔mKKL3q#9c#t Bѧd=[ir]W.چHӁcɪEm}#=zGN[.^]i8ӎ@<@Hs(y{%۶Ukѷ}>7(j_?h_ON¿ ot=KŸ M kw +vK}8xzK+GiupxU_ҷ->]>kQMru쥋Tp(•R^׷wc t`sxz v?mq4胧8 1ǯ9FɤM[GO;태q-6v}dK]Z_H:vӯlzv_k@/[vg냌?O ;]+i++d6~cNs#zkO5m.7}t.^Wv?v}4}QrGcۮNOsnuȴY%Uk [v'NL~=sG{{~˵'ǏSg$||K(io{g~w? 4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?Zĸmḍ|}OJg?0kںRpdNT߅2xFy|_*W? '&}>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC}'zw>ɇ>aC} /Mghc@;mZshϘCI^6jb9ahϘCI^6jb9ahϘCI^6jb9ahϘCH G_r;f>s<(%+i+;|ik&(꽌4{{~A/No5Gֱ?1{? X?{? !"+ x9's ܕԪ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ?!_u_kQfu&2H+^}MV$~ӧiZkges?e6ega5Gl[8g Q< if5ձ~ξ犕 ^7PΤ%KQ-ң]>g,SWzJIEA ĞG#h3N/A|r7OF ׏??p4[?'u,i8=d Gg_Oe_}igZk>t۽ֿ1I{1]NXs{/o~wn_gm]5ơR)]v]: ,u/`xSN.~ mCi_ºuX2 jmZmg3M,-ErzZ֪NKNSlƮj:^0Q5c%9|&!G/_+>5eY.*QMuAC&.U;G~9k:KMc#|moxXË_]/HH e -?Ěޏ,]A0|>e2yuL4#gu2Yo1Yy[[arjK8'O)N].HS沲+ӒUܬNa[oG<\|,ڏt |c='u[k4 wڅ?xiΟ)_<8[(,ujY1"ƶa5܇| O_?,۟[vY̪'Z0UrYҍ%_QU{Q\ҝ}w|Wo7!Wƣ|atI=o5M/U'e.xVXuͶQ"Nҵ=N8, U`06X79h^.Y# ZМn+W퉜uݽTSIuc)%o/ëR?1xI74r/61+^j~-MFզ4nm2]|=}k3Za8Z .'19MJ|OGxlF>xՌ_uX]|~5eY.*QMuAC&.U;Gn[Z7~|<.| }cx߉;ommC^_J)?|-kCcPM/M;xf pfc*0x+ d3 \8(V 0süB9z*ҡgT Q5IGpKFdTgI 1|K/mG?/ˣGf9q_}c&>>( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (<▃mWVi3~ ׀Sx#?m[j7> ? +*vRh9'ΟȲ%+#d,`s>[|KRYIիSe88BVӭB" Ia^5' z EINfzu5;XGïXeGN6Cj^]#x~&Y]xL[5Y4"goi:,.3\.c] C>β̯hp /qN|LEjV9c14(T,\y[ N5YIjS6pKr劊\rRiۦ#|5X|SsKh?FcT_7C_^#'ش:*x_^ӊ3/WK2s`,j\ ˗Qp be.zSU~ʍiKصKEzVYkT\ҩΤ6TQ~;kVV#|71}ѼkAYť߲_bxZ[; ;~.&[[,{rA.(8XNN){sZE~*k>://S28k c)m$-+Qš#X;L׬t?;$xlR sȟuS?z M8Ls<7p/ [2*8T+SNcWlL+It>xƓ8rFi+^.-yf㦗wj u/>(>&h}r KŞ+aoXD4{+? h_@s-c] C>β̯hp /qN|LEjV9c14(TǕPQՔf9Jo w.X%/{| N_¿ d]ힱm3?SMּCc/,^}ݝMŠ<+vWO0dBW_ri,v7\E خ|T."R,XU=wlt #=8-y{ZόdF|_p_TޙMu=;:_־i}iچynu2WѦU`{0V鿴=+k4K[RP<}3L>4-?φkuRjZ3exGZa-oloV k1查WzmZ=-4x ޯ[&]֧xiڝվ%5?4]c~5?{|y~@g[>8<;gizM2}K÷>Yl 7[_ %z?# +Q⑨ [ ?S&kx&AK|Ct 0xC=业OW#XV-Ձ~G|/TK|Q._W3&k ?+?w&ؿ%E|zq _+h`c;koڿįO#f𰿳]nOj-ĵdx 5uOl5 L5Y /M.x_GRTdld6)6NW[A,[D!5M7B.mzE;<x_MvW Qw=z '<<\D$6{:/b>|]j_ZL:֥EuMu,ƇVq-+CO ]|q_[+%oW'ׅ|]q;X,4}_V]]CmML˧o_/V5íj_ $X|=TRhzln'R4.̗2Y~ 4G˫$/<"~|T~~%]./ݬ-ƛzB~P<'xS?@ µ)п 7!h5).nm|j[`Qߧ);y YUKK[lp:%'u/~ 񾷥|=Q k/O ?%?k/otŦ7&Xj/N /5o o-S@|K#gn|3ys=֟y8|ag.ѭ{tZioZn skv@е=}γ<_j^աYV_}>X+Ft<_׀T g׼igk |+X&? }׈|9e[°j|8'SZߍgJwwZ7;bgn,w^⦩jڶ5/:-NPzl~eiP֡񟊬Ғ}CSKc%rֱ⏎ |hF7$Γ#_~emX?|9{ |<6厳z(;_ _~%xk?W!Se;~~'yhT?g:)-/}E~m F7߆ Mһ΅í+ύw[/ŏ?6}׏/ZеAӷc?=?ŸHF-E)q x5Xo kCßh<*ǎjoǟh}_x_>dÛ}u|#}#Z}x\j# ԼEe }VU]|tk^SΖV5oYK_l_>=|7' Iд Gw*i.} O¾>x@ͭxTԭ ˻{-jK 9ZƇ/x+ǚ|_pXICxğڦOmij^-iWT7TwiIz6dKa8' |-xW!oIx/j:~[ifۋ{f$ s[/ #|+'xj2Y$/X6O}+XmlgyZݛ|kL>7u|;Peχ|c?~6z? Yj0h|BլTM%X2jX9u]"B ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( wwWȣ]1>s%WX3 [:_,# Ծ[&ߎ|'FL+FnZ|ҩrX x3Uo?; |\Oq3 zkny? >-_R6Gcxz<_ |be|:YWk#:?YWXmn x'ku +7ujjz6e$^.5'6t̴ٓǶ?|y][ I11M[fjs> #[]Em-Gi~@ė&!%4?i/xJoč"-R_<}Q5-x}ht ЦimGjy6-.V7_|w^,Mk߲u^CM߳Ud^ ݶXx]mq嶆m OMUXh!&4ً8n5W"RhXP{MaK׮-/3ڽ֝o*<=tm2]O{jGo ‹ 5C;G-;I~#Ocw`> խtM'&?ٴ.H5׷Ȥz2 tk T?u+ZƏ(zV=^it<]𞡯[R1:{̗i6&rKX|9? Kzß]|6-b83U(r;iSŋ B (o>m-kg-_௅ I5<)%&RҾdMr9--goϢ_QJ$Ou4, ؔ_zj''?H[[H5MSy:3o!pykǘO״_"'?;w~/xo%c<'uF-/dQK&wj) Ⱦdzo)«C@.?WO4/!Y0 ~,hO5y`O쯮n{N"xS7zx7"YY?f=#w_ /ohxRMUTmW}-r|K}GTe=2'ݯU|ys|{4Oex2//{ĩ7| qˣuWdc?f/xE|u_}KbC=6?هG{/^HjZuۆu=;A//O5 hu?<-[_ 3~vo no Wɻmm#ZxS[Ӟ]5ڲM{ׇhOYY$|(.K}>7Lmֵsv5|=fu3|u_ux^A#/ Ic_o|g/^eֆ./x𽞡E~Пsu~g$CK5CQ8Ermo![~<+u&-Qύ|x5_P5 :-?xZԼ52~l[^ݏ~'s)׬/t_ gx~!\OTҾԴ}~ֺ-ךm4wkz8s.#t(>|@>&e ut~g<];ibV #ckqj?dؽ<}y! h)iՆ%y __'5_ <%g݂xͨxP+CN"ҶMw|2_^.|gsOx:^-R῀~[|>мo厾|ioX \vq&"myE7c?ayϪ/M/6 5M/Q%=t3&ֱ&ssi"뷑M%?j=oQ G0? ?W>R3|:iƇiZg|3o~=VOm7<7s(1ѵ޼>XZoݺC~^'ċuea=6B7ߍzl>7^ui}'_iW qo|u:'j~|hzao/|C/CߴWaJc^]x[Bn4s涚mwP-UGk'&Vu2>|R^!M^>&xb*յ Ou֡qhѴ+*k4_hkpgӯI|an_#_Fm[zO;_F&>? ZumlhGψnn{ ƚ󽋺]6kt~ ?|z9ot N=%hoOTF]_U4i'ujbnҢ__ޖVړcmҼ#eg(ּ+kΥ|yĦğKυ/P[<^xkCԞ/ v>t߈ أů|Cq?/_G$K_W }[X|I5˫Ö=S_xv[_}5bWJA_/|Y k_?s|TO_eBksh 泦6c%v]bPbP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Pc](O>e1`gҿ@pw0d1 ;z/s@ ۃ{N9wL |ON$` 8F9Fxns6~q=9gg׶q@NNW8랜:vNO΁q''徼ɠ6= r8$?@1x2phyXAnA`va m`s0A==נ䁰<{.9cO6q،vd ^zp:``g~IAo ϱub>{`@9 <`6@ sr@`1w'z}0Ӷ12 86}鎝={d;p0I'4~3F@tv۠om`'קA3F c__S@mq،z`cӧg`,c!H 0Fуv~cpwrIs%yb?>$Gч>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽ~w;;'G} qg?q8;S2_ԎLd~G|ޞWz$՛K}"Ny+kw}ܕ ( (8 uϾ0;84_+6}:ucK[%wdikn>Ǩ#+knߎKMk0F=6F=H -X뾊{ۙbwkguk7AHA@P@?=❺h]&ݓN;ۻk=tgwI;c2y `g}^VKt-%mbnKVՔn㌒1#F9hu'ekyY>ѧk{~W꛳ç8} $:Z>-ew;/NZ)( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (3#7" 7WK5>{KkcoqxV𖀚}?Jַ`חP^oj';cKiۏ?Rx:s8zT{'R԰ ~Ҥ]JiPIMvGTGЧ'M:T9){Zuj(MӤ% Fk~(T;᧍AkkZw{46YgNdO[WҢ1r/eLf=GC J<5hѓ5*&ܡ(J"YVrߘ9W1UV hJTW^2IQTJp9R|=\Aϊ-w xLMeV:桠Oxl˧Aԗr$;>a u#*=i5:*N.EZ):^ ԯACe'_>4l⻲}[Cêsk0v {.y,f  _*pܾt15iS՝.iNoy0sؼLV5SYʵ 't^R:<ά=Spt;_i|M:~Oxe۞ ^ 4mOQa.6bM=-aPxTa*Џp9Z)խMӍJR,E:8F*jB55ZX>#N.N?yzR\D8ա'CGhRu!Z6m<o wH ,_þxH F}m_.V{ 5*̮0dWH{ $FKT)GnNi.R|X?5dW6ljdž5_c?v_OM_:'oukX؂i]V˕<=\O|p.ZsӼg.|嵝R0xZB||*5 UzPy?g9EaUƧ'Ujۿ|}g_hgO ' /;2Cgo5=7_ԯC[bSj]Ǭ7 WƯ5VR5eM!Sb! .G%E熴Ođ:T1jv]J?Ko~sFH8:p^=_לRa1R4U*tf JzU&J*TiՋ UJ{d匩`q 8ЖӜiTU[N(ץZrFygRgN]VuFraYT*sa܋RRb躴T.IUNjrUy:O|UT׆-Z^'M ,t_:_գ;lb#IkLf٦ <%l asq,,JuQ,6'%Zq3' _91vxCZjڥ8euf[xD#lhX m;qXՆ4qr>OpS^J1ZQ/h!(rǬJ)ajUVhaAMQƥg,M|5%JOjp(BoGm^nM/Uu3žfeeح_w =*YR{;v-w۵LwGtQOB5OU刄i&RUSPr΢nPƶ"ѩB 68wKғRquiK5WtVhl0v>m>^|_xþ5_ KIhr]^CwzY9+S<& zFAѧRQyʝXZONO="X' CQgGxM rWN9hT*-MN)!<;Q(tþ4_K~}nG|1%g}u!2 jf\,>S:tJҫ)ƌG[ ԪJTiƝϽ\B(ar^>G:tBѮ:0R{*ҊZNUK9|JHotx+ƾ4ԟG5 ;G燯Rpxz}nG|1%cYGum 諙(4J:JNqNrUrEJtiC(էC ҡK*taFѮ,;u|:ƵEJ7Fj)^4B|/⿈ھ#WO?^x?Nw-֗Xiez5W[6ʎcN1t0xM$#N8'F*u1|5W9RЄeeC6<0&7\:p_,5w]W*1FtPHQS1C<{D]|AQ{K)ln<;;$^7VTf(ff[i&߷1J5:*erqN.70J J:zrrms)P~jUriNR6 4X:mThׄ(JOkҼE%_V= }CZ&s ƭwCK1ydqգyF*iaJN5RD0M%uy-L-%|_za Bq,%:z50\^^P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~KcPo廵FNJּ?s͗`[;^4༷+ky%_5*W|!dχl&.6&[>6w`dԴ[}w.]+ v/ Za0?i5S8B4ы*ѺrWs,n 1c*3_ J!UqeIYR:zj.qӕ- mrW մ/#Oy \W?.u]/ ۓiqo]mvtb/]U%uN?[^֜*Rx{KߒӂR\n\7zR<]:V|IN O!lKSI~|Bb/`{uhj^&J ŏ ^ 8V .XG{Uj{/r/Ϟ {vyf7.o}I{ W6t=Ti9點㏆>˟ê5몫RE|FuO?O^).RN? /4j+k>-JKY`-ёQ?U0ul,ƤǷpܲ'RmbeVk,WßRt9hX$a:w [+M=\Cl- ~/Dֶٸb!p`뺰సL4`ӧe"J)w]^Gr - :AX(CӦOiQ\ t7SG}e&=O[V|EO|%]|:o7u+EߊsxYp%Nt]o_FyNyJV\^K *K]+S2sȪWOռ<Õ*7،% ٦`Φ'Ybt/uK3u)J4:2']x&D%5-;ltkI4oxE5=H6ڍ>.t}wSN/9d8X FT' #XBQXBP*ގ/ǬƎc)W6# G\-^Ξ|էNT:RRuJ5,t.(]鶗s_i&!ov-=*.58䙖Ilmqy >:VΔ] fX].X:QӔb;;rb|cN(BR>vj2ƔeQJMNRs)µ}cyO/|#k>*ї^*ͯKkN.agxWW&u}+VW a#ib;ku, aq򌥇aԯWpZOiN0ķ^TONu=' dUSJWXΕZU}*p+ʭZ1tTt0 :x&#ƞԴkڇlukk+zNx@Aw殒XGm&{ex0pxlYWJg$js x;skdT rX8C/`+KxT5}Jpº.RVu#R#t Ar_z'|k[WׄGyK^xF-+Aj]x^{.)ff-mlEx,\+g k֖*HЕiSN"q~粅Ot˱+)BL6 ^X/=JU,6#u'i>ojTs°xwUڟӅauɍ#O|&&pahJj0ZQi8T%S~aÐ)^!aaO3i VjR*(ЩRu-h^ ."rX|Yk6+iaMa6y-2;i! ;zd -FhsBX-rVReSRQR j*;!2 HaT#FJ|'$%biVq:8.J5ƬiR~_xWio]h隖kv:5z <]"umFƗ:>d}Zrqs 8NFЅ'죌x(ѩ'8UJ:9:33S Eҩ .eP,-*Rjq50WUZ_Mm=s_hMBu]қ[mawwSsMk\Y=ȗur5UT1p|:haԌgmS4)J'Be9.51.>(* #8NS)R:P9%XMO[|QMSWn HgWDѴ~ |JԼoaX?oVKo+j:=ܷi<[iZw?F"Jq8ԝyeqCSv.xquMF,JFNKxիqkWK(T(,SNJdVR_iV3)SQOkPx0k^+m:^YxV4Үmt3񽊝^h!وu27^^q1[YRXMjNܩ81JQ>UM*)Eeeu6[m4 帰n4uɟEN?KյyfѮM^V!K.,%fS'X(R_GG8S[S4:U)5n%kT=Sʰ^ZlKӕ5WPҧ FPX^2ugNivP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@PCEE> 8Eb)/x¿E~~}P@l q<7M]_]uf歭ޗX+&NҵEkr:={[W쯧w˰7vkiw~RP@P@P@P@O8n:]_]W.i7u}t+nP@P@P@P@P@P@P@P@P@P@1ss@& ( ( ( ( ( ( ( ( ( ( ( ( (ӧ8Rxs@~[;;+__C Q.Ÿ9Lj?+o~WхP@t rO^?Z?AǮHA=x%Or`>}O p=xA@ w2=G d }8Ǡ$ pH.p988L~spv8<l=3=?Q39u021}p 팁i38l_u#cIwG4l\cI =(ӌ`88=Ӡ8P]3CSu MѨjv6t&"ıDw/>_}W;_9X;us_Qkmm*o.5C$f)|eV^=?_>1H{z7Yi> O, B4H'Ú@xd&}G /kz-O/O|BgOSie7Rx­|A.zFIfn/`ZyX!Sំ9GuYxUG<_3>"7#^:wǞ-oL4}h3'=\^x ]}+K;_D񥍧(,ۏxZv^ÐѷV0gZ$vkֻt~u'g(x#5W5 |VH_?x: !';^ Auä,kOF6ne~vΏ&[K׾G?|% KH\ϸ?|% K@s> /6 -3_"_x73|"mRϸ?|% K@s> /6 -3_"_x73|"mRϸ?|% K@s>M/Kz xs:]ŶOcDt >mBY5Kiommci Z…F%5" Kt;zdP@P@P@P@P@P@P@|CcYހN8?N9p }N@sϊ?g=3P@OlN:ӯP@P@` Ӷ@G{3N2H$sq:{vc#ېqq~O@:H'3}p:gF?$h|>2s ( ( ( ( ( ( '6b??@~[;;+__C Q.Ÿ9Lj?+o~WхP@ោ&V^+_c_ڇi_k>t=s/}c/F{js$)FCO!NXG|#xn~$|@tsujb|-=vt.OMR{mcsuF cBᇁ+?Xh~d?u7~,x=b:&>%u]F; I m6jIm G?|dX_zx |_k&:EWiǏMŬ%ե׵].5B[oԬx;ڤdž?WsxGXKw]x7@t;iMlteiHm-~ƞ׏4߼V7jVk 4V+kož4szVi\iz~iZ|>S5Ҵ73_~߯ė<{^)~ƾ>3FRB:Tw:!oD^|uI}u+z|QmoO+]Otx^Keq ,MiCA7L7Fo+?L4?ǿ w&:|oBءO?+ymqoM|kAxcBvQGj7v7vZvZޑs%?ښn/̈́ozm͑;%xg> ÍoP~ WǿⷰᏁ56WMKú tZ Zsũ[ekgE$WSi:oO ĺn_ܿ-~"'d֖Ci pZ 7{_~sp:~/ƞMw×jN4^+mJn( Aw~!? |E_Ut^2lu?hv*i͆xPGφ:g-![ {Fүu-Qڄ7P*HV==1x4moMﷇ<sI#t&x3Unc:VqḼWZw:ý/NMTiE XEEk|]o|G >|Mfr%״[7߉qxSEt)/O iz&1g[ĺ6_c)> xBO h>*񯊴xT7|9'? ~ xworן~:?N/<'|Lռ :'Οd3hfcv[+wƟ;\- ߈>_]=fᧄ~3N~ :|߅ڎe h7iKXE6\]yN?F.VE [W~%ߎedž9y|)/_P_Ěe^] ;eӮ BQ#{1W?I8?P4[8^o!r;O_;|Y7?k^=֯ CB?j]GMZ//Ӵ ?sybMiVo.ϣ|]6>M/῀o[._Gľ4tg.t ׺_ s\ۡ$k3J2ƩxwN'|Eo~~zW /Z]LJt_xR֡~1ݮy-iW`lh6>;ּ;+[kwgM\5tCo!yu/m5MGOO« &|v|C^cje ߄4GZ} ?|Eiq?kl%47'Xi? 6jI7$;<1c}Ϡ&jV*!.iw^+?@O_t#]|%w ]'[|(]~*h0x>(^ӼKK=@V'A &mi~['m,}W{vX KC$hͿ?sx/z߂ t'.u_Zi_7Mpv"cMg~1ӼA⋯_Z|_aZ~ωٷFv]R~$ĸ ުGT}k]Ў6@Oл/o3V|eKkI?ៈ, ;g6//gjK#Yі/xa'9 +=? w(YOYWuOC^iݵk36|8z;7t8>=x"ZNg< Z|G/_xo@x> 6]h$("RvI|;t:^$! ?3uÛ/뚭׎&\CԼz/j_ F4V&am^'́.mmBm t{ ^^:nt 8 ]}#< fc]KaGg'fN]{Ri?T㽟kq>ag/a>4I/t{{~ys&f+/||1jf|4NT'M?\v<~~$|\1 N%O Q'k?8\'$['Ga E~!|IU x׼Cy9uo[@WKGG-m6m]j~w ֙urK{x,4OïDGğn<+ּ 4i"1+XmkE-͝dM[AZ{y"B ( ( ( ( ($, woį~+1Gغc |K# _FP@P@P@P@P@YW}?s& NZ$KaxiUYZ)D~[iCui|zcb_k;v!{ ߈y:4!u'[{^Q7.~!ҿOw=?oSfsE֝|NN=CG, -d 귿'wOUTt|/lḘ̌yny5CO['ԼCwQ[[ZvK? .H,Mi#PizCYV:>{_/O+_oG;J 4c'6%k;6}n?Fu++e{vԬmԿfWu&?f?g-?mCYKOX6]nEu&5Ě]sx~eo5a{+v7~|kNдxz7<-=&_ 2:NJgm|COohPZ:!ՖPTԠ6y+ƅ x~':]k _Ӯ4Z^iVb{ i<ȬA+o~,CgnK wvsZEnj:"cs)-ۍ?Z4jIN#J-KVV61򰚇/\[IMC:^+K+h~ j{ǣ>.ӤӼg#jvڞy0yϥi9x׈5wj&OZeͅ7R+OOX ^{k76Α{v\_<-g}>.1ϫn7~¤vM/A5C)_jEշo#o2n]"m?mu|?NBA?hVam~>1τ<\x~7Z5v\G}7ŦO?x~%5ŗ85Ӿ9F Iaoϋ`Х_;X|/R/o0KX uxk_MyxRuGPu-Jh3^ aim:2#_%j><Ӵ YoX?ViV=V_;ec|5+sO[.miX ?I X'd{>sx*$P[|߻Ӕൿ׋K?|D}Z߈5 +[aC~ ]| nuHα~ڛß<-ik_?C්?b]i)*N!ܚ̶fsrmo5ci[yV1mK߉!'O_x?]5MKV| x/Zo4emBVe?n|Aċu "!im>.̗6_xWwz/C07$t#u^ݕVoSE?߉O%~㹾il&|KyrOZ-n4_s>4_+[i?x?|/ hşԗ:Nj'xz]+DOԼYVͽu+Ukg{=>j>S^( 5/:ƭs=g=Z;I/YB;wwWȣ]1>s%WXHϣ ( ( ( ( ( :Lw$pGl8P@P@P@P@P@P@P@P@P@P@P<9822P@ dxJw4m'#MPi,fд V+f{eYV)vGp#0U)Ɔ/QӒE cJnd=o8GnVrn*S ^zpsNQVZ[umt|vO/a~31od?._)%?!Gfcs@/[]?"7( v~@˾;Ч^0;A `UohywoQ31od9l .B_x?pf?; ́g? U?SK/yGBa?_*4<  Eo(Qÿ726 V%=AQú[[ ̫[=6ӵ*jok<  z@>188QëOf7WVt-:ݤ~U`'oeZmkbOywoQ31od9l ~;Ч?8z  8ui*n5¥[ʲxl]_MlC}<?ptx'8 8wKc3azkgֵ9Z}[v[MZMtv\Ay'y0O|!GbzeK+^e{%.Fڴ~;tO.FӁ<HWf1Kl2Wߕaqs`SUT#}mgmZ]tK(Byϡs9Wf:72/iN@)%?!Gfcs@/[v)% #=Qû}s1M[G[ ZaZץO?e6]kVK!?SK/yGBa?_*4<  Eo(Qÿ726 V|vO/a 8w3@_/<  }?#NxO!Gf6n-ܖKrIYv?6OywoRQÿ726 Ve?/Ox`yP|!?37NKMmm:]#n .B_x?pf?; ̂_/<I|%qxǂqf:31od8_ W?SK/yGBa?_*4<  Eo(Qÿ726 VGR 1C1Lzg b:31?. W?G_ GBH8 8ufc= \ .B_x?pf?; ́g? U?SK/yGBa?_*4<  Eo(Qÿ726 V|vO/a 8w3@_)%?!Gfcs@/[]?"7( v~@˾;Ч^0;A `UohywoQ31od9l .B_x?pf?; ́g? U?SK/yGBa?_*4<  Eo(Qÿ726 V|vO/a 8w3@_)%?!Gfcs@/[]?"7( v~@˾;Ч^0;A `UohywoQ31od9l .B_x?pf?; ́g? U?SK/yGBa?_*4<  Eo(Qÿ726 V惸cO?|xgᅾS:~⨯u *[9.~oCʳLu^c*L&&8ԯBTRJjqXXA[QR̶:7qxxl$)(M.I0j2x{;~Nm  ?P;wwWȣ]1>s%WXHϣ ( ( (-|s+i? u3þ3-j:y} B{ Kxq%Ro:J/pNtu"k_u[e]ƖUn?x[D[xu  `-5m.sqc,+G/"㏊ e۟m>aXhoo<q*e@V]OGğoƾ"w|=^e4{Qa oqemky|ym*CX6_1񅎟4 $UuYkF-BEӼ-zޫy&ڬ[Ml /rWw;ď&@|;ún[OxoO/5uO).L>(em̧O3돌:K>%hxv-CT>%zt}'\귺xWsSOi~#ЯwiqY/~8|.𧅼 /?Zg𾽻M) xhƚEAck\= m+]6:U]O| cxþBdfŸ2TYw˿i? 3|3|SȣLQ,e_@/>'1Dy91sN˿~ fSRuxe G"/?/dO~1ׯ~47i? 3q|$x~~R߇O표b0|MLc' 9v"}h? 3OK*V]wGfo))%E,}v]Wڵ#7e .}/?/d>dfŸ2TYw˿'l0_ٛ)p? eu}[}/?g1ϦG\]_EG@_ٛ _R˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d߇W_&]^5Hm*$Ɵp/IImk3]5D6H*y˵[IoGzz]G?i<|1Ց*> d7{\5%Ac @f3? MյK]>_|-i?:ƕa[4i:.I/\}'y~f18e ~z57?'ܱ{-1cqx>iJhۗNP弹yongksӘ( O|cxwKx<37/%w߅vN6ѭ5gE]w©ix7<`ӧqu߰J)4pNU+<4)¥ZtOaxjT)r~<> ޜS9{\C~URp)>#7{[Sy]km{Ş tiuxgG׵|5խAg=r1>$% (SWGVT}EIҧX֍Z*:~RXJ;MGSVu*QTgR!Nht8ŹWh_ mM*VltRCF7q5CƽkM[]vغu+:̾.Nqז#ᔗ58Tg(SpU*RTԅj4 9n.8ԥN6 U VХ*dJ>JRz QoVc>I|C"mkw?xY7I//i&<>ֽfkXA'8 Ga1P*4a!NLgJ uʅJqQe5bpҎ*>z)׆/FPsTaJkMN]}MK<#Oxce޵>-EFKXTԭb,VKi.mEX= bR :iJ:je(ta(δ)^Tc(J~KUB {)M9F1OɿVu_ƉD_.n(מ#}=KfVGjk{vZi⫞x|E|,Ib9(WO NaʜeUޝ%>']c.O~-ׁ<+mGB&K*a|FUm-gN .{hsYJ75184+FJ*I(8Υ*ըԥ^N4oe51n 6&:roBIp#RI5()STb+q>@= KoHB?GXCƛh$}7F}>fi5 OZz0㉫^KHB4*0JV P4uhi,c4xjXeVpG *%NHFw]a/U[AgV$ׯ&Ro&& X^,vc.cKJ8Ԏ3꓄UgY`%4u>uڝ5N2ҫY{28*ꌪùSB|V֣uj+;TSw.j}u7_|E4>5k-^`|+OxoQ#˹-C-4FiJZm2-ei>4NNS4k֧ 8,/|q_ i?aTe#nK,VOjwk>L>u ^kCBhQ^1IѫJ4CGV4;rNpozd RqB}#$ዜT+OABS긩ݨp?|aѣ >̗گMV7ifuk$EC ç<30/o)UpF#]>Z<)tQH)rau<aKujѡ+ԧyŸ7<9w&<m_^~o[ZEucu]7ğZ E-bZ֝X^eue=-sմrS!pΝL.7)rJ5Ul-l FjT\%jJQ?g8HXQc*Տ."|%ԡ*5PqnJR)BSU#jZ¯iٛlt_W,O[뺙h- gϳfؒy:0)W0ج 4h9ӠZt(燵pqt ꬣNWCY톩VҚOIO\,^|,mhmC\k81m-UuC)-<+Y-=67RLX`FkbhC]WcpӜ{ѧ_NyԛT QP=Jq_ZTSVU)ERakF1TܪΕiŒ# S2J'*<;xJ4j:gOt/٩sxNLM,-OT5=:%8SU8Yᣊmѣ8S>nJeYn)xx4 Nt U΅̹{RTkզª#a|H?,HЅXֿbtxXۙ~غɇ`cE_>Ҫ^*/ ߶Jź٪_W_XVqq{8E~M֡=7'ϵU\}?.K]<9hi3͗C״?Ox.u6lWQ$'DקUHMƵjR)ӫ 9+(RMJ2eR9BU"ҒRJS'nFR+s  ( ( ( ( ?]~ Zs^/;ؿ=,G9Gwo_#L7@~[;;+__C Q.Ÿ9Lj?+o~WхP@P@PʺU/F?/;+o-[E ۯ߃ϊ5O]Yͦ'^[(ƟckyC'?O}ofh߳ςrf]?TԵO?\J4-FtD>Y"aն$TNt-;[4Jg|;^)ϊ4 'VOVş/Z? h:SYg 맵I5+};OecҼkov+~^s᭏k=.?)Ak/Zм5W-ͼMO^%|}&->O|v4??Q~~4Z[k :xXKXsp--n^<Ě?|S~?<{_ֵzi,]-/nɿt-7׎ ǟ1^O >A5Ŗ]V7"!c.M#k |S^xFޡhaM5.si~%՞qqyi,`٧;#Ϧicg  G^(7|6tui{t^}ŭߑ}gqy$NU o^cɲ]zOU qtyiw5L#MI>.x/o~:Ï^Яo,n%~~ͦ o]H,뭟!.Z[Be{k_ x[ dx-_KԾ(G=GD|~]_L|m #Q.qo`.ƃl5m;X[-3|m/&_xS>KW>[xxŶ l4o |)FxO[՛DSxR>6-6b[ X.6ݞoVV/|Qi:֧gp~lt_G?ZtV;Yе/wğt#_ΡXeo]7Z ùemxb:%wq5}W{tv-gn³Nݼ~:6?|+<+[k#\|@i|>w+zaold'^Y=er:uOߋƗ3|iRu[}r[\+wM;dy:;^ .mb]z8y|Tylo~#ռ _ޭoTKtr)+Da٧}7|p?cvW(|xW.}nd`CFN1Zx:D~-h^ ew?:Ğ.hY<ڞk 9qo8ݾG^=5ǃ<Zi&=+GnY& GIi][QKmBL/u;ӯЄ̚_qg>?k/N-~"b ঵Ě'|3f!mkzf]h6~%-o+n~t |6 eceX~?t|-CGX|W[𵶻o^ťֵO IX9-a>|I>>|`޿OFf?Imc:7;"Vz-/o3Ͼk.m[_K/a[oai<h߆E/|+ rPm)+-[NeƛK(u,tuk{+;yl|֭<a{[΁i/|b|CO|$Ӵ+ռ7]FNc>|kik/A[K(|>XK6j_ 7jQ[^YZxZĶU߈˽^]:-ϵ↵+':ށ6?l1\Z6/ψUɢk:?H|mYxgUZÖz_>!O P7HWƿ$ +(`*ț7,WPS궰Qմ:gVlth]ioJaA 8baBRQVV#G*|ڳ‡k9ѣҧ JUhh>2kwğמ"kmoql|<҃g ƭ/aӡ }*8e*VRiB橆øT-I{Y%ʗ|]\6+'RTUյjq(T9Orƭne]^[_^z߂*k// xž7|[Z-g_B΍Vu CLmϝh䅆+H%o,ap& WkJ5(TN%ƅXTEN6RU%a3,.[ ^|5U:5%I*4 6(JP֚=4oqxoŷ,wWb{៉څrxmy5cA<ܹK:2}q|-:2UNҥ 4 $Z7F_*TTU$Zixl>a?k惣q_9sW̦MψxQs߀[:<Xx7Sy躝EW^i>+ۛM^K__GR\eg[:]70aXqYV& 0kIBU] ''ʦyee9-0TY ^JKݮ1TU7ϪPnF/gq |Ma௉_cCw>?KĺzE-ai=̐ZijO t x1Vk*tbx9ʜ#R)RM&M^R慭_qU=X:kS|ƥ❹-,w+ۛM^KO_GOq|4^2s_I>#T-w[A i^E]5,T|?#VӛZ{3ʳZuTFL T})N5)뚿i>F`)ndcN(ԌqSJNJ9Up^QiÖ7Y{wDj?ݦO~᧎ŬMMx_*k]ZIIaMR& m"K鯔*SvW\.ĿgxOdE ӫ;,)c¦YN|l-l&!~[:+.J*5{ĽQ--5u|U <.]B *UߋB_Qjx'}Mg*k^ھxMu*̫f\案a,4TenwC i'^/|Hky]]>K_|!'TЖ~6a4[H5xu:\BM&s4YJSX*PZ\U8e+NFa()Q8~ح:Zc)IMSҝJӯO-iՒ*P5*L/>*5#QlR޴=b8R :.-rU&R$*єRMїjkAw?|=h /Ǿ rz-_oN|9>j7t frb*U_S0X, J'^]*TprӔG]fx'aGp)PZrƧ~5+Ro4QRC|\ω#.|ExoſkKxSoBL=zxZTΜNj:U9cQӗQB^(Ν:eZJpr939.^uߕ{=l:|? n}wOzo?<h-cSp'ӥiK7qG9QTTrxom!֗j|\K?>X[I.wzg_?ho,~2Qo:]>Q*XܢGGw^mN*:4jeYu)}I¬)r5)xy{K-/d``m[ yt-/I?mnj%]K ͥ,fZ:,6W爫*RJ(ʏ^ϒ-Uuk(*0+թWT*QTCdFk8k§?9)r'['DŽU_xFx;pk4_C}3ڣWOu-#T4D/ziYi7oC?i10ji6It{X/zfa aKNYT3Lv&0NnRiWBJTSˇR5U' jpagJ{Ha(NJjѡ*JҍZ5)Uk<x_x+]IMoj5Ѽ/{K-3 ,-:o[x8,rpp>yZ)U Is}BUJy *%8Ǝ# *UX<%?mNY|,4_-(aKr|;l/wh~ [ ;qxFɪ/uڶo[h4 ON8*+qXmt_Ὤ3:cjb}ejTpYJoaAI'CB_yΧ!xF:q^9մs:3+H ج}ymYzdXxlkZt-6it:I-,!uN*F1YcVqR- +R*-ի)V2ؙԡeFVqۗ,`WBSb)Sa9B ( ( ( ( (u)ekZ=xI$}bo_?M!؏0Awoį~+1Gغc |K# _FP@P@P@Lc$`眜O sF{u< om80: z86 1׌6OӜO ='KӮu[?Mu]rZ"-KV+4mO[AkYۆX;C~X## `шX``zPOlzqNz91(xu1&? x[^.IPRxf\i/4QV{oQkoA+smwc>x6BHӾ)Lό E]wNJ>$Ŗ|uj ? ZGLWKqaO.x2^h k^ xÚ;W//:m^ik_h3x+Pu7Tҵ].-SB/A>#Ӆ7v7?=TR$R Fk[ -<~xc |AO_o5oxE/5|N=*t5usǣzumE6%2m`mb'?ĺ>X5zw/|SY,9 mxO fIyDlTw:盯jzƑv-ޣOA+sm/en_4RT־)r|G?vzCCMWmOM^%O Z][1^W:I[ swW_Sx~!ý?:M|,6~oE}j!ޣBKMkSH 6I[^[57/@ؤ? Fҭ 2Ovo>_18ok9 Cg_:?+?' ag?_va3gQx':FFG<é%<%j]累9Z}gdkֶi}E޷?_lKe3011;@ykf|۷`P31iV/fτ; B(^+ Zݭ.c#U.gֲ[?_lK8:N1upYjT$ﶩY;ku}]jwZiô;>qQ<1rW[yEKZW.g0Vu{;K(b1:`ÿqMks8&ҒZudkva3gn?=@:93sO=k}O1aK_[e\? ^ҮCuoPi|v6|&1|S: 0J*U,RW[sZѷecAΗ`]õg}?lMGcO@88sy_o6],:[k_KlZXYKUo/fτ1# x8sq y+}O1aZi[ )%Z? tߨô;>ct#xǧBA?+\S,»hմ+MZˢ9ʵX4_m_kô;>ct#xǧBA?+\S,»hմ+MZˢ9ʵX4_m_kô: n%cwofZ_]a}.I~CA?#w֭v k奃]I- _vdoda-Unmk{Z^Rwτ=0tWN@'k8fτū~'w8ޥW߃mô~;>zcAwo~1Ϝ3nwV~a_Vkt=%d0hv)##']@d> c_Ụn԰o[McA#Zu>unj/6|&}1bô~;0;@y Xmogvo>_18ok9 Cg_:?+?' ag???lKGE~<? p_63;G {`ÿ75N phv/|awP3 6|%?"cpsA8/[τ=0tWN@'k8ô~;0;@y Xmogvo>_18ok9 Cg_:?+?' ag???lKGE~<? p_63;G {`ÿ75N phv/|awP3 6|%?"cpsA8/[τ=0tWN@'k8ô~;0;@y Xmogvo>_18ok9 Cg_:?+?' ag???lKGE~<? p_63'I{S '<# xQOּU-:agǂ%xQ3LqViaWЧ*(BZRMK'k)({ywf<~ 8Eb)/x¿E~~}P@P@P@P@P@PہqrmV? /]E(w`~_0‹WQ? /]E(w`~_0‹WQ? /]E(w`~_0‹WQ? /]E(w`~_0‹WQ? /]E(w`~_0‹WQ:qp8H펜I-@#4>v9E<GL +tG~-ۥo1wEۦ}9x_}[ķW{i&^d:g!-/RNXW%CJV^?5o jVz|%]T$t7\ޡ.izi%ņ,Z_ r~(_ |i |?o}i&]$߄<'jZoҴORJӯ$HԯH뫋r߇!sS+B]'Uu>.CmiO xKu jizAoi\XkneNZ/%@՝I} }+ ž^]5xF_,D6VJ ,Ee2՗91u8!>:Ui}&k =}o lMjBeS m$G4 vkK⻸}ROm* KM?M\],zΫޕg ZQiiw/wڼZ툳;t_-FgF5/-/_pǦo->#_;>{F;XnR K[i>ֱtqh_+;y >`=[6[/1G Ql[/1G Ql[/1G Ql[/1G Ql[/1G Ql[/1G Ql[/1G Ql[/1G Ql[/1G Ql3파\aWmz׻i5{ֲVKp|Bşa#'w~%؋H|A_q8;S2_Ԋ0 ( ( ( ( (N1Ӑ08 u6]w](Ep .Qu]wPJ͵K+;-BMKv,JmFMY Z]\PJW 5=\˜cP3zQ[[ .Qu]w](Ep .QuS}B+m*K8K;BM{Pɴ}JɜK5"+D#i{yX@ ( ( ( (8xc#Tsl  t k6 cJᦫxkw_ ;uKagu&8hlnIg]j.kc_uX>q#_׾ k1cσ4/xt]6G}-PfIjƹyXM{e"B<DwΑ>𷀾+>?Ew^5!}:Y׼? ˿[MNz]؍&_TE'ems_k?>.;_~_~q_׎~wv:?eOO@~9xKӠx-u&*xrh+EXGf_]/{ .4;W-l> k@E>I~5oM/k~?>8 Ҵu g޹usTk{.l w-.xkRIx3RX=t _-K/c>7|4W@]V7ʷzʒXQ :]qSo<ǿƙ;(Ef9gN#>5>Ϫxhbð@ڌktiu4߇,Va!hWL>!j? 8Eb)/x¿E~~}P@P@P@P@P@PV?IwUZ͜ZR46zNY]Z]Caiu/syjNYE$3$qlWu} Ctߋ)qk #宛x~:,.4^ךIVPxAF[}wQ/no#9JH/q plbo:?|;ןo]+:mZuoͳa7"##zroWK=Jkŷ~խsGu];~0~/|/mqjuy5x4 t\[|Rľ"_~"fZ|=X6xh8EӮ9L"ڍ6DXO, :Mu^^_ @ xZ𖁮z1{ZYͽx[(ec7|y-VnnNjnҬuMCJQN]~/>)h36m+>Vq;:vUnA<;ɦZe4R7,Toi>ʟp_%փZ.\&cI@Omr_-/Ul3Z\hݵ_\,ZkG]~ _%-ַ:x_YF(ѧ=' gm>GVY]/XV}_E43k{757}a|,"Ԓz?ݯ~3;-wCZ<s]L8`q0-&־kn[-2k==|I;Oom":F]Q.;h^{> r#O)-NsDZ8z:v8F=`cӯG^@5-@?Lw01䎠v-Ѫ|u ߈:ğ~ .=euZxú]-g:jPg-4yP !ok_'xx#Yxzu kƋgfzny1u#BI"*|V4ItMJA6/y]ExZt iؾ u+fomut;MZѵNWxkZۯ اwL<;Mk>/x/|!k]|- _BD GsaqV mwIԟ#N)JF/q}[=xLu##zz2D}ZP@P@P@P@qVGxocV cۨcu{`;`_|6x:dch AǷCq x`q9?ך`b3W#x|x_BvmB^wk F? _y3+}Ěy4 akydvvs3z `tO8 道1MA:~{ǧ\@P@P@P@P@P@P@P@P@P@?&Ggo'w~%؋H|A_q8;S2_Ԋ0 ( ( ( ( ( s@'9Xa{rE)i y*:5|]gzK65?3q4m z5O,v1^M$0O(M#C(em[0Nj<KMscGծ4 MnZSigבT^f6|g\'|i#~(@E.ux{V~2DylguOjk7$0hү-c{WR+Vo}7/yK >*|_$/czey^+[wu]Q~&66 OZ^agq֙_mE(QwvFb៌|m|gOŋ[ׄo' >!Kߴ._<-k Q[-sJ4[MIZ5`e(^moՅktΝu|I|=+C5E[>u˝>^7إ< OtOEy)~[>zi WkQw"wc>&ٯc4 |Pu;+7OZ|75K(4 bI&ö:_\h׉ݔmez|ض>4'?.oߴ__FI߰}^|ܜ\{Q1c[?'l",8\c=+B$;-pW50A754o/C͡]x~^0@^1+RMgUkW7aMw+{֋wcLK 'n6}6vk7ŭwt[ y^0q_|iwA.Ҿx-a/_ZVj" h-5]:e ]-[~}~]o|,|mCcMSχk %ϡ|C^}2{E OK7ڦ|0h]j&wŮylvMg?xž%SmSƗ{?z Z2xkeݏ>ĺ= ']R=[WU58lMkzokt1=׌n  th9$u[sy[Ѽwr׼Ah~4ˏj~$۹MThiAZKM>'h7]?붶~#^?P[;Yta[.O4,_̗[>p_~b|+߄(77ĝwz8~Fi>a-kJ$|1r)uO?Mg^ x|E:Ok/%<nگ7T{ߊ|,ͫJU.d/YZd]98JYXۿo5]-ckC4='~+ G/_Ci u0x/Þl) <և2wv:F5ivvQN6KwJ<]sNx }9~ԞOOg 67~xjM9LS[Oorݜ^C.$ESܢޱӗ-<'/Þ3f}7 ͽI⿄|"-_ l./K]g: eHVN=(n[yox/\`g=1Ϧzh|7^Z2 ( ( ( ( >!O Po@P@P@P@P@P@P@P@P@P@P@P@P@Nm ;wwWȣ]1>s%WXHϣ ( ( ( ( ( .H{qQ)y/d?ڷ|?g|_ 1~{ kKVL>|1OĆ2^xO}aci[_mۭԑȏ*c7F۾C?ڷ=qA9&]_*eCɋ[>ק;A1t$'4`]?ڷǁ8'GP^šdBV?-1gBq1Oÿ뷐V?g;|H t>Xzk],9aZ?z<]`gNzy&뷓?ڸ gCcw'pyQ)tkuO@j0p|[n6 h>$S`_?GgOK#_|sGny [@_|G'$`(pW鮯E h89| ~’C.y3SGďVֶv Z[Fq5֩ kOsoxWOH4[v[2ۨhgyKKjֹ30 ( ( ( (#4>vP@P@P@P@P@P@P@P@P@P@P@P@P@?&Ggo'w~%؋H|A_q8;S2_Ԋ0 ( ( ( ( (OГ9$9Nr | ( ( (zoLq!( ( ( :~ӦI8``cA6Ӧ@ ( ( ( (8xc#T@P@P@P@P@P@P@P@P@P@P@P@P@p|Bşa?-b//!~?(CLaOc?R+€ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (8G$a 'W2x'~q@P@P@P@P@P@P@P@P@P@P@P@P@N!Nz$( !߉_""Wc"t?̗F( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (Vs Wij/nem4~P{&LH-,L`6_k7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmk7wmek Y _RZ[K I{gTs#4nYXqG|M'w~%؋H|A_q8;S2_Ԋ0 (Oބmdqnn8_oҀvǿxrF:qL@O`2=ې"0~O$1rF1߰@?O r18hNg68ӡ( Aq}zӌ`88=Ӡ8 ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( !߉_""Wc"t?̗F( (_/|!'g_DCg6ϋ'ީ9G'5:JFw^3K߉ty'ѭ4R(R;;%c{ψ.4OOHOgM/ j..{ <-W%WxTx YWt?>^&mm`Wm sj|-[o|E#~ԟ.O6G|uNyzq^#Ѿg"W:ŝDžt"JY]XQk|t28:CH?txtc|+_tx u~W<}_}?P7/[_h>0“ZiPkO#H#>+xI_? >0?[K/+o Eh6?w3LS]Qo _J![|_?V>b5_8k_G>~9u}/?o̺nރ˴ƚSrƏU~ [a?3Z )WX9Ҽq xjWvEo+JlOE7 #`h'E(΃o?O}_f<f@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@~[;;+__C Q.Ÿ9Lj?+o~WхP@y>+ǚzi2/vv7wv C᧋4 Kiz=%v뺴 )>RΡb:7^_];M7W3]vWamsw'욞}Yxk_ٚg_icaS;5 o:/i7?tnſ ,B ;6D4՟_n4 /]i{QmTqU67) C<k\*ɱT$(L#xSL[?v_E7} >4Swdiҿ5?>l M|#Ob4 @|#sĞ ]#253Glaljum& wX-B+iA[X<]>xk1ϋ=KYTiK >zgs%WXHϣ ( /?W4|JdS|Snq]?lSRJ.YwO? 3 q#7!ӯ]w˿]~d~̣1C<%r0{p % %~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .ifn8s8%, f3C9K3Qe %o OIfaL}߽N' .Yw?G%Q#?1~"983ӎ,eGfo))%E,#7˿]>ccf}Ϯ?.iwo>ccf}Ϯ?.h߀h? 3OK*V]?lSRJ.YwGfo))%E,}$7|I)@C˾eo!~dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwW? 3q)8#>ewO? 3 q#7!ӯ]we .h? 3OK*,e_8/?gq杗=G@_ٛ _ROėf 7 ~_xH;wpYw,/?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .i8/:)?we .h? 3OK*,e_8/?gq杗=O? 3 Hc1\V]?lSRJ.Yw?@/1S?!"]rG#hO~`7s"dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕] \hxJjSAsYHous $kh9>}]Vܿ\qA!@P@P@NJ@pw0d1 a@P@3|Ya6ys$9 Z|"Xf|s[X񭎕?/wρ#A㯅>ï:y%Dzg_io@_O'deѨߊ/:/>*|N7m |%/i ~6.UmVҼG`?6hqE`q]h~AaGğ>ghy/ | |I^]YᙾWnuټ?iV|;GQ[m`o3⟉?n|#4WB$п ᮇxs;ۍdMqyq}nѵm/QA\𽷏 W}/>uL5 |M7ǍC4~$@m|4kMSԢWk+3y7(?^»xVL,j}9_ ?F ZEy j oh[mbm7GoU=|Nik~)>)^/ ZOuπ<{o77{eiwĚG|Qe꒭hliʕ߈7-|R^M7eOj|QG'_ +iQ~2zZ-m-Ǧ{>9O:M"tմn=u[n4E8v0>~о1F;_*o/vCw#MfLoh/+/#\ԥu>ʖ [Ƕx{>9ʖ [ yF o[~6|?wg| {?tLd|qk{sjmO;XCW>FVO|6|{xAan&ַ-:4TcIkckM%noxg[)/Fz[?|i[_SV~"KKMJ~hW$S^k{iZޞ~11Ԁ`O|yOE #VP?mğU:A|%+OQūsU cm~[%ur tM$?Z~!IuۥDE%[|_^xU'^)<<G =t+i3_JN[>T=;º㯋>/Cὅ4xH־&xn4iIǃ$ҵ:>n;y3|Ok%ӯ7Z>)ߊO||3–]s:f Zx3]&Yjo]6*W_zwÏ~ GĶ,"aJz'4߉<x?QG|/i/Ԯ6a{؋&u[JVӮlV발l ~%?xP<+co<9Yxz  ~*~7xoPiw^i^_/Í Xdk߳Mom4v=-{l.xV|vo|<^_M4-K~,h=k AgXմM1m:]fjv?h_T𝯌x/Kgߊ_4WJ0G›7:晧MDKx54lkx[*H|Ap Ǖo|5|D?h.K]K|UVMOş>O:M"tմn=u[n4E8v9џuc]E-kViؾ6mqf/Kk/įmkkJּ;.]WTşS!fiiyt?{{k_~VogGmo"!wK/O |KWgĝOMamizû^>)t'!jW^$S}w\xQuso8CiP@P@P@P@P@P@P@P@P@P@P@P@P@ei{j{FB0;lcVB ( (?-b//!~?(CLaOc?R+€ (>f''U6L?CDÏ~Q]opEx/ƞ9mwėxUQRx٨KpXm|6~aKΟx~qu':i~'OukMמ[V>* mso]^D;K\#L6@}/MnmwQ,]O) kr_д}&Ѿ<3>?x$\cCѬ4#P,Ϧ^L~o{ᇁ|+㟈t 5'tuAtyt/ ^ifIkf'.d VKkP#y/^I7sIY$ӼA/TjڭOxCzu).V5=:{+v ~,ӣoɤk6,w W7ec|=?:VKkOXHAK5 l4iSKm+xþ7>+f4K}.M#]yd5{pob+u:W*߇?>7ҭ5kG.3#M퇊ezu˫w[]V(ݮ~E߂n{??>/xB{mWOx:ƕ5 z[ۍ^8綾Qgh]oa?K;:]G⶞=Vޚ}f% ŞsW@nW9$?aeզL,emi[kxž9#@O}ij?h>ƛwKa}JX?[kgosJA~? |EW#6>"_|+,|C <6%e֏ ?Ito^Oi36|9w~ [mt` G]<9T 5nn5xl㾖]FswX/x{⮁hI.𶋡Ez߆O_:4xBaK ~τ<}?VI.7%fkFx z_xz=ơg%ھ-ֿ=RY\gmm<]WƱj׳5˵|nVGa/oillt+.G6 ΃>;]-/_Yap<3j>?x$]YK SᏇClLJ4+FyY}KX/߃?Plm_Y xUF|W[[d<_W7Ʊ4^ Lmwx_w|Xi_jM')ýOIx_Z;IСia;`>CVbjWW.WzMς_\xr]MW4$"IU].`+]݊oFytY'hg+iw3=H? 2H}LD{[)tDM{ejV鱟~<5ow_[_ |/Z_|GC|*~*5asύIk;ѡt'ͅ 5w//+Ie>/^w\Y:?»K+@çi66Zbڍ66X?f"֞>_@}gƏs8Zө}YMSYo2>'ֆ~v:3HMiWZ+v{ |;%[ W]~~z|QE7s~|4^FiKA!xZҼ-oٮhoKtWQӡ;-Bsug;v}BAԶ:5Wy/55kjmnZ?@n﵎CH.[OioMJWCI^㋏Ki^+^t2K[JEw < OCh`oD>ֿ{Aty M%ҾåO,- ?C>xBVjRwUľ!n|Òhe׊G>'DrjVûx z_xz=`j^!&+[Z&Y5_Rx#}GRofkj[|X</|<4O-/ ubkdmWĞ PGu&Լ]A,^n+`Y.W ?p2xP@PNFN:d``? pp{g@q( ( ( ( ( ( ( ( ( ( ( ?> EM~MP@P@oį~+1Gغc |K# _FP@7ş8?eiL #G~+k5ҧwgwߏ^xs¿?狾,x[~/O v˯[យ}i'~*EoK^>|#?_< _ !m]mO.mx0~!&mѭ!.4?fW㿍> O˟3W B>$ lt xᮇs_w:qws}lm+PmXM(o0_3⟉?n<#Ǿ?Z><BҾ$п ᦇxs>;ۍdMqyq}nѵm/QMXVǮ^t+>9xEEo>&ƁwjW?Mhf5|&o;F)v[G3-1EQc޽xge>#Wῄ7Y7Xh* }-Avѭd>٤f|ş K/<]'|]-_k m&Pk3|OkNik~)z_|)kCE<k>핧5iEJ-WՍCC៉i?|o->ũ=w k.|aOmoٞ>*Kf-㲂hJ{kN²K _CC?{#~ "_X{l[?:L^_p.mDT Y#"xᏇ4K|[𥮣࿈~)&ωoតnjxrv}-_ؚbaX^/~#XO1a~|>!d>Y?433M[O_nY辰L]Q4Q/]/|QxN ~-Kঝ?HSsFY,3KDK5)tof87V ;x:~!:FD%> 'cGJBUHt\G#]WTşS OYZZ]&?\%ƛ-;²>~VogGmo"@.]WTşS OYZZ]&?\%ƛ-;²>~VogGmo"|]SOx ᯀlK?.o}2׋м/|lҭkeyu*Ŀ+MgO{=~#e|[?ڧƞ$G|7_x|]eů_'x_P٥[xU٫V:ϧ7{XG;h%p >4'< ๾ľ#'->-x= *ƶW_-h49A,-OmQ @2A7jx\ |e_]w閟]}Cfm[+˯V%?fZh>4_{c՗rlįmU:A|%+OQūsU cm~[%ur tM$?Z~!IuۥDE%[|_^xU'^)<<G =t+i3_JN[>T=;º㯋>/Aᾟ}DM~=5 K7Zwe|1& -ovΏ۱E6 ] k[/ĺ?am76^2ǽ*[wmy| Es$eg6$Mze;$˷,|E ƞ(u_gZO |"|"? >%!ТAm]JO>9P.u X%ʗMŏ^>!nk3_n;yK\%?e}#J:6WoZ_׼aXIoĦ/~٩4ȫ%}>3ӾBb-~\<5? oJaC/C"F:C>#*|c>4;_'"6򰚷ٵuO|Y=B %h3Lͼ2\i"Ӽ+.7Ik}kt}F݌v)Mr`|-u?[ WKvxk~ޕfY_ |:D:t(,|FFVO|6|{xAan&ַ-:4TcIkckK-7GoU=o_:q ~"M \ǶxkwV |I~Z(_V6WM~%urx/O3Ut߃=f{_P:M-ƾ q_*]; -'x[o<_wIѾ.ji<'<'Cvw}3mf`qxmsq2 _yhψ:_]1f"[W+B4_x_8[3C%ΗW65\^]}e ⎷ amwo^3O |9e/;+MZ5|LW#_|@u9R[K?h_T𝯌x/Kgߊ_4WJ0G›7:晧MDKx54lkx[*H|ApmU/~#XO1a~|>!d>Y?433M[O_nY辰L]Q4Q Ǘto~?g> oGiz6-E|^xwE WcuO M|MSOtk .$|)siq~$ O'sQIƱX!(xG|U5߃h-k_-sL牵h#ſ4?x_w[\~%xl3_ZUεXwl/#⟋?KǿI+ǷZKR}­l x3zW<g׀5_ yHk|G}gCAՑVImKsK_-b >$|~:k mSK>[x4n]%jz:en`8ԯ/j7R6Vw XxwD^>\k~>wIΡwq}_SGZƭekCm[\WQ6f4hV>ºU[GƟZw|_i{<3/75(|iǹ&>a{ᗄ~W\Z}^&^3/]j:z^qv{DZ+ |Q4ռ/\t+1'.,j8ܞn[)$?a%LlekZý 4Cm<|1uxBAMrj':jloIEs4<([osOw=uMXҠKw{qg7 6V+X> kAM~?gK{+2M|`l| "Vu  2P-4}>#o_2gχK?|3?B]WT:O>-R_x[G^w]KI&t2Tx#nh+3V:|07|>a/l4umfox/5b/Po1w'\C>ⷌ<;CsiD5ۛ MCWV+ao xĝC~+% \o M[%ך_ش%ֶqqM4</烼?Pl#Wf5I{sŞ)Xn\_ƦD_^ N X</烼?Pl#Wf5I{sŞ)Xn\_ƦD_^ Nmc~ |1c'*}CV՗PIU]GRm7Nyu\p7_{wGuko]M5G>|#x [V]A&k{mW[uJG:<]r|G M|%4߃_ t7:}Pյek淶uQԬty.Mӯ]c)=/ |@D/]|07o|1%e֑g:c}6{Y^Vw|9w~ [mt` G]<9T 5nn5xl㾖]FswX/x{⮁hI.𶋡Ez߆O_:4xBaK ~τ<}?VI.7%fkFx z_xz=ơgGo.oujz$I]ŵCg Xj7:n5%ZpM[yX.࿄ >ܶiP<'▫iw7wo+׮<5aLLQj2h:Kޥi6fw WIxT尛JWğ?]3K|?|L^q FMfdQA^.I6V /<a5׈N'9N#Ⱦ‹˻:~0|^΄~#_ώu*jw6qK.~9лCob _ُ͖[x{_V.o>#L6@}/MnmwQ,]O __gogy.6㏋tsT@ԼG|E_?LT=#N6֍n7s=ݑpsW_~?^>|4Dž<^F4*Wo ׏k>ó^hivVP隞C4vZռ)-_SA⧈7~.4_ iυn<1j-yMDok%E;A}-kX?f"֞>_@}O4|h$k_m־g9+/nk?ڀL=G955mjox/5cQᏇ🋵Tյ߷>aޟn>!QX.㏶> kok!i}-& %axKm>BO,o]N>gkv: | k <׏ o w\'Su'?Koi ;kMݯ=/7y9xϭ"v~ 4/D?<|_Ᏽhѩ>*3I Sۭ}UsW_2ݿ(>|4Dž<^F4*Wo ׏k>ó^hivVP隞C4vZռ64>Û/PooP ZnysXGhVM@>gWGOH/|39es--k]Ə.MlM0Zj(n{$ڗ]%_, 5r RP@P@P@P@P@P@P@P@P@P@PVz"&J?kZb ( ( wwWȣ]1>s%WXHϣ ( cT2i|#?iSB3;ǯ|E_ <9_^u<-?G' ;K?e-LӴ|xO+.<2ڷK63|O|I<ySw}l|O7m |%/i ~6.UmVҼG`?6hqoEkvƟ|X|Q'τ~+xáiho6:W<C|9fo;eeoh_6&V׷~AaOğgw ái_h_VzOCּ9 Ʋ&oZhڶަ F+c|/mio:Oj π~A< }"D7|M@5M+}&4K_xvM3A7 ;i-?Oz ^<}}ZM3 ,,o4[jz]Wi~>𖠻?k CKl3>tCW>FVO|6|{xAan&ַ-:4TcIkcSIuۥchʾ'? K_n4|QSs/_5xo xn4W%+! Ŀ7O ?a|Jwq{{>?'7kjQ^XvAn4k]˧aY%3W~ kVE|Jе3Þ&? }ms=MiW:ׇt;b˾qc<=9?W,k+e ;U!bh so$"a5dB"xᏇ4K|[𥮣࿈~)&ωoតnjxrv}-_ؚbaX^/~#XO1a~|>!d>Y?433M[O_nY辰L]Q4Q/]/|QxN ~-Kঝ?HSsFY,3KDK5)tof87V ;x:~!:FD%> 'cGJBUHt\G#]WTşS OYZZ]&?\%ƛ-;²>~VogGmo"@.]WTşS OYZZ]&?\%ƛ-;²>~VogGmo"|]SOx ᯀlK?.o}2׋м/|lҭkeyu*Ŀ+MgO{=~#e|[?ڧƞ$G|7_x|]eů_'x_P٥[xU٫V:ϧ7{XG;h%p >4'< ๾ľ#'->-x= *ƶW_-h49A,-OmQ @2A7jx\ |e_]w閟]}Cfm[+˯V%?fZh>4_{c՗rlįmU:A|%+OQūsU cm~[%ur tM$?Z~!IuۥDE%[|_^xU'^)<<G =t+i3_JN[>T=;º㯋>/Aᾟ}DM~=5 K7Zwe|1& -ovΏ۱E6 ] {7:~~.5O xO~Ҿ,k6/gHQ񅏈NJϦ7H6mnǘ1ĩGF_4?ҿ%>g?&Vx5Xh?xĿ)K߶jMfm|1ĩGF_4?ҿ%>g?&Vx5Xh?xĿ)K߶jMfm %}>0cS韲čw>hiKWC}gMO+7/fk Ѭ$7S}zlԚ JJ|>gWU7hw ǷF36qNO<%Yv1Ȧ"Z/>7>(H~#&W$t?~ G$<½~ Of Ox%7ש{IԬ;EY+m+|y@3< 2_^1]ww#֮M5u$LFӤ]dv4<+:Yjz 4KKKg#[ xd|ExV]'oJm,UMr`|!gtRKWIğWW'>xž <0OxJ[yCzIB5 %nyⷌ|g*@7/ƝYIGS Oπ~&R↋xL|y5+Ok$?s-_T[诫|k&?~:9<[|'ᇋSz*:]P߳=|UG&yc_[eѯww.d<-/l|;m5 |;;6m30[8w6ﹸr)XYD2U𧀵o>(گ{?DŽ4? u=n{j }H-S'~K׾6|~)ᯊz+_ cUѯ5/x/fe|/#|E\iwk!= kWt5 ~ǚ|<-;㖍'Jo#7:ceqxKP<7'P~&6>t5okt/ږy kR~%A6oMsMY]/#⟋?KǿI+ǷZKR}­l x3zW<g׀5_ yHk|G}gCAՑVImKsK_-b >$|~:k mSK>[xs%WXHϣ ( cT2ip?>KFwï⋨COx^6߮k V[Fn#KJfu-oa+>tM/_:~돉UƷ)twھ>)u7^{ljVZj6ڕͿuxuo,`xGgK}s@k/+Z4|i񎅧x'?>2xRƞ,{Bk).^oܗFX9w>xG>yůkE?^6^G_ͼWi }$Mu]Mp7Gu{M[MNWBxIf㋏ 6ž"LN"]^YkIvȮV;Z?A?6|A?kQ'|T$$/Fө}Z]9+.vY[3JHȾ‹˻:~0|^΄~#_ώu*jw6qK.~9лCob _ُ͖[x{_V.o>#L6@}/MnmwQ,]O __gogyKj|t+\%-<,u/xwſu=#ė{N,a,n䶏Hӌ i֭n7s;[kyg .|?qk~M7g)U[åm'<5:Oך47]:fuo9z[kxž9'@_}i?h>u--*yud2\ymKkr7utSxO:-5lM;~XAI>׫ߍRM]NMUmSSӬctٟ&ӭ<~~?׾(x\^|hө O .x5?Tڬ!{g1g7@5|_|}_iMͥ(e߃_kx_޹/|)Q5]|.cޟo~7g?%͆ڎX];E^!G]Wž-U:?Zf 1)9EU|L>!|wBW.$߈uw|V|3n|W:?h]FsapɨjzsvV=w < OCh`oD>ֿAt ĺPK<ĺ2.<p ƇOjڲ 5[jwV:j˨$omq.X]_NjR{[8^=≵o_Dkᎏ<}GOSM|ڮ:%iuˬx'۸;(^6K~!H4?x"_k_Z <qi_aҧ҅u?A~vkᎏ<}GOSM|ڮ:%iuˬx'۸;(^6%<{RW𮏨Z]5>xK-ŭׅh^$o~M7DY4 dxwCzoo'.4|=?|VKkKWGA4۽B[ +:T[?;~:Ps펿iͣ}Ux/i+߇!M>}D4'R,~ϦOk+nV?omΟs=߈6\sxOǓojI%խsx/Ky/G/AMRy\gom<[WƱ>j׳-ӽ|nV8_߅:6]xw>#SZW{q˫xgAߍu;ÖRK 4;lEZ¬Ű|{ݒ/___k:6Ft]Wǿ|GK_WK |=.wX7i`5+"o'p,u Km?躯~ ~ EΖ~x{]FoHjWEOk)Ꮑ |X7ѵ_AI>|>'/1W-#zF,4NI0im`z z_xz=ơgGo.oujz7/1hW-#zFjWEOi)q4"?t񦯹5+um ͞sWAn/xB{ojxWҠKw}>wK >gcc3ex_Vռ 2 -쯴>Ga/oillt+.G6{>|8Y[&ǭ^jk>8JuI O ixþ.>%~cƥcy%zN4OZm{wxg .|?qk~M7g)U[åm'<5:Oך47]:fuo; oKmcG>;]-3_Yar<3>?x \`|={?__ok!#K_g/n-'?@nV04ٟ&ӭ||~>(x`^|hѩ?>$/Fkڋ{T]^&&'? A oEytY&ڶg '闺#=HtXײGhezZM%u[|n4/ُdžmK5{H4/Oo76z.`ԏ|l\iq5hкhɧ[~, :5Aa-&Jox Jm:BKҮm4}>#os;[k~ [旨ih|k}k7Rֹx~4{}&_O xTc$N~w.G_PY&I&|I -mx]mK.ѯfRyK9( ( ( ( ( ( ( ( ( ( ( (2?)4zQ_/6jP@P@PCEE> 8Eb)/x¿E~~}P@Pd~0֩}3H9xJ߉~=x,:Q :ï.o ?k/ x;LMƿ|@/|3^/u> h?;FK|$+X_4⏉?.|#Ǿ>^>7 LC|)Ҽ7y3|?\+-x~GմBa4o~;~$]?EkOHG JB*|7wfoxWn55xzFմF5`j1[{oK|AЯ2xW_|k!oއqi_ I6Zòi hiOAm}G׵WziPm^xwKAg޿cyxS⼃K _FV_f'Ï>)[ ?-m| ?7X~Y|Eaeujf^ߡn6[`xG|U5߃h-k_-sL牵h#ſ4?x_w[\~%xl3_ZUεXwl~4|Ve_%ӯ7Z>(ߊ)^/ ZPuπ<{o77{eiwĚG|Qeꒋ}e9muo!Ρ㟈zxk/@|} nGRPR?&//6"B)l&PHP@%ω^!i 3.MOƖ#'j>  SVNښ/Rm|'&Mœ=Ero_>2|AY<7m{goA?]R'-h^.mr;/QZL/Ɨ{pm*Qw"|ecj<Ú%-RQ_?UTwg`ηG]5m<9}[]g M1uGFK/]OuS'sDwſ Z> j,|릭/kqu.(ՁǗÏ>(<'k?ZMSNtq$|)qi es|%DpkNvR|m{gCC?{#~ "_X{l[?:L^_p.mDRMY#𮫪xϋEe|?o|,--.Gotog.mMYt x?IK[+]6c^ k@𮫪xϋEe|?o|,--.Gotog.mMYt x?IK[+]6cM@.xOik^>6iV5ai_}CjE᧽? >[-`SOx ᯀlK?.o}2׋м/|lҭkeyu*Ŀ+MgO{=~#˸rokjx\ |e_]w閟]}Cfm[+˯V%?fZh>4_{c]Ö{XgߨyLu  O<"u [|xƟ6uO˿>O7~־Vk+>gq.fn~*Y ԾC|_'~(1oou6 -9^xP^ji-ltrNCC~"ԭ /_ǀL~*zïW{Qˣoxmúω/SEk]*I]WTşS OYZZ]&?\%ƛ-;²>~VogGmo"|.ۿW^= SE?Otk' _l5e|3CGBxcZόgoksDVFVV6c~Tg#]di_UY3|  +KTg#]di_UY3|  +K/Aᾟ}DM~=5 K7Zwe|1& -ovΏ۱E[|.x?Oz ^<}}ZM3 ,,o4[jz]Wi~>𖠻?k ͼK ǿe/7ō~7dWË=Gŝ;᷂nu[OW$v-|MYa!+xF[kśRx] dϞ%֟3xߎaj $FO_;g6,.nZE.OO[|<Ǎ/O}#ⷉ}/#B=;^+CWm此jW>)?A߇~&@ΈmXE╅Oi,/_ x V~?<_G_]xCJCS֠lĆ!rۭp>2|AY<7m{goA?]R'-h^.mr;/QZL/Ɨ{pmء}GC_>)~Լ{M?Ğ>{u|vq]xU̗-4wz;I}/ Ydt?%+N㦰4sշ4n]%jz:en`8ԯ/j7R6Vw XxwD^>\k~>wIΡwq}_SGZƭekCm[\WQ6f4hV>ºU[GƟZw|_i{<3/75(|iǹ&>a{ᗄ~W\Z}^&^3/]j:z^qv{DZ+ |Q4ռ/\t+1'.,j8ܞn[)$?a%LlekZý 4Cm<|1uxBAMrj':jloIEs4<_>B?d7{u:#?_IuIjNǓemwkOm )-χ<-a6 ?LW?4|iԅ|Td\ gVzě>gc>|>a/l4umfox/5b+xþ7>+f4K}.M#]yd5{pob+~!I4?x"_k_z D]yKaJh]ki l8[|މmcC^ x; 5 =;yu}fk{xTG|#x [V]A&k{mW[uJG:<]r|G M|%xS?$h|#x [V]A&k{mW[uJG:<]r|G M|} =qWG-.}KRGao xčC>+% o ͠Wmޡ-*y`m}ϝ(WU 9 kv_д}fѾƍ\ic>ˢ^_iW~^?go'a+ßQywwۿO9Оz_Þ9αA C^W;eo?hX9"*?> Ě hXo< go ~Gt/  t'K Nɷ5irVjֹg#_~gs+߇%E>@Ѭ4ie鷓m`zx___k:6Ft]Wǿ|GK_WK |=.wX7i`5+"o'p>CVbjWW.WzMς_\xr]MW4$"IU].`+]ݎ^ x; 5 =;yu}fk{xTGO@{>4cϊDkS?kfTeuY63J 4!O A?kT|U]rj..we9*0{s 6u#o+'fothV>P/6"zZx%0hڜ^3k\j=+떗ϸy~ο>\x~_fG cguF1wė^+ex[뉼Gɩ|Xφlu Lm\ 躯 x^6w:X~x-t;xoHj7EۋIH_:'[?u/QmruOnikvZۭ65#ghiÍIt[%m&7OO=/W4g|-{6:mǤnSha&g%jV鱟~<5ow_[o /Z|G_ >#7O76znj.!ԏld&k;ѠuѦN\-f?ZGk~[| 0?MA7 <w_}NaaW6> 9-K4[4>5֛^~\( (o?p???SkTˎ>AGVghy |I^]YᙾWuټ?iV|ڰQ[^a7gŇ?~.xG៏|ߌ|'$x#|I~Y>MZ|37<wȚݼ=ij^z5]𽷏᥾ W<}/>sL5 |M7ǍC4~$@-|4kMwS4g[s|b>E+{Xi4ς˨|G6n<; o_o?ߍ]jWBq^Au5٦;hWuuCu+_Kt}cOƏ5?╲o1yK`j^ ~&o~7~$io=vj:j=+k rh$Y3W-ES'~K׾6|~)ᯊz+_ cUѯ5/x/fe|/#|E\iwk"} '_:Ʃ |9Xx߅-uCOuY5O|O >o  {uV×۵{l`T{ jĸ>|DX>"xᏇ4K|[𥮣࿈~)&ωoតnjxrv}-_ؚbaXyz`8|cSv3U/Koߊ_4WIG›7`њ_W:'_GK1`|-,>!nǶx{>9n;yt 꺧,VZ}4H&x.4iIǃ$ҵ:>n;y rxjx\ |e_]w閟]}Cfm[+˯V%?fZh>4_{c. >4'< ๾ľ#'->-x= *ƶW_-h49A,-xOik^>6iV5ai_}CjE᧽? e9lz{m`g8R'TĞ(/k/LD/ 4o^]| /5|pGYkmew%xli F曫|Uw͟cB5-EJYZ{m`g8RD2xϊ|O/-w[h_>𪵽'|io^$X^ }|k+Ofwoo|뱡ş K/<]'|]-_k m&Pk/Aᾟ}DM~=5 K7Zwe|1& -ovΏ۱E6.|1ĩGF_4?ҿ%>g?&Vx5Xh?xĿ)K߶jMfm*_Ok2MBnj>3 --|-u/ hZ#ᅟ➡j>< 6򰬭{ZǧxWU7hw ǷF36qNO<%Yv1Ȧ!5ˡxϊ|O/-w[h_>𪵽'|io^$X^ }|k+Ofwf!ۮ?ڧƞ$G|7_x|]eů_'x_P٥[xU٫V:ϧ7{XG;h.˸rokoħl~ 4Ao}OZ .Km5+ s^/uY [M z +> HsR)4xօԼ_xt? ^u W2^|@Y?&mwڲ.-c |KWgĝOMamizëox^>)t'!jW^HMrLZwYGM""yVH ( ( ( ( ( ( ( ( ( ( ( ( (2?)4zQ_/6jP@P@PCEE> 8Eb)/x¿E~~}P@Pd~0֩}3Hq_ 4+ .~H]Ex7m=vuOxX_70CqR_jW5n+ XφЬ<;iz\|OҮ5|@NP/)NicVU^!ԭmKè|wc??_+_I]~*-EOt-;>/=F OF>4cYIrx俟p0cq? + .-~H\/z=_Z}dnmHeu "kBk>xCVǿjbjwW.j혓zM׃5_\xnO^-ftͦ[ZKEwafo!ٶ>O@{_Ə:j˨$omq.X]_NjR{[8^=≵/]Gu{M[zjWW.j7zMς_\xr]M_IV]2P-^X.Ҷ7=?:VKkOXHAK5 l4iSKm ~,ӣoɤk6,w W7ec|=?:VKkOXHAK5 l4iSKmj˨$omq.X]_NjR{[8^=≵o_D>ao xčC>+% o ͠Wmޡ-*y`m}ϝ(WSgi1otO>j˨$omq.X]_NjR{[8^=≵o]>.5 XOqj^(gZ]x[օMvNg)dtEA𝞍It8^ǡifzrcG < OCh`oD>ֿ{Aty M%ҾåO,- A~? |7ѵ_|A+,½}ctK#Ju+"m6`s?? /.>w!{:]kxs>94 a{ y64\nJZ>{BA]_Y$'x=cźqk}{82;ѷȖcu<-Sil<_ׇ|!{ ?}Al4'R,eyX9.u^^q_6˫3[xĚ=޹[Gx_֮/cSx"}CRկg[z6 X)߲hweΏgc}6Co+ XcOiT.5%Ztmˬ9,xn$x&ז v(iE~+iM5m]_|_wb_Y7> |qqt 6~#KN&]Zit{IBȵ{+m`JX>Û/Pu [4>5֛^+\ xTc 4K~ѧ,ο@rֱ/^yh Z_:$گf''U6L?CD|mcƶT.L`xWׇ| Ui[Eխ=d͡5\x|9%ZƇݪw|Is~=1A0ZgğM5ku}NYYn.o>jͫ ~|X|S'DŽ~+]G>0ZWğUỿ4x3x:{q/./֚67Q x[ڿ3?t_h |x.;SJ_gI LƸ/M;CNz h?p>7(?^»׵xVL,j}7ú_ ?F Z_%.Z5µ~G4ύؼ#? )<'k/6 Y7M?UҾ!$ίyi<|2>8uFM'[ibʒ\-Cq[{oCC?{#~ "_X{l[?:L^_p.mDRMY# /|JĿO7t}?Gmtj~4;Qm5D_Gҵ?zp^X|嵄k94oh~,+}") DoE>?|=WV𕯋|U~1ժך<m3E|_iskz"d|.4صoRc/kcT>,<]o–!:ÿ>'{7uz=iz=0]bi=5b\yz"|ecj<Ú%-RQ_?UTwg`ηG]5m<9}[]g M1uGFʖ [7 lc<=9?W,k+e ;U!bh so$"ju]S_|_+-OP}>fiiwp?{{kfiiwp?{{k4_{c]Ö{X<'TĞ(/k/LD/ 4o^]| /5|pGYkm[==F03cc|iOxs|5 |Gw OZ|Zu"zUl.XZx_P8Ei|cxigsV];?_4k w%|W>&C`m"YMM>*z1j"%[yX,{Z}=F03cc< >'?4/_|UZޓC>4ï|\,/]> yzYX;u[7Ⱦ[uCW>FVO|6|{xAan&ַ-:4TcIkckM%nr\Fmw@_~z<cV_h ~x@:]x+lЮ|Mi*+]:_-oRO_ 꺧,VZ}4H&x.4iIǃ$ҵ:>n;y&t0>Bb-~\<5? oJaC/C"F:C>#*|c>4;_'"6ڷٵc?e}#J:6WoZ_׼aXIoĦ/~٩4vJ|>a?e}#J:6WoZ_׼aXIoĦ/~٩4,|k41X|a',ٚৌ4{~#y&hB^7!k I%ծ|]WTşS OYZZ]&?\%ƛ-;²>~VogGmo"|kCq> K/?,,ּ_]7-(|8|cSv3U/Koߊ_4WIG›7`њ_W:'_GK1`|-,>!-~>P<7'P~&6>t5okt/ږy kR~%A6oMsMY.dw"|ecj<Ú%-RQ_?UTwg`ηG]5m<9}[]g M1uGSV/]O+'/M|lS_][V-W>V^j^_|:_}ͮGs2I%b EJ=mx]|]/yO?|yN/(Mk|?4ς'#⏄>} !=#w?zn-owmV!yÏ>)<'k/6 Y7M?UҾ!$ίyi<|2>8uFM'[ibʒ\|Ǖo|87 Kor/,WxMaSZxQW=ƟJTzo|eg =ֵd$6-|5|D?h.K]GSVMSş=O:=tմn=u[.4Eyzo/ƞ1>3|V/?G~xˢx/z]⏌E{gwÚ[rjZck>(ӡ 54=B}]O@j><"u [|xƟ6uO˿>O7~־Vk+>gq.fv3~о1N;_|)^ω i |a%ď7nu{3N>։îj2i:#KT<}> |? w!6Em_Zĭ \?|9m~o^;<:_߉^4 ֕sxwC!v]->e HsR)4xօԼ_xt? ^u W2^|@Y?&mw5dU[m oω:>TχV$7|SOCPԮ+=2䘴xZED8=!@P@P@P@P@P@P@P@P@P@P@P@Z}L=(c5i( ( ( !߉_""Wc"t?̗F( (o?p???SkTˎ>A8g/߆o_k׊.'Wߏ x]|s]_jQִyYZijV6%MCٟ/Z?h":x#O M Jxu yGr_ϸyo`v8eo_k׉ck|ExڎyW_\j76]2:5֡u5¿`iE~+iM5o15;5] I =&.<7'ۯ}3I:Otuyf-%"Zְek`h7MO' ~= }GƝG9Px\ NkvZ,ĻmgQl+l{"O_gogy.6㏋tsT@ԼG|E_?LT=#N6֍nXg(i?M O/ ~>'c>4jcϊkwTp5/_iE̷oJ5@Wg}~!I4?x"_k_z D]yKaJh]ki l8[|މmc~_{w@r7utSxO:-5lM;~XAI>׫ߍRM]NMUmSSӬctٟ&ӭ<~~?׾(x\^|hө O .x5?Tڬ!{g1g7@5|_|}_iMͥ(e߃_kx_޹/|)Q5]|.cޟo~7g?%͆ڎX];E^!G]Wž-U:?Zf 1)9EU|L>!|wBW.$߈uw|V|3n|W:?h]FsapɨjzsvV=w < OCh`oD>ֿAt ĺPK<ĺ2.<p ƇOjڲ 5[jwV:j˨$omq.X]_NjR{[8^=≵o_Dkᎏ<}GOSM|ڮ:%iuˬx'۸;(^6K~!H4?x"_k_Z <qi_aҧ҅u?A~vkᎏ' >r_ڲ 5[^&Xh];OiuܤQ<_{w Euͮw.+߇ڤ{KGԼQ~'ýĚ3I&;MM/hpCӒ4<=?|VKkKWGA4۽B[ +:T[?;~:Ps펿iͣ}Ux/i+߇!M>}D4'R,~ϦOk+nV?omΟs=߈6\sxOǓojI%խsx/Ky/G/AMRy\gom<[WƱ>j׳-ӽ|nV8_߅Z6xw~"SZW|}qừgBߍ57Ö2K mhvGCm%п |X/ѵ_|@;,½}O_ct+#K_i~F{=VgK5vcZ^higF|}~ Ol4 =1m4CFcGj7E^\ZHm`ru^^q_6 o6ouTG ĭs[[⏊0A ׊W LH׮l.Y5 ^\X|ݕoُ͖{_U&kȿ>&Oi}I|?[&NfmsPoMO.gkmc>|?a/l4}B=i [Z-vG{ |:[Fl5; Tߍw3$ux ?51g%/FkڂZ]9JZ'F~~=8 s9ZD-ҵR]£R|]S.嘗W:L |ss4˯x}/I.4edRVþmo"g~eqW|C |=&yh__~6?go.-d67+GBC:_Əs7$> \ wKlQ}g7%~%-gP?φ>𵇋?&3_*@Au'xvk.3StfQޖƇ߂sy:mGtZxMԯ]'|G/~6<}^v~!/J_gl_oZ|ukm+~!Y3hvn6ItV7jiŇ~.\G៏|}|''o:'Scxn? t=gÚf_SV[fiZjiEm{yߪwIq~=~1:'Ugn? 4=kÞ!k"kˋvizjb=w?_d 3/(4Cy74 TҾmDd1 4i1NӶ?m/kc'դ> .ڼ>o !EVy j om;$#3O|yOE #VP?mğ3i6]ˬٶa8o|8|EYq@]kgNCqW{iNᥗQ?؟WPk^Fm.m }"|ecj<Ú%-RQ_?UTwg`ηG]5m<9}[]g M1uGF'^P=^y߂5 >>G?dlh^)[(V@\ }!PI^ ~&o~7~$io=vj:j=+k rh$Y3W-ES'~K׾6|~)ᯊz+_ cUѯ5/x/fe|/#|E\iwk"} '_:Ʃ |9Xx߅-uCOuY5O|O >o  {uV×۵{l`T{ jĸ>|DX>"xᏇ4K|[𥮣࿈~)&ωoតnjxrv}-_ؚbaXyz`8|cSv3U/Koߊ_4WIG›7`њ_W:'_GK1`|-,>!nǶx{>9n;yt 꺧,VZ}4H&x.4iIǃ$ҵ:>n;y rr]ïG*oI׉>.x.hg<GZoGS=Y`}ĺ|뱡ş K/<]'|]-_k m&Pk/Aᾟ}DM~=5 K7Zwe|1& -ovΏ۱E[|.G|U5߃h-k_-sL牵h#ſ4?x_w[\~%xl3_ZUεXwn6[`|/|__Qÿ5th_ԴZ6g/[)(x?::uƩ/DuMcMI׬.S,]v:z^.?g_\amn ߴivZ2F5Oc)k/|Cɪ|;ixg[gۮݭǮKc&h Ǘk>.xV|vo|<^_M4-K~,h=k AgXմM1m:]f͵nÏ~ GĶ,"aJz'4߉<x?QG|/i/Ԯ/Aᾟ}DM~=5 K7Zwe|1& -ovΏ۱E6.v㏈}[ž/O|j/szŚ| ρk Y5dž~ 3jUڶw.fڷ]}/|SxN_ l<o/~|C_Iu#M^Lӏy&eu|% [|ʷyW1 u _Lnoysc?xNVgBmi*x_$އ4Gx[o<_wIѾ.ji<'<'Cvw}3mf`qxmsq2!r*_͏E:M[Z ᶥxGšԫ}mnnkzkn˸,7GoU=o_:q ~"M \ǶxkwV |I~Z(_V[|WM~>P<7'P~&6>t5okt/ږy kR~%A6oMsMY.do8|cSv3U/Koߊ_4WIG›7`њ_W:'_GK1`|-,>!M[/~#XO1a~|>!d>Y?433M[O_nY辰L]Q4Q Ǘ|yOE #VP?mğ[uCW>FVO|6|{xAan&ַ-:4TcIkckM%noWU7hw ǷF36qNO<%Yv1Ȧ kCȾ ~՚xJ/.< OVß\fxş|3mGMqៈ>3i6]ˬٶ`qcq> K/?,,ּ_]|DX>"xᏇ4K|[𥮣࿈~)&ωoតnjxrv}-_ؚbaXNn;y rroI.l/:">(A ~:17qc|>O;X^k? ѵZ+j"־%hZkG~h~޾䶹Jfִkú 8oCY|\79#4o x<9񨾛hZY'>Ѭ/t{d8αiVcjuܺ͘պ'-;?~,N__ |! >/?jj\do7t_o ]k~/[]_4> ~՚xJ/.< OVß\fxş|3mGMqៈ>3i6]ˬٶa8o|3/'㯍3űxGx<?|C^5 M3Z~"ilז>5vP[Ww~/ӰY% |mg:7MB?gǟqo L"]>| ;n"Dg_O_g'b0jx]>%ZK=jxgDٯ,|k;Kx젷5Ү^ӰY%+)o>#]_?uD|Q=?3uc;nOMŭͪ>O;XqB:|ez_>&R)W5_>|Q48?GoWZ'_ɤcX,[RC o8:~&@ΈmXE╅Oi,/_ x V~?<_G_]xCJCS֠lĆ [XO_>2F5Oc)k/|Cɪ|;ixg[gۮݭǮKc&hV/]:oVoYtO]/SQHolxsSߋnSQ KLmgt1A)<'k/6 Y7M?UҾ!$ίyi<|2>8uFM'[ibʒ\|Ǖo_G?~?x/Ú~IߴWn/[gx+j`Qi6o&ՑVImKsK_-b >$|~:k mSK>[xs%WXHϣ ( cT2ip?>KFwï⋨COx^6߮k V[Fn#KJfu-oa+>tM/_:~돉UƷ)twھ>)u7^{ljVZj6ڕͿuxuo,`xGgK}s@k/+Z4|i񎅧x'?>2xRƞ,{Bk).^oܗFX9w>xG>yůkE?^6^G_ͼWi }$Mu]Mp7Gu{M[MNWBxIf㋏ 6ž"LN"]^YkIvȮV;Z?A?6|A?kQ'|T$$/Fө}Z]9+.vY[3JHȼ//ᯃB5FO#6>"_>eqWC |<.{5i_N{XMWnV3?g_71Ҽ}/oW_/=1o_/]u\Yϧ&|,̾ Ҵ +QXkoj4-E?AK ᯁ |!y}C@[n}U[z^'|Oq_lXf|kO"{h#o oKv/h06Ezj:kc7ֿAt ĺPK<ĺ2.<p }A'#y/^I7sIY$ӼA/TjڭOxCzu).V5=:{+v ~,ӣoɤk6,w W7ec|=?:VKkOXHAK5 l4iSKmOn>hsɤ /;]-3_Yar<3>?x \`|={?__ok!#K_g/n-'ۿ3z[K6ahUt=,|S/ oWz|,4}#KG/e]fui3Y\K|cQhvOuށEk=_0jG>6i.4cyh]4Y9^^q_6˫3[xĚ=޹[Gx_֮/cSx"}CRկg[z6 X<<|,is Xxi|8n]%~FIv]oa/FMx}LC{[Tӭ>ktпf?C|;-/#L о|E?X0R>IqO{v3ͣB&~_|}_iMͥ(e߃_kx_޹/|)Q5]|.cޟo~7g_w|XicJJi>On>hsɤ /j׳-ӽ|p>s;xGZ׆!RUgz.5KƵ~nxQt-.4xc1j{nYd_/\|2>ƕ|2.|:.,ӓI>^x_ZOmt(tl[_KGBC:_Əs7$> \ wKlQ}g7%~%-gP[~, :5Aa-&Jox Jm:BKҮm4}>#o}<+xþ7>+f4K}.M#]yd5{poavV=;L=G9/֛QյKQ˿h ף>(ër_ Sźj]S>' ]]!ZoMg___k:k6F4mW|E}K~x7y]GkHҿJH>y=Vw|9w~ [mt` G]<9T 5nn5xl㾖]FswX/烼?Pl#Wf5I{sŞ)Xn\_ƦD_^ Nmwx_w|XicJJi>On>hsɤ /)WԮ|4Dž<^F4*Wo ׏k>ó^hivVP隞C4vZռ64 ,u }4 {K9t9Vecpa] +-mF~^]-}oCt'&t|siPxPץ᳎YuyVgcĎIﹶ=rHDl U&o>#|M6@Oiͧ_x7B;i7]Ůd} w/,i_iM')í}94#eenYXt&BK[{Qj,E 6=j]WTmoZ.iQxþ.$~cƩcw%zFm7ӭo`{mo#ۿzӎsѱA+bv# 6=jmWTmoZ.kAÞ/i>$vcƧcw%ZFm94CobuC{+fxڎV5#ė^v2NJt:Oe-uRma]4ٟ&u?k15?FoI5|@uoJK[%gA |0u_|@.lt>=//Zw5ޑHot/闷ۿg?%͆ڎZ];E^ G>S~.WU~ݯ5z|QC7gE`>47u}Y|EqcW7>}Lg쟰i2K[ZKE/em;;]-3_Yo1j>/#6WX{>)ϲ藷=#ZFWEe67+GBC:_Əs7$> \ wKlQ}g7%~%-gPٿ࿅izz'V&u+kNJf''U6L?CD|mcƶT.L`xWׇ| Ui[Eխ=d͡5\x|9%ZƇݪw|Is~=1A0ZgğM5ku}NYYn.o>jͫ ~|X|S'DŽ~+]G>0ZWğUỿ4x3x:{q/./֚67Q x[ڿ3?t_h |x.;SJ_gI LƸ/M;CNz h?p>7(?^»׵xVL,j}7ú_ ?F Z_%.Z5µ~G4ϙ< >'?4/_|UZޓC>4ï|\,/]> yzYX;u[7Ⱦ[u~ ~՚xJ/.< OVß\fxş|3mGMqៈ>3i6]ˬٶa8o|8|EYq@]kgNCqW{iNᥗQ?؟WPk^Fm.m }"|ecj<Ú%-RQ_?UTwg`ηG]5m<9}[]g M1uGF'^P=^y߂5 >>G?dlh^)[(V@\ }!PHPx3W~%y#?o ˥S񥿈?ڏi'>;ӄ~"Njԯ-&[_ ɣxCdo\[yOVO _&-/^৆)꺷|[|3ƭWFԼ i-uK\Ke$Kqů|{/~#XO1a~|>!d>Y?433M[O_nY辰L]Q4Q`/kcT>,<]o–!:ÿ>'{7uz=iz=0]bi=5`q뱃5O V/ſ~)|ӵ]'~0{ n~(k7exCFi~\ |Q.lTj![g{P=^y߂5 >>G?dlh^)[(V@\ }!VH|+:Yjz 4KKKg#[ xd|ExV]'oJm,W<+:Yjz 4KKKg#[ xd|ExV]'oJm,So5ˡxϊ|O/-w[h_>𪵽'|io^$X^ }|k+Ofwf!ۮƇ~'H//tI5w}jx᷌|[C: t7An!<ڧK[>#\i.tyW1 u _Lnoysc?xNVgBmi*x_$އ47 #qwo_ >)XYD2U𧀵o>(گ{?DŽ4? u=n{j }H-u oGjc7ƞo,|=?k7(4/Eԯl/΅zXdk߳Mom4vVi_js牼%oiϫxSύE.^BԼ}wρ>6a{؋&u[JVӮlV8oOOmQ @2IÏ~ GĶ,"aJz'4߉<x?QG|/i/Ԯx|3~n;ym\c3W~ kVE|Jе3Þ&? }ms=MiW:ׇt;beao}GC_*Ӽ ?u x@a}s 4 ~~(o_ h\ĻmFGm ai+[X9-c~|DX>"xᏇ4K|[𥮣࿈~)&ωoតnjxrv}-_ؚba'^ڳ\o [sG izxsQ}7K׾,е/]O| X^"ɮ<3qcVҭ4մu6պ6[w>.M+ ?4~&X_G~"x;UqR[Y.uAo?!ۭp>2|AY<7m{goA?]R'-h^.mr;/QZL/Ɨ{pm 꺧,VZ}4H&x.4iIǃ$ҵ:>n;yK\O>"|z>4  x>6?|?,x~[Ѭ/t{d8αiVcjuܺ͛jv-7~о1N;_|)^ω i |a%ď7nu{3N>։îj2i:#KTm*c_x+24 I5}3>*_͏E:M[Z ᶥxGšԫ}mnnkzkn˸8Yo ~c'Fi_~?_Mxᶗi_kϕ᷇}Pȅʿ?~>P<7'P~&6>t5okt/ږy kR~%A6oMsMY.p3ߍc?W! ~4ƖO<7J||7–4]s:f Zx3]&YjE}Ym[]_4>~~ ;º-C@ĝCW<5[\O^=ӤռE|оjZ}|)Jپ66f컃5O V/ſ~)|ӵ]'~0{ n~(k7exCFi~\ |Q.lT5o8[g{"|ecj<Ú%-RQ_?UTwg`ηG]5m<9}[]g M1uGF'^< >'?4/_|UZޓC>4ï|\,/]> yzYX;u[3omcC_?_x?ºO$Z5Y-MᅺxWZ߇ K OS%wQ4][ ]WTşS OYZZ]&?\%ƛ-;²>~VogGmo"|]"5Vk}[ž/O|j/szŚ| ρk Y5dž~ 3jUڶw.fڷ]}ǟ-ˏ_?uk?rMOw,? ,"Z5iwon6[_/kcT>,<]o–!:ÿ>'{7uz=iz=0]bi=5a8<+:Yjz 4KKKg#[ xd|ExV]'oJm,So5ˡ_ KIl^-Ogī_ww# B{o~!Vu[5反|ioF:Uߋ\v $?w~<[7@G> ;>=g&z> x|Xi鸵9[|k?h_T𝯌x/Kgߊ_4WJ0G›7:晧MDKx54lkx[*H|ApmUA߈7-|R^M7eOj|QG'_ +iQ~2zZ-k '_:Ʃ |9Xx߅-uCOuY5O|O >o  {uV×۵{l`T{ jq뱿xT[*t¾m? .ؼK~w>2iZoj~mx !iN(4ӧh wm=$r>]ïG*oI׉>.x.hg<GZoGS=Y`}ĺ>[uqB:|ez_>&R)W5_>|Q48?GoWZ'_ɤcX,[RC o8>V HsR)4xօԼ_xt? ^u W2^|@Y?&mwڲ*-c |KWgĝOMamizëox^>)t'!jW^HMrLZwYGM""yVH ( ( ( ( ( ( ( (+Os‘Glf1P@P@~[;;+__C Q.Ÿ9Lj?+o~WхP@|gNOmjq~48 %B߂ZE^!xBG_>4BӼSOi|@iACO=Ρ5/7K# ;߇ #B߂Z5~"|qxQׯ#u VKF+[GP&.XW #wt?m={ &u}櫡^ى*kT?Ծ׭}TeuMj-m|I^폈4-/ 6Ftm[|C}JSU{E4kF,N{X}6GnV  w/,i_hM'IíOJ>x_i^ç6 VzZh4kOZc `φ=Y^xxOy\i 1<+k_km nƎ{ |scjMͭ'e߃_^c'|)SS^Xꗧ!k: /x{ᇁ|+㟈t 5\xHAν%isKlY [|މmc9錐GqrutSxO:6.g4N=P7]VH'|$.{Et JK4jPjgwjl=g~ 4D?||~?'c>4B~)2wڟ']sL^k?ki/%~&-o#Tih06EkMե(e߃t_x;A |;<]Ǫn״]Sox"_P 3HZmGV֭lfG._/x|)?jnmS>(]]![PyO/+Q?txbNWo |.O_s> ~+ӓ_ {tob+~!I4?x"_kGS.&|.?u t,iS}.<p Ƈ|Glog}ɠ?қүNfmxsPoMOA |7GtmW'Lm]쭵faa}e4 IӗIx_ih1mj4-Eٗx¾9#@Ň~+ {ij'^GA4۽B[ ,Xi3K--<<Hc>;]-3_Yar<3>?x \`|={?__ok!#K_g/n-'ۿ3z[K6ahUt=,|S/ oWz|,4}#KG/e]fui3Y\K|cQhvOuށEz:¯[+=_0j_٤L9泱mMy,9^^q_6˫3[xĚ=޹[Gx_֮/cSx"}CRկg[z6 X<<|,is Xxi|8n]%~FIv]oa/FMx}LC{[Tӭ>ktпf?C|;-/#L о|E?X0R>IqO{v3ͣB&~_|}_iMխ(e߃_kxc޹/<'Uu]wXw k:j!]xz톹<2.|;'N]'GU{|67Nh1Yikoj4{Aal-w_g_k:k6.Gt]W|Glog}ɠ?қүNfmxsPoMO>gkmc99c9N#f?ZGk~[| 0?MA7 <w_}NaaW6> W3Zl ZuxAa}%.I>0u>úY_iw6>/[~, :5Aa-&Jox Jm:BKҮm4}>#os;[k?g(j#񯉇?x/Bپ"h^ ĝs[[⏊0|A ϊYG KHnl.Y5 ^\X|ݕN>|>a/,4}B=iWY,vG{)ϲw5#zF7M^Om1>E߂}mt` EAscK췗I{,;h]+3 z_xz<`j̰\zIyxWֶɬxjmo{839oCr]KWSI|_qGNM'HYy|hV7?IТiZ+Qn~_>B?d7{u:#?_IuIjNǓemwkOm m n3HZmGV֭lfG._/x|)?jnmS>(]]![P 7/烼?Plm_Y_,wxW.cXPԵkڍ@p >xij3MZ&uC|t+\$-<,u/xsu='ėwNlԬo$GӍFu[ >g tK_@}gƏ:ؼs .z5O]:־g9+)YF{si9 X>|?]xszOO4uY-%"Kk~ |7ѵ_|A+,½}ctK#Ju+"myX #fK!i }?ƝO~9W|Iޮ/k_iՖoJM@] [旨 ֵ :OmoOs75xƏuoK OZj2~:hӖV?AikX/|<4O-/ ubkdmWĞ PGu&Լ]A,^n+`Y.W? czӞi:znx1|\Q }cv3#zbӌ`88=Ӡ8 ( ( ( ( ( ( ( \_dG|LA@P@P@NJ@pw0d1 a@P@3|YaZ\v">xⶱ[ *_~&g|]oxG<+xDž'Di~'v/o%dž[V4iwωu/:'>*|N_mxᯄ%37?C^5x*-WC f.<>mcCn~;>$\?E{OO t-3O J~zχ5 :}sw7MVҵ ՄҊٿU?>,>)v? {c?#t-+O I߀~hzׇCq/#zyOK /g-}ZvI|?#Fg Ÿ>:|\>y 7:'~O ñx]}}kZΫkx mgo*Nu8 _tO|T[h @_ xKgo~5+|kោ> Ui[Eխ=d͡5\x|9%ZǧxN> \WOt OyayoC)?Y׭7.> վ6uo|[=SPm'74oh-=WP?vasr.v z}Vy$-_4⏉?.|#Ǿ>^>7 LC|)Ҽ7y3|?\+-x~GմBaϾ*Lωx$7>+xaX~|.ߍ45G"߂<#ӼEKω5{?΍jF{c4fW㿍> O˟3W B>$ lt xᮇs_w:qws}lm+PmXQ[^g7Y}᷆,>)v? {c?#t-+O I߀~hzׇ& r[swi?< 7ğ|I!/^|Htm6V5+4#V4KOu O4uMH 4u3PfwIK>٥~mq<.?~/fv&h>5_ Got |2yi;"t|*_fC5VxCM{|9R xƚ#|Iđx"]ďjFo5m#RII}Ϧ=9?W,k+e ;U!bh so$"ĵdOgįK4xGGK&K TO[)}+SwEMuo^[XLF&ɞ"l/> M6Z^O |Suo Z^ gZyx~4[/}6Η-&HK_6(;_>2F5Oc)k/|Cɪ|;ixg[gۮݭǮKc&hV%Ǘ '_:Ʃ |9Xx߅-uCOuY5O|O >o  {uV×۵{l`T{ jc j­?z_~&R)jO8`>Qo42>85J]'[َ;oa վCp=!Ρ㟈zxk/@|} nGRPR?&//6"B)l&WU7hw ǷF36qNO<%Yv1ȯo5ˠxWU7hw ǷF36qNO<%Yv1Ȧ kC'ǟ_Z@о|}?UkzO&IswG<):/?jj\do7t_o ]k~/3i6]ˬٶ`qc잞~11ԀdO~&@ΈmXE╅Oi,/_ x V~?<_G_]xCJCS֠lďo|kX|4g ? y-fj<gOCQ>!tcriŬ [X< `>*Ӽ ?u x@a}s 4 ~~(o_ h\ĻmFGm ai+;[uO|Y=B %h3Lͼ2\i"Ӽ+.7Ik}kt}F݌v*&tP K5߉vڌ`x<d5BV-nr[Ǧ5|D?h.K]GSVMSş=O:=tմn=u[.4EMXNg FE\xg:ƭZiir6mul|]?%9aV~+ |dxo4izϏ<5OUռ%kx៌~5j5O[Llï\w:^/"_ .-~9G_G?~?x/Ú~IߴWn/[gx+j`Qi6o&Րn>/kcT>,<]o–!:ÿ>'{7uz=iz=0]bi=?6/{Y_t __:ρuB6?,K mh,_O#/탨N./e~&|-I|6esk׿6x:kn-cCY|\79#4o x<9񨾛hZY'>Ѭ/t{d8αiVcjuܺ͘պ'-:|yOE #VP?mğ(A ~:17qc|vOuS'sDwſ Z> j,|릭/kqu.(ՄcaCHώt? ?u?/ke>ţj%~ƾy6q[:^~ыoݘ,ݒaw"ǟ^ t o:?~xo#/lezUݮwy=jX@RIύ4m:I.Y%~]"cS韲čw>hiKWC}gMO+7/fk Ѭ$7S}zlԚ Jm|1ĩGF_4?ҿ%>g?&Vx5Xh?xĿ)K߶jMfm %}>31}p &DEd|_^xV^+<='ø{7|4aDv)ңo['zoiS7oI{O G᷈,'b.(ɤYk7w 9oŷ)(6:SN)'ݴVH=7GoU=o_:q ~"M \ǶxkwV |I~Z(_V[|k&G|sW_9i}^--Pԭ}+5/ι&>)[}ev[]}m+|y@3< 2_^1]ww#֮M5u$LFӤKYzW̡׏|Co&'O]/ ozWś f|%B 7Т0V<9\+eOk&|_so|?4=OW )4/4^⮧lm-RSo+kX%/1_t}>?5Iok=z _%$]xW\{m#7ךe1ig7XG6 Ǖo ( ( ( ( ( ( (2?)4zQ_/6jP@P@PCEE> 8Eb)/x¿E~~}P@Pd~0֩}3Hq? 4+ .-~H\/xo-~u_xW3QM} + XЬ|;\NҮ5ANxP5+>*FiWZGo[\xWR6f|? jE_~4xCSOĉAiN/x5^ICsy/+ ;߇ #B߂Z5~"|qxQׯ#u VKF+[GP&.XW #ot/ڏ[&x+u]IM"z.<9&ēizGOشuI-&PekZûJXfo!ٶ>AOFO8&|5/]ٺ>g/%^%l(Ff='1V:>k6Sxŗ~5}/c|3]žn/F> k_6\uأ XwD\OҮ5BVxP/)FicUV#mmGC|?~ t/Cƛi Ᏽiө^+Gu?7ۇRzlO _۟J5*M=H:}k3xŗ~ ~;|A ;e3Ķ>("xcuMm{S~-kBk .dW&_\<'c?G>7?;:ׂ/a noO[\xĚg{o7v1TO^rVm,tO_tKB\յ]/E u]Fajʖ}E }yY"s`g_ .6T> Kvn}4$ɒOfMMyX9c~*P4>h۠_jVWZ歮Rz-pK0F[ TwTﮚ([|2 ~ _Ix7x-z5oW_\7EtK}޼muMp7 վ_=<=?#=6DMo;^7c>gs‚M+ץյc s~m|Bq}_m+x?1tg}.oicyn~נG.&ӑ%& ]v/\|<5}~U!Լc}rW4е29>omm-嵂(.q|: aq|:|v&5QL}[ǂ5^/ӮxKƠ|M o&&m'cxS?$ ,u }4 {K9t9Vecpa] +-mF cxS?$h=//Zw5ޑHot/闷m~=|-%NuIa>)Ti>xf>ڎ .}3RӴ쬮%;@ c1ᨴ;}'ú[xo@}"?^ GS[ᵍ͞ 5#4dMgc<4.,ilx/Ky/G/WeGK#,UzNjjX5675-Zpfvo?φ>𵇋?&3_*@Au'xvk.3StfQޖ鿳ÍgYgMo4hz^详["l,tIt7EM>]:퍾CM c1ᨴ;}'ú[xo@}"?^ GS[ᵍ͞ 5#4dMgc<4.,il.gug?%͆ڎZ];E^ G>S~.WU~ݯ5z|QC7g_w|XicJJi>On>hsɤ /(]]![P 0IqO{v3ͣB&~xz去4v֏iqg>N27J,n :~kEc`VX"_>eqWC |<.{5i_N{XMSo+wy:!|3?|wGO xuK o|;au}F:^gm]U)c6]L=G9/֛Qյ[Q˿h 㽋(ûr_ xOڪ۵FO<@Ws?uVAw>G\|#ȴ"?&Ԯ/\t;1/n,8uϥ'I.4edZWi[k5vcZ^higF|}~ Ol4 KqZ-emχ>ȳ[KZn۽ܷ7ުu` ߥy.?xƫN/.Ե|QkψU{*Wg x{ڑm'/cxsXJO6V#wt?lƚԮ0|]t;1/,tÏi^,^ &]VitkIvȥ쭵}ߡk1i-#wt?m={ 5+5]K=&z.<9.xszI~ä˫M.oi(YemUV_߅Z6xw~"SZW|}qừgBߍ57Ö2K mhvG m%W߇I hMa.éxŞ##Osiuqz57ោfMVMv6$^{S%/emo/^yh Z_:$گ( (o?p???SkTˎ>A?5_ğGoÐx_5o_υ;ƺ7|=$^;GzwWiy#چ{9Ѵ[Hԯo|RCF;yݪw|Is~=1A0ZgğM5ku}NYYn.o>jͫJ+kĺ&XV>l>MBğѼ= l]C׺ڥH1Zo-_4⏉?.|#Ǿ>^>7 LC|)Ҽ7y3|?\+-x~GմBa]F>%|L|uֽh=4X|-־񧁯n;-7 hg^j #VеO'Dl54m]v8xW|]<>ǀ;|@o*C}_NZ?[=[E5|vzWؼB<3_je4]W4ϝ&jƓixsZ_&:/G[Z?Zx4(|aï{ЧO?л$m:oVoYtO]/SQHolxsSߋnSQ KLmgt1Az2z''4 :t@@=31@mvRFO^P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@eiR/2h_mզ  ( ('w~%؋H|A_q8;S2_Ԋ0 (,0`MS.;Of uC'7=t7s%WXHϣ ( cT2i~lx }Sǿ jP~_4?\[m:V_^ߟ1.c&s5so|6{|9R[m_|C}wm:,>(r? {37β][74}[J/VV׷3KNϊ#ch>k/ x;LMƿ|@/|3^/u> h?;FKC]h~گ|AaGğ>ghy |I^]YᙾWuټ?iV|ڰQ[^g&a@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@޹aHɣҏ3V ( (?-b//!~?(CLaOc?R+€ (>f''U6L?CDeeOT5P{{}SZ5JG7Py5%ͥu'Ƿp4w_@2@3s@`u8<ӎ( ?@3t$ u'ۮ :{c#W"=Q׵O:_xKKw߄EWt]'KO_x@5Xu?u֖:dxT[*t¾m? .ؼK~w>2iZoj~mx !iN(4ӧh wm=Ւ=O?~,N__ |! >/?jj\do7t_o ]k~/ >EK_.3Ym׆l<Hϊ,}RQom- KIl^-Ogī_ww# B{o~!Vu[5反|ioF:Uߋ\v[O=9?W,k+e ;U!bh so$"`jc '_:Ʃ |9Xx߅-uCOuY5O|O >o  {uV×۵{l`T{ `qkXx]׌;e:m#{KNnt}o,_>#m>TC j­?z_~&R)jO8`>Qo42>85J]'[َ;oa վ@m`|5ϋ&_<]x>O>5t={ Rߋ>g5b,?|g5m*Lm[NYm[qc욒O&jƓixsZ_&:/G[Z?Zx4(|aï{ЧO?л$m߉#o[|0agƚo WžռOj4Wwң+{+8e-['!9m:'ufv}'#-C/1uCEJ_ѵ_TZDkPxwOZYk&6\=v!@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Z}L=(c5i( ( ( !߉_""Wc"t?̗F( (o?p???SkTˎ>@: q#hc䞘 j& )m)KxOjo)xKKc<c=ީw=@u=lc##L6@}/MnmwQ,]O __gogy.6㏋tsT@ԼG|E_?LT=#N6֍n7s={ᇁ)㟈t 5's.6o^jibҧZyC%Ǜ  oDCL=G9/֛Qյ[Q˿h 㽋(ûr_ xOڪ۵FO<@Ws?uV!]ş!|wBW.$߈uw|V|3n|W:?h]Fsapɨjzw:_XҾҚOSۏZ=Ŝri:G+@0MMYZs봉04 V/ZMCV;]-3_Yar<3>?x \`|={?__ok!#K_g/n-'7+w<FSڧ|_{Oz56sL [[[=;I/ZooZxvPUԵu( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (+Os‘Glf1P@P@~[;;+__C Q.Ÿ9Lj?+o~WхP@|gNOmjq~4>t xvU?m{ݪX\ܴ]o%lg^t+>9xEEo>&ƁwjW?Mhf5|&o;F)v6g {V_|bm|X~xO|8|oo [YӾx&U[mAkۯ./ⷄeY(BImυ -B_~ß:g_Phoǀ;|@o*C}_NZ?[=[E5|vzWؼB<3_jeZ.+ocFg_~1x/%5 Ei~=Wvkcm+I&,>7Ѵ$kڵvI_o3ȾTg#]di_UY3|  +Ksin;:]G⶞=Vޚ}f% ŞsW@nW9$?aeզL,emi[k? i|EZ-}|kIuq_w+h6 uqO:MKkRuJKcȴ"?&Ԯ/\t;1/n,8uϥ'I.4edZWi[k5vcZ^higF|}~ Ol4 s%WXHϣ ( cT2iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@޹aHɣҏ3V ( (?-b//!~?(CLaOc?R+€ (>f''U6L?CDP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@eiR/2h_mզ  ( ('w~%؋H|A_q8;S2_Ԋ0 (,0`MS.;Of@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Z}L=(c5i( ( ( !߉_""Wc"t?̗F( (o?p???SkTˎ>@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@PVz"&J?kZb ( ( wwWȣ]1>s%WXHϣ ( cT2iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@޹aHɣҏ3V ( (?-b//!~?(CLaOc?R+€ (>f''U6L?CDP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@eiR/2h_mզ  ( ('w~%؋H|A_q8;S2_Ԋ0 (,0`MS.;Of@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Z}L=(c5i( ( ( !߉_""Wc"t?̗F( (o?p???SkTˎ>@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@PVz"&J?kZb ( ( wwWȣ]1>s%WXHϣ ( cT2iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@޹aHɣҏ3V ( (?-b//!~?(CLaOc?R+€ (>f''U6L?CDP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@eiR/2h_mզ  ( ('w~%؋H|A_q8;S2_Ԋ0 (,0`MS.;Of@P@P@P@P@P@P@P@P@P@P@RP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Z}L=(c5i( ( ( !߉_""Wc"t?̗F( (o?p???SkTˎ>@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@PVz"&J?kZb ( ( wwWȣ]1>s%WXHϣ ( cT2iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@=b0qϦp9O7_T4{iwloV-+L/Z$fs/[^I ݤZC,trluP@P@oį~+1Gغc |K# _FP@sVqXs^ U'.SsGpYN^Q+AݺlRvMw=oBz?sG$+O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG 'lI?;O>7Lkt ]:xYwװIuE jבAɫ @e_ae: Yk6V7_rwWwqy-$5x,4ticnwy}u]ܨ}-G7+E2B ( (?-b//!~?(CLaOc?R+€ (cԌs>=`v_r~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲF܀?:4ѧe ΍;/~ti}N@v_rӲ1sۂzہǷh/ ==s^Am @P@P@PKZzRPK.A OEBPS/img/spgateway_browse_1.jpgJFIFLEAD Technologies Inc. V1.01  }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyzw!1AQaq"2B #3Rbr $4%&'()*56789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz ?W#<Eg@]>*T_VIfӴ*of-;J!=ԤK#VݎhνW;~th{Zv+쯲=R_k+͵mơ.O{?u#wvr+5/RQoWRtlG&_jnM4K q;N8>cd\UWCŇ;*Sь~qϮ{ j4]{F8,6kgN߷6t֚oj?Xɇ3NJio:?֚ojذkgN߷6t5?a?%&*S7:_Me0O򈿱`kgN߷6tU?a?%$*S?ꟷ6t.) 5,?Z_S?ꟷoZk/R?!/%;_UWڵ#rXh}1zQʇ`H(H\`Hh~ktDՒ4=|,WO41Qʃ %a|qǡ._9cO4~cޟ*+ZZ9R,4JBÏ~=(CVytzsr.H0yqdcR嵺"_jFx>T+l'Ge(An8/d'G`?1O}`H쿭|%!|aǿL~rE}X<:=99R$<821߁})r/d#<|b*A#{rIX_7`qsKX2#wʾpVyv_֎TA oߦ?J9P"լi ?im>mnȗڲA摞Ǿ1O [ F=9Pr}/089, };_p8E}X<;/G*_ 冉H_7q|_j4NzbT46w~J\@KY Hcʐr}H9`Vz?̄H>"լi/rD/8ҎT>HkG@=1G*BGFF;?.[[r%yg1SH9byvQ|bT_i+ =irByvNW_j8Q`h0@w`tGQqTjޜ<|k^,./kކ wʣwtd*܂4qY5HεϷM;hzO{o+T}5_zbVew 0zi!%kqqqyWW{[ӿ? gM5ox"j ,<[.mFtXԚH8h$T7,k, mj*V#fYaˮ?xim+R=mJٌ:mykr]ZC;Gy%7b}Ƚş^o©Gj6o~&;pYxZ0Ҽ=kX麝YPk;{mMtk;=^oo2Ry/0>|_>!3>Vs&W⎳hZŶmYZ Giψ} kSԵ=0ԵoKe#5zƯ|D 冯$zm}[D;_Y5h>Me̔%Dp >ŶE~h+ 6OxS▱XeWS;=& kMwkđcOKInӍ͍tG_sg:ֹ{_ ?4πuo CNtiէ5oQS}JG:IgvO"_?jN_v? ~Bxh_?<9KKL9<hah-{qm^o !8vW#Wß~&x*C+HZ|DH=g-u /mtKko+y[h?.StR7>|qO [7eݾFıhϣ{K|p)yV-![KcZx_ק]R[}:Rxa/o+y| ۻ?šh~$Lt+N? YA[j}NoO5u atvw !vkF:U [Sj E^%uk?=mA"J nO$o(6?ϥpmi䂋W% }ȩq-6A$ek勛K`]POMkt 77kyu vho"KgT]JW-Y&X>o՝_r,yGῷ|Sk mS1HcC!|۶L|i2L[cijuZɭ#Ab0[:i5mi>a#/RR譱-%).#b$1I~s\VEw3kX|IZXVOKOϤiCol3X6BsXJRprQ[3Сvj)=m}kOUc BFH%𾸶o6q\G$%{X>bq}=׆])Ӛ+IIE=mm59a}׷s\$LL7ކXX_Ʋ"xҽV-%yr&նԛaӏ[kY/\u8g9U}XztON{M5Ȼa0KtLJ ik{ zqhko1{?inAXO5(SkIy*[[a| zQ-}CoPXtMԵ u8g9U}XztON{M5Ȼa0KtLJ ik{ zqhko1{?inAXO5(SkIy*[[a| zQ-}CoPXtMԵ u8g9U}XztON{M5Ȼa0KtLJ ik{ zqhko1{?inAXO5(SkIy*[[a| zQ-}CoPXtMԵ u8g9U}XztON{M5Ȼa0KtLJ ik{ zqhko1{?inAXO5(SkIy*[[a| zQ-}CoPXtMԵ u8g9U}XztON{M5Ȼa0KtLJ ik{ zqhko1{?inAXO5(SkIy*[[a| zQ-}CoPXtMԵ u8g9U}XztON{M5Ȼa0KtLJ ik{ zqhko1{?inAXO5(SkIy*[[a| zQ-}CoPXtMԵ u8g9U}XztON{M5Ȼa0KtLJ ik{ zqhko1{?inAXO1zJ^ѮO ^zy\]K!T~'bqlGI#)r:ɽ᎞U~[]{$}kx7χ4ؚ^me#W}J(`حǧ&x?f5/{Nݥ[> Aц˕7?iρ~(xWJ%CaWN AEW[ڻGY#Y8ym䑺t[F͜-ߴ 욦zdeln<1V;n8K(+{Ytwcܾ xgRTč[ï~"*h+}GF𗆡SD\;?ƷZ!o{w >ְ[dyxGxIռ 8k"\Ú/_^1ӢnQŖ>>/N4 4k>0T5a%䗩~͟֋iQa> u=/|NZhϦ7:n \˧i~#_gݫi[uψq:G}#|_r{>'[͢xes>[AqZiρ\A١:1Ă_+ӥG~)| ν$)|y)վxd^>,|^-|Vto'bƨk}7hۙf9+>sg™/EjAۛ#g|F_|JN㫏 \k0MG_ //ZkIkؕz6x__|2/߈uo2<hNK?4MguMFt~h敶RVMc| TZ;obG|!7ߍV>5ugU>/ _&]@M>_ ZOZ5w+gcGcci/tŸm]g)Ӽgu'/OH'}[LqgC𵹳לּS{y+/싩~x=gH>^&:WíW ;QuwuHu{-:Dv^3]t[:WBiM.?4ok>$|X{g Ɵ jtZEkxR,ҵ}c^&u}kRdwR[csOs^¿޿~9|9CKP_Əj^iڷ 7t7Gi_KMh薚5Ym'ׂ/d4 C?<9;B4m+Ifsqwr ιiWwfw>8Ȣϰr8YW2Z>,1Er*_%դMq{j#76TO|T8 2k$ :gnG6B_6SVVUiU5ʲ4lIb@O[Z[jVu)=֗+$OE$N{f*Qqq-m]i(Y4izvV/t+<"x9 )dQmq^w(,bN?O֧PQ\w.4ѳC؊~{/CFaM^as}xwGEL<K>+"]р^;,*#hƤi8+uuvHbMWon-|nm_lkZZAwk,\Vhյb[Xܳ*C K QPoImZT%yTM/s]5ͬuݜ X&qt[y7{EFqUyhpwt+E.|]-`v6ҎUUtCk,{Zoa]=r`na_l~YvX t>r`kt9W nQʗM=?P@krְr7/Ӧ? WkX9c{֣.,WMsc˰rtP?.,WKX7(KʾcwOoOŽTlq_O҅,WKX7ݍv}~1PZְOot9bl[]+‡a]?ϥt9bZGG*]6Uw{z~r`m;:~(9bZm? ?‡X{K`(]6;}(K ? 9Rr.,{lE. 7co(]`_poq_L~8ְrǵ&G*]6XE`鰡OG*]6XoQʗM}ޟt9c`N )tXo|G*k*{:cv=a7.j9Rrt70/? ,,WMJ9Rrt{{ŽTl7t(Kwt(QKrt{ J9WkX9W+k tQʗM+Qe9bl(vQʗM+vTz_o)ѧxėʞNxCn\躃jQ(+ )_)U|n}5*`(^XiVV2 ebjM--tG߹qxdH2vWȧueo}][J!\؜`tI }&|̽,;͏zq3{.N:$X2~>_+CKM͋q2x8 =Oh58zK;< x d`̗WKnjocsc01Odbhz柧.cy>ioizi麖w_X]ZM43ΎXm= τXc~<57Zw/.u ϰiUf}BO*5̲6ZF$Au2U#r{'t.ha|r`7^9'p Nyd 䁜r2O\ta _ c'z{I5h,`gzۓO9.u $WrGaǶqs%ץL?F.4 >@U,n]Rm/Q&y4BMXݣuKrQNktCI}O^Oo![}K~8F!&2FƣW]䣦y|ҋ H<ױP)ΈJWRl`}FFpzQtץ gOp R~[A'o+A4o m5'TzƟD-݁.%DsѰQ[,qz){dpv^vvTo3jN\]R޴ގ_U4p33L8k|SWYk%Nf/3s( A2rzaREQ K$;jR7xr~)~o>:~z_Mh}SOҘa~Q7xc.?sMU0Oy/V?ht/Ctzcǿ&.)@oOhk_ٻ])cOtқ⚪ &5o?fvqTǦn:RZjzc[ٿJ3 OK~A&55oy/V?Xo}?tMU00Ə=?1EEOKOҟQ-0/kzqA ,TS[yR/A,W?cj1T.+VmŊ?aD?oε_EU0=C?l'3:B⊿ oڇ,W=?rHQ7x=_o)5UAY4?!1&_T?765Y4KO 3 OK|~Q_ kŏ5o7 |h|UQ/KGbCVl(EEOK|çΔ5m<`hC7S?765WZ_bC7S?7ohZj/j?XilCVl"U4?0oiURoǐ_ر_l'3=/ v֒⚿ ZbC4(GEOK~{MEU0R_ɡAOGfz_OΕ?MoXA [7xS:zPK[I~pCY,! ~"=,c cZMD]=C+O5o/HQ7|S߰?tqMKiʟMlo1TO(Zj_K?l#TͭMT_?b!Na/*z_OΔ֊+bo>omi5Ujر_f~#OK(|SU[dcMA q OK|zGW6XcAD_T?7o)5WZK,*_l Dz|S?7o)5Oh>/*z_>n?tqM[?K%O$/3w=,c'OM}UX/f%OLcӏMiUO_ر_տ3oOhkRX5n7S?0^Ujر_l(EEOK|wZO*[?51Ti cADGA=58>%'^.r+]wÃN̺?Tɴ{pd`z+q%c N_5}SXдhZֵ>׺%iq) _jHh VSszQoJ9Z KQ{QrHdC o,$QĥycDDgwrTNI%a" Ia%&Y#u aъ9HzG+$Q1ۧN۞k/ycGuHVgh7nwrjd9'r.Fj֙[w\%V%X]D3oxeI.f5FG~ u5hJR} VBվ*%kmiwq\Homcg1ɶTdRϸ%KneN}|NH`ʆVyX#+H9Z+onG۝k0xsH,ZPԭ rB?zJ-EKR͇luMu:icKX- s25ֱ{6ЬQ໏xZF/ ˡkl4SOkx`-.eo.{i!l:6_[z?eR>Pp u¿أ>$~ Ӽ)t(oE/p-S- q?@K[[OV巆4m.VAiyc^^QC\KJ2"֟ /?4\ojw>`ӴO:ѷ[v,+xe#mH o!mbwU/t*ږ| 橧}, =(%2F$cb@a;|}O&>3h]CRI-_s.< OY_Uշ5-`dii_jOnoi"oZIG !Psޫ6 Z:kZi>yj7?}O&Ifo $wmU,@'wE#CďHL3Ls؀x^G0= d=N2:8&lMB; H1ӡ[Q5n[ ͱ8 DL>n0yrA\Q蕃n[##$zGLY@Q13ܛl|#H9ӎrV -?|1fj7FxtVo4nMF$~e$O1wn,Z޿tJH4~_!g0y^K -<=ٖcGO\m+]PmizG Ƌmiz? ^onO$h`z@C,OaE/nA?YGo7av]H ?-Z $/K#?d: ,$;ۢ_$|Geoo".ַG0=2c Q.vnO$h`z? liZ޿tZy!?P/3FB=2˷OaEo7aJH??P/1KᷕG >Sv?qYI%I%}?MG ‹%+z蕼h`z8C,mNt_p;,ž.+H??YDaEG6˷et_rGt0=Yvikz|H?qY cQy A@M#qhm/tG0=vUm}?ps /3EkG0=cC~nO$h`z8C,OaIZ;+}EG,hoh`zG6˷t0%o {@eoaKm8|H?YvO7aZy wemiGL4h-eixI_ :3c[9Y)ua|/ m7^x$Yed6gCUEqq4vɮ?׎A OckQ V򽤏!/a 8/D~`@UkVވM#EsI=?.kz"TVյ +PCz|YEۅ{+Z@o,Uxt٢3O&dB/Scv.i:zyܮea-̫Yg2@6;3?fs|)n|=x5}~:φt=RK)aNg$joGYCyIkQӱqW?r h:n}(MrN-kB.& p|+-3ľf` ^ּKiچ EK=+C5?re`"#],]:VKo_M˨iEې4 OwvE\*PU е-mЭ}Ftox:f*iOxZ_ &,Uԯ*;_S .cw>zK~˿ϋM_>>xO⶷Oz_> xrV|Q[|9 xAռ=JVlaMRܻt*-c`_[?ٛdooh/|_i>O hog0\Y_Ik!Qm4G~__9Vw,ncޤ;|7n[/ۛ4 e—I,l^=]&W_4B jj6!tĠ5K;M}Aޯ>% V7Ś+ˑm/Vܵo+8?(qK?#]*OFE:.?GL~1I5Ω H\qjvPB4*K-rӛ {_Y'2Ȫ PM5OxN<7-S,g˹|sgm;?6_iöY֮h|U񏅼QXh uhu SNTӵ => mKz-l-na_|]PGƾ^&&qڏ B{][rZ[m'V'g¾"~&߈J{m-nD+Q_jvHaxc0x o:|DEwC-+FcWg,ignG +",Som9k~3e 2x^[{kF V^xWHoMty<4frm!&#ο5GƱ/>4hj-_ B\ץzKR+٭tCK8#Gc'"<)&W K^'ExK.nu=UѵKߵ̑Ma: 7F_/F[f'|__6%|Aе{K北4~Ɓ;Q| ͨ|Ie? K㸼+g. FWѢ:l:tZkQ][Y]"=\Ckn_>6k/lhz_?hY_{=OnKu?@Wiw~=]Wo_-ǃ}%u!:o'幆啞>gky~k^;${.}{_+{=W]f]oei\]IG^W/ug+im O 5 Jmo4}sYJt8mmO{K{yo4l&2I16}Kk|6>ʹ ]x\]vBӴ`qzEӭԷqqa=lp7osu2 ㏈~4KFm*X1#R6ڜV]YiȮ'-EoxwP<5H.n4p>(j/IKa ¾,I]y6jfa!7Z|<мKo h6_Yt-GϋKa wh,o}?ƚ T>oj~🃴ZtA[JGgᛉ"2K{=SC}*j2ym>uxB/UѴFE*.miSne 4hIU>/Ҽ]#G]~-"~<|Pмc9G})׮Y3}qQ^(NմݍiY CLDkV+1Us8P@P@P@P@P@P@P@P@+@k2y 9"hccKFV ^4u6vV= ~ NWB񮓬j&mh˧FURC"V7\09dVxGfF>/sc@6ɾ%j~xlxGҼ;f^ z@ҖKj]ȏ~wM9~ğ ~xkfƟ< ⯄v-n<ZjZo"]|]K7n*|JE3So 7xKx\On[M.^|C u6x~״Ռz:M6n~qww$Ј%|oi=xtϬ]|qwu Qkum:[6t.OxZ>M~ s]o],d-ST˛#N?K:/+-6\ [+K}_MFѼQySk__z+_Ѽ3.ajvKҠgߕ,= E<7pQ5owi77Z4G\6-&EwoksϩZ]\ZZ[V"H?ӢX!6:?I}"zg5 x,KF۾!Moim_Z,1TpΓ|6<]ŚLux\Oo?)6쭬5A161)O=;6 } xRa\^j <G|MyyIx]GnJXvkoi|6BL#4{Z|OkNŅ6\Zƍv:pYLm{gS)o+^9Ծux4j C\֣kzzwj/o 3, x(HMcbᧅ.|[h4OAIj:lTZei`fm?OmNBMo+"Ix=gM<]㋟ xXuoͫiz<]iai-ƽ^iJZniFimт!l_ozη_>'Kx-C]<:FuYc^75xu^$r̭+|cgQsfR|iZxkN[xZJXk3\B:.yn#6<Ai}SUQ-SY6Kuo;{4i[ծ;EXjfGBlMGL ;OVWm I&=O+jzSio%[b8f?ic`=^ |!qmqj߈uԼCmrk{{Z]w=}[[+ ;k[KXT0#K+Jl|#P(dz2=E Q@G#P(dz2=E Q@G#P(dz2=E Q@G#P(dz2=E Q@G j}mJO-܃oἹVڽ3epN j3( A9+hrM 11;N1mUؒ\}$(sK?)h38 :{*O&}?!9ӷOoG+ sJ?ʢΗ ?9qQ;3=ʢV4A A+3_bZy:kxD_\Nӷ'~Y+tjJ_ D?zNӷWnYi.tL_Ht`18-"Q}i+s-?OH4N?wps!;?[#T$@Zyt& GF]V7Hc)b1ؒN 999&ܿ>x.h_Lٸ 4qG$离/!?1V@Qs=?/c SC޿?1fIϩ!O_Co?CG$Laٿ7s ?;0?x@cs]Bcտ  ch ~rM˩!Mv&P[}oo9'>=5_D?vn1/;zQ??9z |!9'>=?/c SC޿?1N:J9&Laս18#shSCL;?~>2w!9'>=5_Co?CG$Laٿ7s&0P??9}vo (hA ?;7C4rO}Oz~_z !3z8 _zt9yL;01cƃӎSs]?ua/|rO'Iϩ!Mv ?;7C4rO}Oz~_z |!9'>=?/c SC޿?1fIϩ!O_Co?CG$Laٿ7s&1c|!t9'>=?/cVM^9z`ESC޿?1ߍL}NNIu5n=5_Co?CG$Laٿ7s&0P??9}vo (hA ?;7C4rO}Oz~_z |!9'>=?/c SC޿?1fIϩ!O_Co?CG$Laٿ7s&0P??9}vo (hA ?;7C4rO}Oz~_z |!9'>=?/c SC޿?1fIϩ!O_Co?CG$Laٿ7s&0P??9}vo (hA ?;7C4rO}Oz~_z |!9'>=?/c SC޿?1fIϩ!O_Co?CG$Laٿ7s&0P??9}vo (hA ?;7C4rO}Oz~_z |!9'>=?/c SC޿?1fIϩ!O_Co?CG$՗'o>V_74[ O՗2}M>oaheA{ SE/d-Y|'o>EO%A{|t>mJK+t0w}_0 W]#A‹/]Qw . waE?(G}_0 W]#A‹/]Qw . waE?(G}_0 W]#A‹/]Qw . waE?(G}_0 W]#A‹/]Qw . waE?(G}_0 W]#A‹/]Qw . waE?(Gc?:Wk.7aӃ끌`F)K ۾l:z`䎜u׌gKE~/A]+yv鎸mȡ_\/-mgW_{zvx#.֚&-/[۷N1\ p-:i ۭwaӧdcB9ۻj75k`=O'?=O%˺H8נ }O'i]Q⭧[rvD~?Z=徝%$]m =y,s#V/U?w{/7;zc@ROP:q3Evw㺻53{QF^a<ߺk;tv_[?l]Kqp@ <c&^Z?I Wu#Ѫۧ_k ۭ:~1w W^KK [)#A‹/]Qw . waE?(G}_0 W]#A‹/]Qw . waE??O xRz_0xoL( ( ( ( ( ( ( ( ( ( ( ( ( j? qk_m~j7$xF|eXMc3O=Ŷm( '5֕(Ԍo.nU(j6I]  RN)8M(FVٻS{߶2㾵M*Ѧ׼s}3Ѭ| ;|W+m<[(|oxQR__|B ZƟaX |9B'uX{ťWF80N.jv{ou҅UYr]ByԽZݮ} |a8x'^u_d Wn]6iK<ƹFZ7yܐbdUuєyG2i%''XRT12cIQ^ro4/#|%^ > <.<)S!Ծ'#JE#:m2d[ M͝Ao8ZSRYK*~.H9mI&ZNiΊ5ݥ% [Vz3Q7B|ke<Mԓ|x}VżQgmOj73n{b[Gԋڧݝ+ғMVimoȍ=]wW $r?'N/ BZJ1'Pbƾ? c~/G5{_mx_|G[77qi_kwdW,5g W-W9.y\j,0h(EE8'WGc#z}z%j?~%|`jڋ?_O#G/k^. tͦKzƷ}xPh6\wikC{+1xt)r]QKif> ւRj9%֚-[t={v|eߌ^ _(7h_4o7cJ7_o Eͩj`hHP1xNzqX׎є*ٻt[t18N%nVV|J-zům'~5?d~/2wDY,i'toG)>ge_P-RKmJ gM+bPI+PvSJkm"IS%RqMAiw'mW~za3 E + ⯇l^2N%xI76MԑȺWZuX'Wݥ+p\}f]{a((c#JIǚԚr>e}'_B|Bt|w/4υiRNF-oIljFҬݽܞIu;8`JZ'NI% Qwr! 2^N+ب4M[e=W +}ũݯ?x OrQ|Ag5+m*KMil|/M J+#=5cJa%V J\ɴ-Z.o_yXB7q䒎.mSucеۻM %k{?['[xk__VZEK6Qo[G-Ώ"I:>mN3v:/C8[ԫRCN ^*PWTKYwnxW𧊿jga_x@ӼkAwhR(5/ j:%]Α1o%m$&~jፄ2.QRPI==km"gp̣.T]+۫Y 5?Z4+[Y_oLּIe-N1YOb-kf8Wܘvɯ}].TVdƃ)VZs&nۣU-+/>Ww>'|+ ִVOCsk岷ZLK:oc IUJmF7m%w၅hxjt։k^xgQxwWlZ%ށ:3xmoAдO'67CGOo k\O{]'TQN *Z)uwKNjљSL,0MҽڴZoįڇ߳<es7֮W~*Ԭshi^xYC<6izmxiSTrGhQzFz tpVJ)&{Z{kKs8€ (<#^,Կ^/g~sI ^W#)P@P@P@P@P@P@P@P@P@P@P@P@P@^{9cX>Gߵ>&x/N,>i?!l>)*]ĺMgx[[ [=emFg^\^sW_Fauwi4xʕ`Ea7j[-%:sU/xiܒR|և-|wI~~?d{O?xUtzv? $՟|#a_ZI5 B <8!LCjUC ,"շ+n\ek.#Z SUQii>}K%ߊ)O΋þ9_v4Z(ռM[?'@<7Ap eSE{7*iBvvrU9UaK UH¤&*Jɻ?  tsXmr-yfxNX,ntۧ7iOxZ|zƏ}|58v'5 C7<)_ڷq5dkx+{ c?|!o-9}|\kaK˧k ]Z.:)ս'pWܖwI:Jt̥%<ߵޟ+/ xV_wVG|S?kl|'Ø[\kcg]SL6:ޙm&g=ЍJ'Nt%qӖܯf֩=Y^TŋR-R+l>_9C~/g5+- +_k$7λj~4_~( ڝj謺ү>ζ41c:./sRe;Ig+{hM.^xkum\UZ]}%9^/oWŏ8-ht+<'.V: nCG-抈 KzagG r^hF>З+厷3clVE$).O++.XCd_Y Y'Xo?c4Q}t o.J-)B8:ӿ,)?_*R2xtaksE->>׵+'?{-[ĞuSډmGZj7Vsq5[;W/j፷F%xF]䧍8rJQK])Go.~'?Y-Z&yqi:=xXG<'φmM-4;XAPNQa1rFN2WnJN^z*|N4jT*ѳm%hyۭo]mm'Eot_ٷ~dj@ ~Pgj?L|`oJ Kq NHU TԛqVz֧?d0V|M(/}=T.}j_~8xIuǏ}ox7xkZx;fK]3vWZޑcYsL̈́V:F%ݾ.i*ޅYV,]dRѫrݭ^ڭjQxuB:t 4nM;rg~ |T?nm<(|_ȼY6}6#y0УRY\(M[k8%V37U.V[~鮋W_ෂik2cMS6𾹮㿉_ o2~1GssX˷Ky-RXFJ7*jF2nSuQI$k{tΝ\qqN6m I'OUm-k|Ï= _ H^0Cy IY%J{K``>hRN| iW\ܭ)ZWNѳw/cuj:tqԡGݼ[㪾6="| ?Pyx@^m6  |Lt #Ě.|;luP3Hn+kKz5h+J% ;WfѭCEy^7{矋g"Ǎ/-ĭ?ⶹuOKPToH樶my.^ܭvSi׏g5N56+\4բC՝iUMӒjZ])}GUG?^/t/|EkqgϦO-_Ú3j</t(Nۛ9OK{dܴ+jft¯3S^\rMFs]I]kn[*(֕)Sekr9[qz;Sᆷ_:/E{'>9ok.,0ׇ^cғiƭ-i-KmMd4T[Sg_Zs ?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w?]?/EpQu]w{c? 4 q'dppN4^^붖Y*?1 /}-c(ONkۗm_t=8:t-6i:vzr=)H_~ޗpxH~M<[`WZ'k|;`z{QT6&㷻 c 1M^^MYֶ~4<𭧅"X^&z5x@^GB-#QX%ݕխմyKwcesg_K~'G]l׽kSzi7M=׺rם߲oïs?@5[KC>M|7FOH/|?vy{_̜6NO?M+Tr{H*k{=u{YX냍,9 ?J[rzmΝ2yub2\>zPZl }TONNJ/ntZ4L-zk`ۏ^g[$M$Gd;׻o[:1ۨ#۷Z]i[nh5o:`cӧ3{m{?o7}- N1rO9w6-zwdlt1מZ?鿨Y+{[ue_vt*F1vZiEsk Y%m4y81A9#99Zz5%AiYv߫a3l#88ak/uk.F18 F= 99w +|[-]=~_i >ljkx{Aix:#QVMnu j5h[HZfKխ9xb! s;یwӇK )ʜ#y%y%$V=^^w-i^дquuDM{47^E1K+#3m(>bR߂Ke')JOyIj֪͛H~_Z֢~j.~_ZZ{x'lj>)G2r>~Ia0.KW]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_( waE ./]Qw%A‹/ W]I_0K_ٗIE^0ԛSW:+g.·?^IP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@4OVJ3Ϸ?Js xX4|PK?הtP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P~K,Gڞz__nڇ{_L0pHws^Yдq6 ϡ^alPu]/XGԬ5KH5 [IM-SA/4=sMkgtQu>Nf[X\ʩ,"A=00:\(jzjZ~hz귶~iz}]M-$q 쪹 4zdwaq=zr88'>?=Oހ3m[KtK\=?Ftk [W[m;JҴ:KCQ ɣfy#HB:Fnt霌=F3>{s@kOA ˀzg9O_CJմwKuRXm#Xү-SJ-=6GNขI#7226mcCױ:P}9 : wˎrO|N^hZk}kMݾjVm톩YI$}ͤ K"KѳbO8stuxA{a{{{c<|A$qu=Ӧ G#࿈kohiWZςIxJ&g4K˘!bA"yT,m`{c>lP:~l{rA9䟔kO 8 2l<`d7<pyɷZA=:`G O8ȠF}j7ZzVkgyosqsiZn }Ώ~\,r]V/w8y^9ǧ8@Nyl:wz߭;`#p;t񟖀:z~~2Gtt8 ( ( ( ( ( ( ( ( ( ( ( ( (Ě|?6ψ'Etjyw|Z6qch!8+_> E">! g4 3Z&υ1_,e|5=SPWk:NJiqGi\idZADB~4xuvcoz>🇴 x -?էmFR+~6k?Ec-8k>5)hΟ=*c|lK_SKwgW5iA=уPh\IDvRSĿxOuWލ 5|_}~Y4: ~T;/%?.5~^ |#? 5k~ ?g_Od]o$4(jvo5eV~Dx? .|HU dOx^bHo#W<;|>hb g<7je`Γad+$п OuNݯ|$yo|h_'_[W?R㿈xz+-#WaѬ#ԥQBkhߴxY>)x\uU Jzw,?g/SV1{XJԴ)]j?("l-t@ֲ?]u> x .yql|mGKhMԞ-iַMkiKcivR:.x7Vo\|_ڞ0[}~96eOR뺷MMam_ e/5O 蚧0bմ~E֨]6oW/7ݻ_x٭t J;%mFEO-kb?lxsziN_&hDZmSUֵۍ>-cZ|^%o Ky i$=w4 ᯄxWnO3e~3_t_lx,H,/k+F¼ӿapޡoʝC~+ǖF7.K7I|a0Z~̶WuOz.V_Wĺ Ou!ǪjUuJ3o!(M|-ϊ|Qm-~ xᖃ}GDֻ]hŷT׼%k>&SuKeY{/ۭ6 rZ|/=V|_u}a;_5oj:2>K+k5=8Z=Ⱥ|D_7'_v3xz[Mw Ɨ um~/7Fhna'^XS.->Eou*t_k_˧M쳡5_o8xtZNm<|]5`x?mcLmTE[ե ^kCH_;.?j~_F<%mS:߀cKMt [xĚ+mzIZxeV/?o߅+|smlamx֕? M"/^ Rutm"2\됝FNma4zZs^I%Jjnk6Y~*QKu][èGi}wvRʷVqa{ żDrDP@P@P@P@P@P@P@P@P@P@P@P@ߤRF4u7SiVj:N4;i_]Z6:G<5Ȭֱ33x|K#'!:#Oo?%=;+loS]御xo@ӵ OI}c֗vk(񘶸I }P#(/!\F$}8<-_*kixx?:tx_ƞ'Ӵ>}zk[iχ4fpInm/si^&м%Ꮕ_&-þ𮉥K[mF4if b-5MV4^^wݳȮmߥ {OENJ>xľ5w#/=@>6_t鯴jx3<[$vbm :WJ/R6Yu +X[H5 /Vi, 3SH/9eky9]J&˪Y|֝aiZ~-VqetD)&Gy/ڤÿ ]i7Zǃ<o :\6iZ-v63=9vRT.c~촿hv~ {UΑmKu>>+4h΢wwA?@_g\:܍K jω~7Xx_6~%м[i džUEqr B[ 6eM?_K}w'O>=,#mx (m{45 sI=.ﯦ[u"y34Jm~O jJ6wkv7žB// <3EѴK ki&캏?hOͪ]sq#1Ÿd6?bsOCKӮayYPoϔ^@С/_gCoO^)|`t#/zQ:_5=*h4;V٬R66m!]>|/ +῀!>Ooi_u 923wVCRhtK=r]6]kNԴ?WݵK /Yo56P1A繂)'EEgm t/]- \l5XҵVtWKԭtKOH/IekixWУM O+敥iWwmރIhwWv\躬^˩S>6{\ֺ9Ts`5? S'm_ٿ^wuh?ڌt_uks<ǶWf/o Z\>^$GjNƽ/?ұམ^V44?Ε ~Otmz>[T-KScg&wi?A=͚C,H̗Ӊ@b>o7A^+:>ҼQ3CM4Gn>x]$z+h\y`.`h >xW׎<1>Ƣ?i1O+;*ϳZ x ʀ(¿ÿ,07?, >?i1O+;*ϳZ x ʀ(¿ÿ,07?, >?i1O+;*ϳZ x ʀ(¿ÿ,07?, >?i1O+;*ϳZ x ʀ(¿ÿ,07?, >?i1O+;*ϳZ x ʀ(¿ÿ,07?, >?i1O+;*ϳZ x ʀ(¿ÿ,07?, >?i1O+;*ϳZ x ʀ(¿ÿ,07?, >?i1O+;*ϳ5jsczQY+Y/|?i: i-ޏYjv]$q\Y$IpM 2L@WRO[|c =GC:gAqr(Ƞ" 1r(Ƞuu'0z`Fr1'4X:{c׀}ѷvgzҐ ( ( (<kYgc?&οgt0?{,( (= ={ gfE)|Askn}/B|kz_WUZGWkckqi6G>e#wC|,)^k>.V~YSQ"ΣVO\0 |@RM&[=Ijܯzx/^>᛭AⱿJ4o@%] Ti#47[]Aqve8-XNc|aX4j^4 [5x=3Z6u _Q6mzYY5Ya?Ikmĺ_5Y2k*/YTPZ7Cu+KmWN?K_%Z.+Ǻ|_kPXKŤxď\Yj~g.^j:LΓz|nڭ6H) ׍<1}h?<=}M`&n`$7imVvr><ெڧ/]k~'[EGĽRKLJ]ð5v񴺼F۰@Kx_Ě+6 >{xO> Sw6 xGҵ0ehan&|-Bo` @aЊ?Ar>[Ѡ-,<]sGCW-|7>wGkupk~fI[w%>VG]18Q?@E/ПС%zU:~y>Ki:Uyj2_1wiڬz𻹵K ?C@c1v/x_^3]h(-,-OtgY y<""&7VU{|.Ap?MBzK? x_]Cw_[_G>6wiQuZC];"X8ԭrqJZз f?h ?f&[Mصy}ݼELJ!Po?|,!zOl(9Eh߲_d|,Zo&_ğf_34dxV| J !Iu>D~gy3Q)C? 3?N๤?g߀őEo$<N1AsK^6| ew $k|0Ʃsx[F m2 d&hd{y JWJ:g߀Dpq$`Q9^/"sIuo3_"_W 0t$ !p~x0cyч<@#4E迲"6>MB|D~| =Kᇁ!mW῏|I\;K m>mkڌֲyGIbV'sIu:?|/C鷃L1|iwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgDo)hiwgh!mMti.|?xRRм/i5CAl&:n{qM;O{a-PO#Y39k~Q'klxeZk^Si:fA"еX/otgZJ-cn- u&kyS庐1QѮW᏿gWo$_Nӂ89}-`Ih᏿g2`c\Ҥ/n~%Kzt}zd /Oc/~- gG'12pFr|{Op/ x[ ~(..gAg2Ck5.t[CqXm/g?R?t, ?61ހ$߳K#?,`G_NО h>>ۥ&f/"|Klbig-xE?4]eytriʗ',76v|EipGkcvVV@2I,J4Hqc6&IFP@P@P@x'$?Mש%a?h׵^YP@P@c|K<5wNO-WK|Es-GW^Yxº&K9<l4+Z>*-w0nuM~o[/ϋ5huۏM,QռQ[Ꮗesl/?N}?i$60jb+Sjޅ5e] ᏏlOk=Xx{B΁Kmqm{o|KGMv+KB4YZyXG}"p:>2O~OF~<3~>]M&_ _g^' Ƶ}kߍOoW]Wx=Q~%}WK}o/]i'W'{(UGt7M[\E=^{[:N>!vez]oӷGV"R_Z?wžӠokO0iWaV-))N6Ba׉!hv ~;h.'o:GEkr;4)O Tor'oQiAh_~ |~~ѱڟƏ/~$wGtrFҵ C !!-havck/<+-O|Z-Ah1?gKA{c7ஃ ;zf_G^YxSKO]l<{[F{NjN1Jkσt0II|#eUwV˧CWCǦ7O~n5'Go⭗6k_:'5__|Ol"67fQM.ӽ!=Y tʿR:%qxo~꿳->b0Ѭ_}&i6qehhAuco_hu|>/o S#uŮjQ\_hnMN=gŲRIKJZ?/~ >Q;گzGį k=B㟅v.gѠ?iz6xKִ}GK:궖rȶ t}8n20#jw gx '.tM[-kx57S)xck"-G\t}SckF"䳻 6E%ПC㯃v.\[yh4x6U÷6^֗{mBk"M [Ҵk)ix]-.e |>Ҽ5 wпeO~9v |Wrg,|A1xV(d|{a/o\k_ƭڛ]4xJWXk>"KtҴ}'_g0m~"^|]4Vƚ~"k)G7~#}+KVfnuXk;lA&t}CZZxZĞ>>V:t:?*ףѭ++2[2zZwOiO |] >E1$|oxo ~#x>5`DRJV]vG}X᧎-Wmoޛ'+]𗁾[<9o4E4]ĿmũܗD4'(ֵ?_ʑ4mOC|7o x}5x h0;xKVz_MխHKG{ Zs%ׇ?Y=i6|LgF"ifmM;__4~|yK|5&i:߄Fú|:<şt3K{+O4(uքڧS7]%^D+)h|˧)Ӽ[}m"᾿|/xK:д/~,ιm|8Ӵ[x Z޾/t-oH,)|]78EQ/< J]_Kio^iv7:}ރ&h`R]c4w2&g?k⎩OڻI~&Ė4;ndNzM 5+sq[X{XF.C<^5῀Z?ŏѥѾ7Qh^2Z5ί z]²kk..=OдAiIs>JtA1RO V=I ?xYK<Ķ:΁c#|)^|M?⨿%O/4/\Y / EIP@P@P@P@ ?> EM";go,㎝.IgX|Q ? ÷ZG,%]NHԼ-eVn]&Z7|A'+ zGZ~z߇7Ii'/NKO|]_u'???o$pdxY>!Fc7u~(Q:.n"czfD5hˢzIyǯW_tSMύ>k3jך< iqs wDd;&a-9-/aPrWOeyş| 3c)׼QH ~ ;Ḓyyc,t((ܱQK{{cP#>=3uHeX] ( ( ( OٯIe~0ԛSS:+KM0k=p (`9o+ן9sF=Gn`c8crl~s߾3p1Ϩ#p>FO QoAMoDEhuჯǧkúŻRrbHފyO; tӯ-qpGrsP|@:p8^BrsAߚcus/x_IԾvdм]SU#[Y KM CHY͇v8tAl=cqa@m/Olqt?\y 19963?$c:z 6#˼_O㫍/\R]CJwK;nLK 1m>iZ],`oZ*$畭s+r~\/Z}|]kwƗ Io>$Ƒ}WCiQZS83s-5+uwE.=4{8O8x1[李q4F9JoA'4K}CvMWV?cO Yޡ BtͶD7 9icJoZ1P{Oj->b\yZ:4{,0(-3φ4I"Mi [YQ_ky!k[:~VmtzhZ҆=4kw[J/uwaz+?.s!ΟQiuw厈q?lw_[JJޟ\xQ^_j7OS-|/|aӤo?& gy3qq<-=?|O#xi`tV{g .[} ;t>wUsD Ӹ|}CD`xOo-ZU3G4%vi^b/5 >{y34 #31g!7+ccxy99ZM[;|C 4_vOu?4v7jVSxGĺ Տ<]sz^\iƣX7VkOOEx+zƻ?`-"}PZVޫڄmBUu:%1ݶ?amgY~>~Jdyd_G-#|<%%bI'a?ȴo? k3JB q cRf[MGCkNxį_]xWռ]u^jޓ貟7Et S@O'h4d2}M> S@O'h4d2}M> S@O'h4d2}M> S@O$w?LPt:{t)`( ( ( ( ( ( OٯIe~0ԛSS:+KM0k=p ( ( ( ( ( ( ( :~>~;$Shkx[R,cm}m. fJ.+2ȅ;CIXf_+]sT6?Ew a 2]9Xc̿W|=*V/.ʪ6?Ew a 2]9Xc̿W|=*V/.ʪ6?Ew a 2]9Xc̿W|=*V/.ʪ6?Ew a 2]9Xc̿W|=*V/.ʪ6?Ew a 2]9Xc̿W|=*V/.ʪ6?Ew a 2]9Xc̿W|=*V/.ʪ6?Ew a 2]9Xc̿W|=*V/.ʪ6?Ew a 2]9Xc̿W|=*V/.ʪ6?Ew a 2]9Xc̿W|=*V(KMp\JOɨxc#즽<#]fȢymdӼ;,q$W6і&۵#q?s$ uWҗ4:I}cXھZIzVe&ivrޣkO譠T̕A94]kl=C9럦99@ ( (<kYgc?&οgt0?{,( (vP@ |'xƾ+%gÖA~#xuë="Zᗍ)?X|//4x.GkMO%ՖcM.UߌO⾿/My6U4M_U񿆴}/~+KoXO Lj0$֝t:V`qI7u4|P<au OiVeƕso~0h|[}K> O6k:n{OK{yK[vmAT5Ѿ*x.Al>v/ï\|APjewQoئ"Zc^(<=^5^Acx~&]y5./|ED[iUmB[;k}PCĿ~,oO}M#7'^:/>->{{/^6(|ua\]֎l?w_x&("@a|D?tmkg[Vx/-+I?qk <+=ơ5)HPj=? -C'_h>=?ÿ~xC.gZ:_gច<M|3Ļ'D63]D -=+ǚU|(|ύ/OZx.>inYz߆vڗtoeB][ŖvXgkRע|xl~kO4=_kk |)#6jfdּE垲cim4P ǟ߈.?<> 񧃭|9Y֩qxFX5OA%ηv)2ᏄxK4xxP}o_:4O K&+ŗj2ZXXh[GomVmSPFJݬjr%_\(ռ)z[W?4jKa};VKy,ommu-+_Ndl7 sGO~}=CҵxH'Z,{KpxK~3>{^KhQxHR?A?O_O{iu|.RY~)ŨsUoz1XIu c,)Zۛ+/U+~$eo|aS{-w+@Z]<;IO&][/QKmGŸAxÿ_:־xC_uKENJk?\xUo|ۍ3Pi +yqeiet]AH|tkmo=j>+Կh v[D %kh(l>mk[i}9ZSԾDZii|?о fsǟ2|?4Zx->6"E񇊮/t?VY-4 H-fwJo;RΡ>m>{,r]MpNBM~OW4 @|].}Ğ ux;Ş1ǾL7o}"]Ѯe[}oOV.-|?4ox hWi~} i(#Ѵ#׿zΑS`Jwo~/\iX\wl{OJM%[-oOfHzOxĺ=ugih_Y]6N6՝| a߾M7Z >"^PW~/җJׅk7ڝx[ uix{T/y/%o_:v.:ۭF"[ X6ilu=oH&&v5sA_;šzY]рߴOO k-&,M⛩a4MO{jMMYo>.Eǃ= r^_nvD~4>Xxih(|?j-jhV[͠"5>|_]@i^&?2Þ*}ni&K Oëڷ=GHAYϩZj)luy_}jm>7v].ᮕZj?~ ~߯vZGxPi/x/kKRcYӍ񷕄Ռw+˛[iZ2_>~(oͥW<3j^xC PV^hVOBW- .ώ5i77> tjh!l%׼%J_*Z}k!wk\j]-Wtp7]SOh<;wxrox{xgJ_x\?--:/ x__uKDo=[[ccZZ&{Pҵmu 5ؿ~О>"_G5I1ÿ5I|Måvvc)u[u堖Kb\mӵm[> oi:|7񕏊<7 (R~(ѵ  ^(|AoX߂O-%S@RX~j>.^>Ox[XVa:<ny\Wڮoo2ޭtmm7 3_M  ïYjk]+U|?'@&ux,e67Wm\wZ}@w( @] P}€p.w( q-6y[KIti-x>PnS3ՅVk>]KEm>`Կi6Ý'VүnSJԾ,xfQuO,nhfdGH.iVMw敾땷ȯQo&7kCexçl:3(iA̠85BM<|?;~: +oP@P@P~K,cڝz__nև{_u囅P@{tӶ(ľ6xTޕi|0%E<\XiPjCxD^񵽥i6a&'KȼHŬic=Rv>Z_,S~xZOH=o#ǩ$<;b ~>W7gwl_mW+⎗]|?sᏊ|O Ikwrk<Kmo#lu2CDuMv} VK#⟋ 4iW~g߆Y/\.ş?@w5 _>,țZΕ=z8_|zy௅ |i௄qxg^ǁ?_5vxK-eWNi6o/c$Wrw%Et]N/Z-JU#Z'"G/I 4<,^EE-cM].kxuُ7 h.~,:: xV_$QuAv;ƺ,iT=-~.7^4g%y^HmMZk P;񟏾̞;3@ቾo6ԾjgX?V=xzo$kV]X|-Ot.߄^/?j:oټ/3>^}k'_/w_ӥXm嵮{o%Ɨqu:VsK-OYq[Ei75+E:^/;z-|v[t0> x/Tok'ýrOּ^ nn4St+2kKY亱kky 292c^Gw ;>(O >uiiY<-x^_!;2CQItЮN~N#/xLh4?|g ~J~Ok:ӏhO4ok^ԧƗkrӢ*J3⧉(Ҽ[FGs]=~|4{Yz:EE]HtdYm2V: YVegx?y 5jf= ӭơ>,mdu]o/'|DӾx_C/Sꯪx_A7xR >k__Yt0\&% Am~ 4 k-ԧEO~ҥ?~?usOsk֋2:xfJKS_Տ) @84ϋ>ijmqoϚ#=G\~/l~(=\An2╟D7ᶃ E*_T~0OD>_KWiO>JVNo 4﷧$;MmcϋZ oxGWgO~_ wٯ|ow>%_iU7GN𞣤ICG|_kv"X!'iu /2Mnja2kΥs7ws⯊mŘnxyHU(4oD'uȺCt>5_|INJe;]^,ּgi:^WÛ{kk~ :/Z5c(#Yfh("[t<__1xPO;|MSC-z_/~"^Mi_'yLim/|G%{3h:fۺ$;[u#~'xY&j6~-I{>#״?j/m~ ÓxrB >B'hQ Ilj BOib{o6Y ۮޟyeK~>Ej> j.}g..[ߴ©| x~kY٬Vj$OVYhLR_܍K)r?<;zt1 |n~3|tm [?|.m/ށuDgķ >-퇊oe֋o+O(ń֗="e]Wwğ |Mu;;k]?X> &ډWb뷧$Jџ*x;|C⿊*1W?N|)T~|:>7>5[_MI/E_n™:~o/vZ_An?f>{_ ~?>P~m/Y|_mo@Ѵ.]v:ķ%wv!]%C5+k&յh59 fqjABZ|SꯢxJ-4^&֮K Zw~ڶ#'x?Ou5uM>|5ßڦƹ?|%_mTe4]n!z[c>.x3UtGn~'񤯩<Lq_x䵵vEm%$>|Vk?N6WUN χ~?;] ѫMu#cƢŔvӊCNkG<8 5;I~s|\ƍ|^;| ?kc#LuxMԴ}ND-}4`E]y~C⏀}C__x^x\o9_Dǂa<ŹWg,a3'֝/HNѽ ?ﵩMwA֚ص[ 2emvOe?=G\W]Z|Az\7'KMj7vצEpGj6qҠ$J~wW<?jQg?z!|kxL֭)eukyhUNqÓ1^{oKwo>'];I?SBA/N[{?~._vV;FCi3]RkwF/HAF/H;_w 4@i(ӿQN$F/H;__:ЬխI#ž(LijZ-"jS#LmfiP*ҶE|Q$GuU/Y{[ӯi+6tk:Ry:;kImBPd; e4itxRY~[X>#]#]ѵKhvc{:B9#y+Y#¥sF ݌%-c1霎\uB ( (<kYgc?&οgt0?{,( (?NOr}{ott=1vx׸ `c :p&aۃzt6 A3=}IG^99?@qA G9o~}Gs2'9#Ӛ`ǧ^:{p0l$4l6utZVszAjtֲ\Kml ڭw3F$7 D>GxZ,R!xAqpp3G*] Dcc?G*]G`?O|K18{:QʗD-#K/~灞p8 9Rah`_Zrbק\ʗEcA?@1?h1t֎T%cDžcWN}(Km DuAɋ9ۑZv_ʴ_-Bah>1 LN8s׾sK.|C_-c$X#?!QʗE -?( 3#9}/?c80qK_(lz >8$㈽I<*9c1 '@}+A}9Sӓ9b]c Vr2@9q(] 0>C`:qG9Rm D8 Qyt9R?|hZ I9_oZ9WeG?A>1}>8G*] Dװt.O%@@4-tOq+;O3G*@}/fU} -_ ?@Z'9WeG?O/4rah_ih]?ʻ/?K}v_t D>O3G*Jǡ?tZy%$ok >=C xP,^+8x#4dKdE60sݴV} DG_kzE|[jeݶ6%eMŝ嶐^A }g_SFOiWWA76zm$y#䈖PhikmT\~迳2G$9#)rqFV8ERdGR< s}8i{bz|2x!rq@wG?$MO FA{ifn?gqƋ{a f|c'q{a tfn8<#1rOʶuw?@/r\M}uwscfcD8=qN]vuw>c _ٛ*og<8jtflC?.w]vuw>dfŸ25!#7=?>dfŸ24l߇)8> fg|=yMA#7=?>dfŸ24lSSF|?GO_٘v?02]?Gfa1SBxD36w#=c3|Sw>dà$@C;h}/??dilSSF|O@_ٛ 3>'=O"=3@{i1Ƌ33oC3yX=>cg|SG9;v._=?GO_ٛ ?.({=X>dtRB׸zlؿ0=|QHd%wx<uwcEp|MO#>|"@$c2V`0LD}Nzl/̧ω)sӃƁl|gy72GD9#瑐=0O? 3 O?w^p|?@/: OL`;c:flSSF{}S?Q=).l~ `')+ڿlp? 3{ML~gX=>c/̀gӱ#.^KlSSF=?>dfŸ24lSSF|O@_ٛ i? 3OO?Gfo))#@{}/??dhw̍ c$0?g?/z易/ǧ)c8{CMO? 3pIY|?Gfo))#@{}/??dhw̍#7=?>dfŸ24lSSF|O@_ٛ i? 3OO?Gfo))#@{}/??dhw̍#7=?>dfŸ24lSSF|O@_ٛ i? 3OO ٯIe~0ԛSS:+KM9?{,( (?JM%M|@5Ȩ[ X9M /~|ymv>)@t/ztz_=xV>&Z<8>N/;eG5# ?ŝ{Gfk45ya{tZ 4<;w77?V~/i?,)/S]g5Ov !jVѾ,Y=wK?\kB;~;;??<AX_  M?fj/^.|Yťs-kM;GĚ5{tAvGxt-?'x_!j>~M߆~&hpjV?u١m2M!ְ8ڇ 43_bkO1UѠx >_o+{]KTdKu Yt)%eO1Xȓ-GΞg_~ x7LK ]/ū{}GDŽ/VaRu-%yO _#i(N ~'ՓOπ4|>gE-[^-]R'':d~v i/~$>/_u5f6'<7gc-jgS𻾛[+%ϣ?!-OWh<|C c4<wk?qg;A%߱PHP?A@'įxǷ}Oğ|AlR(>=?i^oIo G%7O{e!2|AZ?4|9富M[Z.>6xS\~ g|1 ۟h>(;j.K-S<[YZ\;>2ů ekd\$>!`]¿xAW_ u&&Rx4^q/~fz5LjX忸Pixn)/C~о14}wU|m?kZ8KJ kk2xBkF+ϋoXjw5i?/K+3al#"1x [+HF x?Acgj ^j>5i>5+ cL-"Ʒm|5~ҿ{+Yra(-?jxk>:T |yo ]o ]a NGq[oW־FJGice=xrK>>RFqܐ3|N#~4׊ľ.-xOBFKIO d lO@E~;KW^ӵ_k$z8Դo-|!:W_宙m/TuOjR:N,{6ӵkha}Xk=-kM?Lqgz}2O G+|GI??e~:l~Tz.?ڿe=_.C~,i |#m|1gU}omWwl>^^.𧌠UеaRguo43a4r>(~"U+w@/k?ÚVokC&x@?u|7(ko mCee@PoeC/5]S? 蚕~Kiw|gZj2ZK5a?TR[EK-"mhοO1+h |Sⷉ iN4s^&E{?xP>!C lj :fmZ;YᵺkE ix>,ExS]gu:?O ~-YR]f\‘Ih(~,|ExWR~r:-ៅcb I>*VDŗٗ0RxZ2J9m%//i%O ?KKy=o~S|Ay_|U..,aHg$[NїZW4-JŸ%GDԴO:&k=VJv/εXK[ɣd!㏉_c/,}|ZM>Y2}\ψ?ξ5?-?H)+y4>_TO[hu~(foKIo%<5RH^m3'QO鋢xX<8&%I =-cuO|j|7jW>?a/0״ B+;Koi,s O4sK'ƙ,4];Phfo SJk÷%7RHwZo7,hw!./ c5_M}L'ƙ,4ڇ];P M i? ~xzD3iD+rx#QOcDwxu|OK M`VFu|/>SRN.+<7Wϊ @n6/^ /5_wڗ 'z^ Oc}{xž2Uеo2M:M&7R^D_P<7‹(ާ}Y4+LMx4_jxMռ%Cho ]ekj+N/~~' '!ޟƚsM1|sGKYxh -1~|R|]NmoS=mhO!g#CQ7?¶o'Wvol.V>%> hm > yǾo%$~%xs%#X0Š_}Die}|!:&G¼ju<=pAe;C.ko }"#Q@l|8tkU:_R&&i_kc G~XXx;ּgCT>,>^⿆u;FZR+C?ğg ⍿L<9?)J|I|I:wf?KdwM"_C^'5񯉬{D!xPIN:=tm"&l~}H۟o__߈$bھ(^7P4_KoGt|CgIqC6>V{Oլ_ j~8ޡK|9z4 ^;;9gko. xhc6Cោ Ğ1uQk_Z_kI}ax:0Xi-ci6!>ɤX[ٛ|9L _5]BOMsIЬlVo&K4hͩ]ܽvVf&?6>oN}''Ɵ_s//n/ >ɢL?xk˷hم쭵;` $ag_;|">x9־"o_|OE7sS/\xbCZipɣ<:uaiVKlm&Nye3⇏3M="c^=]E+>^ ~i^xVZRh|æ!+[cG?Ƈ¯%:7$? oY?m o)%&|.^N./5_YYuOoi ]&4o qIe hZ`Htr^A~ עӧ/Mx\`ֵSĚqA{[Bז.4 pf@𦅦 M& nV05/_ GK6^)JZKkSEK5ӵ++u A$|p765_]|? 5/_ uK6^)IMϨ%Ʃfn}xvF7nifo+KA6ҷ`Ծ |2c\+'U:O%Ʃfn}xvG&I vM4Iv/°0 .P*O=cB|W!n#q[|{+5/_ uGK6^)JZT#];RѢ]7PI4_=ͤu7s_/6ӷ| 7#2xH#Z]?E~ G>^krWž/T߯.޷I*8?<>߱}G?@|| j9־"h__|OG?wSѡ׺/{iiPKuaz]ڟCOb?|QeLjoYh :~h^-j]Zm6>'K:ڷMد%|I񞁠j^[_^o|YxZvi^-:ea][L^݂[ 3tI<]x꺭_į3]j;kS^ 5/*isiZtfv uk9{+ݭx/K]>woxƚޱouZY"Q4F?tx exWCDCDJG# /O%cL׿*>>>h oOTG#^h}k5_Kv3oni.]oMw*Ҽ_RxZ_WƷNj˽-E9VǵqMqpOssώu#Ğ"mgZx?-KXt+M;OM!ӭ]*,ţ7'kl`xc1 {k44-Þ4-L:?tYi|}6 ω3gskz]^ m&F_È<{y2'CFM/b4]gT}hw^urKo K6zv)kw4N~{$=89v<+amDx=|-|A'l<_+[K]g[^k.ǣ}K?xgecF6_#~>ռco{Pu(7>'͎Os_@ IŒk5X|]|{KP T7y?G_OQkN\8qӶ:R6ӱz\j>oN}|P?4i> φ!|AgO_b? 7cW3^V44f[$.ռz)U~k>%cƷ>?_O FBQa%b--.gb_'4}_5m?_])i]C%i5ο?9n4_x̸/s;X9/xKT6i 5?xx\jk!ySNb촛 -4?4-4X-a_C?BqA[P@P@P@P~K,cڝz__nև{_u囅P@@3~ӼiitPi>EGuy*?Ѿ!|]}>o>mA>=Ox⦇oOK5㿈7-+N3kkMrR7MElՓ(|zǂ~*|MiFx>q5_ 5%wk+߉G 1WY2o6C[m5eվ;;?_Z.tٳǚMN;?x|A%|\{cOZiS|4ţ_o>8n|EMih]ճPƲ{-(dz| -B2xHW|[~Fej GoV?u١m2M!z =C->0N5&iN4 o| ڟ+{c/|+js~nTk6ĕ IYic9$0pI,x8`e󟆾*YW >FDc6~%jQ?xK#|/Լ]mO[ͪhZmu.oYK7n)/C~"|hZwĞ wkNxMNJtoT>|</$>_u5f>'߇qjmx\'eF?hψ_™<9?+_6_'e¹˪6hST| YPb%H|s GOWmG nqWg >Fg?y16 ֊g}HӠW<[|=KO_O~ o jz|ƩmhZ4^/ xO7#rz懧nh~ G=Ěn_><񞫫KE|5 | <3xl4/MCėz%ՖZj-o,OЮEKÿ|UOׅ2ÿx.iV0]._I+ٺ5ӚA--9u 8ԼJlKI%kO~2xů jci\$~"xOG^?e?oPƺs]/,T:j^#6K/la#]Uaa|oZ5KJ|/ᮡ I9u>,x]Ua]֚5mUw6VfIz GE?^c$?-чW_7)Pă'?wYbmhr7ZO4]J¿$?.ˋ[}{Ǿ35-u S|xg}iu"-%dco4q/5]Whe_ǗZ]͵h࿇:ֲKgYi^_L? -&Ɨ<3K|H3IW<9kԖ;KW^u_k$Z8Լ?M]cok~)|%-tzxR׺pcٶ^{@z}Z}?u^@?|>$𲇊?p?@6?Ɖ=_hĊV/EῊ|Uτ>NW|a[|?O;5o /ou/[xSPKZtkt}3ĺ\hfiG}#'._x¿t*ֿ9f溞4?jǎ4P~ض/״F++[ki,s#*L K '.-N5_Z>^~|6{} xgTw.Z7,h ./ a1k/BQlyğ?\k 7 c?/V'dx_@<+x??O- Y&CD _(K§f<.?kfOx+Es?8xoe%/D+YM%',}~ӟS/H?ψFv嵗{e櫪xVR>_K^]OCWÝBk9l4j~"~j1Y_Z]k_{YeؔH!㏉#Cdk~Ӄ*s?f<- e r<|m TfM~dV:m?DŽO|4?/'-xIwN<:47$uuxf;K6j-jmwh%&֊/j>5i^5+ ^.𧌭4-[LN|IN)/C~"|hZwĞ wkNxMNJtoT>|</$>?t-f>'<7ge-gڛjv6 EnC.k7g#CR7·o'Wvo\/C|KF|@>(xϑ{_xZZ?lj`}o95O:/-5_3O>g DxbKOA莿M7zXx,[pL|cI࿉ # ּ1iGI>5,#!tiwwc̷Ьn/&Mn|ȗ]|IhƧρiǽkV= kY?gON+*C=w?DAȗԿjCI_)~&_ WjMnx፝ܺf\:|~< $wu.<#%C]oFM6 > e t KӼZk,<E[j:ukr6 %}'įxGǖ~ ]ZR!}o_j}ڗ"/x{O7׆{vk/Л~Îpp LP@P@P@P@P@x'$?Mש%a?h׵^YP@P@~7;ƛK?iݳ3PM-Rب/&xv-~ [^e&o]]6 ' ,<;i0Xi N6bc.a Wy\WX?egKŭ~^5 gľ.5'3A:nAO'߰n|= 8>gn߳7K/vg7Ư3x_Sχ5R"O_*žK+9tiX[[m+[;|3uñk|S5xw彜zdZx:zT~"ѭtâ6"1gi[ۧ.;:]?⮠>56G]Y|_hv_^i0+J񽯇AԮ|)]$i`զ T{+v`?Mox;(xZ|iѥxO>쿲 af MzX? ypie쭵08rď (%E||s|Dd|Ej>'GƹxzkKZ'BӴ8%:D=.Z|c^_^k_-|_maGh~񥦹odׁ;K{no|O='W[ZoHuoo)54">A߉\|M}zUZ6F{dׄ,uMw\2iͣ}m] ^.? t~OxIf~S➵xZ-@ x:B?^kWR[k+y6fhģVğzxc /C]1 V#Kol_i_cS;-/uoMiof׈4)7û-n xR5cEԬ[¾kKSi"oen|}x?h:>(ex~;8?|;"<_ Yo;&Q?`m7N[sϊiڎ{I῀'y.^m»J݃R5W~>|#as⨯T >iڎ7ni'o+KA6i%ؿ .?C?lj?< _?Wɺo~,ǒLemPԾ |2au/]=CIMͨ-֩iڍ麆{9᷀nin^] m pii>1V:Fپյ[)Q ף_O~^krxWž/WVEԯ_>"kZ_{+< >(+_=`c#CC_O>~GV9/@g^_cg?? $?))gQwy6-g:]OZĚ}z9/]𶳮z<>R1vaЭ4 (%њ :F!44?A_Kv<ğ^&o۟h^& "ͩh44|<{׵wZދWtxMAoG7F"Vgx*^O/75_XYԵO]m{ yhz <3p%{-%2#Ko7)6^:l<^ޗxJWUo4n|+jŚEcm~XRi݇wkmb(?fki%_}%Q$({4i0˩ɹI._~ ע/M\5mKU&=0^kh(<3%K- !||8Ƿ!l>1oB?cuS_K 'QxBUu-5&4WX-ܰ쒲=8cсR[i'~? tI+[$/ռ|Ծ)T־ux|- [a3z7lY[}0>fo ^(}_|o |_ͮKlj7 <;/܍WX{CV]=FTR| }-SxMOA]ì/Zfu}ON֛gMmu,Dk3K)|vпُnj//5k}_uafxsƖ忉~ =/Ế?,^m{^kٿ"Ն'o7?QkU˭> |MtxPΖ\^ __QIv{2Mu%x]xPFQֵ^ĚF~kCk" sVoow>$]o^ Ozn6S)HѾ-t]_~[[όZ׀U?bė<#FPе h PЯ5/O7Ex¿=+ hemރ FBҵWi>%-6 WOdK%`V+Xg\Wvxbws}D|\o5/xWO> xo‘f᭦{?5~{^xsz׈{mrP>z"չvGĿg⍷Lyo%%$k:w▵i?|AysJcKkoQ-.5i޳ G/mt+#mh|J7S=KG$̴ڳ[~$4/|Wl*|7ZU5_Qj>"| oCMn] dIoOy5kIva[ⶡkx'Aoݧ9/^)|QOiV.߇*ho]xKڇ|'y%ewǢ$/|cm@u=#O~'3i~.]J\_KŞmm?s4TG5 [I5 u+KGM`i~'x]C&k7#׍YICmRNuǃ-o(y_)kǂ>Z?w6OM3+\|+;<%u7Z/ƺs]GOŔ1jgCR){tڳ[>$I#OA;UY7 ja4^|6׼A/~X#OoBKmkt k B=8Ӝ`u#CH>swB|7P)¿G[$??e~pp>b`x?_hm8>+hz3{KZG4 |AG|4G-<9htVWė Vl%T[FZtIմIͭ^"?!Sl4V>#m/ln,z0Ii"eW/^ ÿx2f8~ sOAPM-{:5uoE#<1aws[|p<Y5猧<#g.Ư_]鍠}4΍fxQ>V}|O%O ~:֡j5G/K]-m3Z[|I?m|}>EZOLqGB7^#WXͥ5H Aoٟ%NJu\Zx][W;WOƉ|~OWm/hy<,-֡uj>2DžŞ 6¾$`>'i.A[(@=h_ؓo x6@ص\z?'oút^=_'kI=!wI4'u|Aljq^}{ƞ߅>K`wu"5wɦwQi5e񰮒i_/Ž ^)}j_(H4?O7_ںj:3hyjZww%|_QJ5p5J= xjۃă%oH5G Tg}45/uGP}iu +Y,v-# OuWl/h);+t<¿V~&|O߉3߆<)?_ G߇+ex7x+ڕy]>ZRhL:5!$wc:]OZZy&wP𾱮Z4>Q1yhvuhoipGY-}-OQ?X=@𓾑ᯋ|/i5}Fᯇse+]auZtGXu R;JKV5&F?f?0׼;~ֺ}Zk&_xÚ,=_ukɵu&VCk;:]?xJuc擡ٙ|;=4[xphτ9qi'I6:L3jvrw6Л[‹j CoOT_ ?T?}|6X?cךuׯO#Lд/x]2|ae|<77>&Ş]ͮkuy7$ZM@_~17=K7vPv^ּCGgSVZ>_k΋}Y x}noif{+v(Ž ~(}_{w<5 6&u}Bᯇ|se+]aZtGWԵR;JKV_#z{v'<Oa0oN)\V#c+Hp>|~OWm/hp?P|W}xxO~ µj'\$?(O 'g73JXcvWuq;P7\мIKvs^ k[xZg6??v J/Ǿ Լ+< |;|/ ]SS<VwcZ$VzoOç|c8==y oP@P@P@P@P@P@P@P@P@P@_?a6^u#Ou5퀏ڥ!/łR^?w~/xcG]3>h4}#Z|9> qg?+we7|B6.[SiۢVZ/kxwVooo%|\ AYx \t-3@1vL_/!\-ui&.Ǹ/@պ$ú~g'Ob_Ÿ5?ؚn6xR)Gg/u}ǎ,3Nqj$jm`[-->0[NO9}pP@Ncӌb8}{{9z@m (c?NIG^zp?,( ( (8ǧ:q9<sP@P@P@P@P@P@P@P@P@P@P@x'$?Mש%a?h׵^YP@P@~?9ƛK?iޣU*:5K_1xRߊK[Pi㷹4D];S!M;Q#F]ZM$_<y;IuhVm+v9_gK_k><'iѥ^u?ψ6Ba>gE<5/f|ռ_٣%NJ5i]cǺ/Zd j:oR4/q_ %CxkK;M>osY%edo km $eAc Oǿ5;:I:'<05_\ (nOk7}H?҅.~ .|Qxڷto>4mO><=xs]|/컏 b;+)֗g gam & RjO|ӁF;zKo"v_O#𠜎 ^;rzzGN t?@P@=}=; cWs6@?:{=(`p1F23w;Ƕ0==r0|@Psӿ0?x'9al;G'$ A9v=:@9ǰ~~GP?NG9Ht9uZ-^~4migy:o5M[PR귾Ҵ xfEմ7닍[V{~ǮvI{:|!sῈڧ]]?@'e GDž/ys ^"/xWDM@ϥ:7{|V9џt61xrxOx[$Io#2Uϛ+F'laͰ5do_|Qg  M2 |}x[*1ho.o|+1!cy!mXp? 4|V|1K|ߊZ׊<9E+_/ᯁkG@<zτ/wi$Юd(\;FZtV?V|@]vmm~xzwMӿXE:g+^OXa-ӳ_2?joYxUD_u_Mƻj-W@mw}D S:w4BY- ɭ\4^r%mv;3_ɨx;o4^1ҟ~|G<xbM~/g`Gjm`u|2T>"eW/^ ÿx25_Qj>"| oCMn] dIoOy5kIva[ⶡkx'Aoݧ9/^)|QOiV.߇*ho]xKڇ|'y%ewǢ$/|cm@u=#O~'3i~.]J\_KŞmm?s4TG5 [I5 u+KGM`i~'x]C&k7#׍YICmRNuǃ-o(y_)kǂ>Z?w6OM3+\|+;<%u7Z/ƺs]GOŔ1jgCR){tڳ[>$I#OA;UY7 ja4^|6׼A/~X#OoBKmkt k B=8Ӝ`u#CH>swB|7P)¿G[$??e~pp>b`x?_hm8>+hz3{KZG4 |AG|4G-<9htVWė Vl%T[FZtV?V|@]vmm~xzwMӿXE:g+^OXa-ӳ_2?joYxUD_u_Mƻj-W@mw}D S:w4BY- ɭ\4^r%mv;3_ɨx;o4^1ҟ~|G<xbM~/g`Gjm`u|2T>"eW/^ωx.i|R1K_\;cM/rX%Ŕ1j؉x\ߑh(aj>>dw Cѭh ~4oH_jǍIԬ<1yaohz"B ( ( ( ( ( ( ( ( ( ( OٯIe~0ԛSS:+KM0k=p ((ow75;NJ8LNz6v:d} DOm=p@x<1Ko+ o+^ {yk]oTr)mKk:IḴ{-CD?M>;czCh!eccNpITAl`7a.4F͗%(xG Sx'T0}ӲZgI"HrtIWǍKMgY-ߌCBo^= x8O/׼i [ֳZꚗa :xW+/Zx?~ INR[KKie`M(ؿ ;1x|xM#_._돨lj? <;/j/u͢<:楨]jjW\s5Q G?+׿*+A⁨n?NG/]𿁦$?ۏ͗uuhebQ.H5+.K[Ԛ( x^?_Z&hzii|uoh^0m~xbIկ&5ֻXj"v_tM+dzmՏNgfeD ҼqmjW>Ʃ$i0ͩ]Boen /Hgf.W_+jgkmc'^_^kק^km?O3Bм9[MvxsA񅖕k\zFw6jQ6G[|{x/h-Cx;Z~OYkpS~/җK~k:-d|-=ء ;1~uIH4䚁 럆͗uuhe-^RnH5+.KXjM|]۠8=='c8xsXhx#QI?#i?u_}쿹w‹i A_^F= qpx"sPO>|zNjtCGsB'.5ۏ7s{W/Ek}mjַ;g 5+[ȿ?ï(e hOxSŞ/WKоZ-d|-?oi[x/KҼc B>/+O{-<5|?& k^2^-KӬ$.( ( ( ( ( ( ( ( ( ( (YI:3Ŀ3 '37 ( (=Ҁ>f?K?i⡎G#;ҚTw_?ȽĺŗhN% V^(Z|5Q/]}T5Y³]bV-?@dՁZZ2xGO<K۾7mKU_{(tj#S_fdv m`Z'VGj_|/់:[_oS|4Э_h:F>KMP^x_khp]E}HT鵬Vz%kl1b g'':ة#cu- x@k⏄?Ox[N~a'Ŀtx|Gۿ*GHltCoǪM իKN)/w?M% /, U?K§:'Y2 O}sk4J _?E/H?~xͯ  +VϩOQ0 #?0 #((c?NIG^zp?,( ( (8ǧ:q9<sP@P@P@P@P@P@P@P@P@P@P@P@ 5,jug_3fOZ~gזnP@Pz|N75;NJ8Ȩѽ7'ڶ~i/Ɵ2x[SdžT ލ*¾K+)֕iv}V]_w5k&H-6'Bxzk^9xSg4쟰iPͩY9 uKm|~Zm" +{73j %|E[ص4 [4񥖇okҭƵcu/ES6l}$3He4/e`N}#b ڥ仝4 xk_{$Q#E G I?G$Py_m3ӥ>YFϏ Q"J?{?Am~Dg{t;| ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (<kYgc?&οgt0?{,( (?Jo/9 kr>"B ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( OٯIe~0ԛSS:+KM0k=p ((osAT)ʎg ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (?f%1RmNLFx/7L C€ ( CiP*;OR$( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (YI:3Ŀ3 '37 ( (=Ҁ>f?KkwBܨ}?H ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (<kYgc?&οgt0?{,( (?Jo/9 kr>"B ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( OٯIe~0ԛSS:+KM0k=p ((osAT)ʎg ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (?f%1RmNLFx/7L C€ ( CiP*;OR$( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (YI:3Ŀ3 '37 ( (nӾ \5sP`IZ /\SXԘ,K˫ M#љK*:5{8:gPՊktO;NÒ1|xsLHAwOF.?TGÑ,]ˮ=?shG 2/`Ƕg|,]}u^= ?4{hv9,] ˶\Ahp/sGg|,]ˮ1`$288'ti k E}TMu|F:Ѓ`a D]SZuFtO?ŧ'Q_ `>"qӧۨ]r50tϋ^COecߋ;ծ x4g|tO Nn٭4A8C)P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P~K,cڝz__nև{_u囅P@P-_u?sӅE_>C3:mY/|;ƕ.^.ixwD{KxGT t;;W7wmҵS0ӯy`2ok{;_m/kk|EO i:|OKş<+xW¿[>_< wk; hbFRRq>[o[|ſC3GmjZ8k:Ն 7wƋizֻmk\,ld;o_x=Ʃ^DM9d啟[GN+uugo~[/:PAD__&>9<^ ֵ!'Y|Cgo-ktO|P˰Z {{[4O׷ڱ/|8м_|;<]C_𽇅5?k^6<}i4gt#Ebsu&}gN/t4|Bk|/Х?Xk2>TajOW-7 N㭗 x?~1=g↗im x>Wk9^6#^x.%#hEȯM몊\ž7k g? 4?v^#ĺo=eQOaN ]KSt7Z'Z@ubLQXԸ뭭bŷ [ٽ?< OtPgoRZ2 ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (?f%1RmNLFx/7L C€ ( :c(:ػjyg339t5Шב4CzƙkREtbΑ!|?em}e62+ _KO}CK,RѮWm/gO=뫥jjK)s}&䖝mA|:GS?>Gnu}#^tct-7i(:7lJG+umy<=ނ.TezޏMe){A^鷚M˻+8<qi; B4&VZ|Xb.uOmcR? sReoZŪq ˯]?ďKh֝iv:v|QyؾI5i;m-}/=Co+i ugʼn{{X$q$4 ۾oZi/4"-|lExöQ&iq[[''-9\Soo F/<X/txWu<^v-CN՘^h<%OPRWKQړToX?Cм '[~~.?x9sUkqKE}ZP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@_?a6^u#uw5Z_|Mq5-ھqyf_O,h-1ax!o ^"çh/*/ny!5c̥0|Y'9E/!5~/z|R:t6蕼AIx*_z/A# =F!0???T,_z_Kg ?DB0{|R1l`xI=?c̥T,8Of:xK#K1Q8蕼>^!JŞx1 cm+y/3^@?ctWO_G+~]6.fBÏD|C?i|ZNp8zz+胙xŸ'W 7\Hd|G&M/-"7 {6Hdib%E +`r{l{%"B ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( OٯIe~0ԛSS:+KM0k=p ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( OٯIe~0ԛSS:+KM0k=p (z`ӂGGo}{.A{d Qǫ){o=OtmF[/:c## [t^m{kYtd醴txp3>Y.OMzn-TKdSro}]6ӡ`2@ߩiwI߫K[+IJjz'e:{c1'OUNv Q>ՃN^YwMﵕvX{Z[P@Ol}F1@zOUOQs;`7$z+wi^5KmS{ѻnmkio?+Olt{~Cֺ_d+M4-Mno:{c(k/+{ i&p>N2'!I\↭ISWv?ʓiV^݇:=ZiV]Z֪ _@A@h}H ?@ЁیP@?@nz@o׷83я4~(` h<~ixQ{}/1S?>1xh-}O8?(v̍ |O@_ٛ i? 3OO?Gfo))#@{}/??dhw̍ f'OR= |ՏbϏ.4iD^6\[!mM͚yG 啭-S E NկJ9@ \xR9]+A,9uoK=Pa{8%$Kj&VDEYWc@5(œy v'ʗ@>,?i!鑏 y1:c *_ihh߲}|'c>3r*`@~^ڍʗZ-??x~_̃F/|8Mq6~[.k}BXn[y.Ylm'XD쐣0fdFJVqsGLL?tgLH }v|t8@?O^3ϯNP@OttǮGAm+xn3`a:~@tnܐ (?f%1RmNLFx/7L C€ (10G)MUmokum|EKh:_I/Cߴi>"'?>&dŚo˯OeZ/ຽ6Ӯ?%2zGW^ǾPc>8%5e'9[Ωgb0(aZj5St^[aWuyEhv{ŋ+^1߆|-ύ~;l<-+=_5waDzzvjq]k0K xQ?iQp-[>j>qaK71W0V U*-aq+(IV"%Y]QJ<%&6ߖ|v'Å})<TGrOI_Ï ky7_E΁]X^!;;^%٨\}uOr&|gcb3mlҭYC, ԥQ_Bnt2l{OiV"T=Ib)ӴIS59):q+\{> |\~.^!ld^D/> R_xsqO?CMX&I:K]NTwq#00S1ps|Y.mJ1 +#Wաx\k*a1\ZNkNN䓧5Vq,̭$Ӽur~-U|K(i|U0?Z8:{ock /j/ W񵄭p_`3,syp EF0ԨfU%' E*pu*&a#..ٍW MMzӧ('CT[.FK+Tn5Hu^'G. kž&5_oBB{@ ĭ}JZimo-^Κo`ḋ3~Ws n[e|ڼ|*8$|eFxVƴ0ʃ\TthB3QIJRRqRJQQVݼƋ} /%]jo~3Q|E>>u '{bOyt_^.- ҥ* 4yF,t 쵻]g¯k>>xM)x;ڤZ]~*/ A2*Le{=s[gï|q?? /??plm?_KD]N>+]xp%eTskYj^ͲC |2V?b 4KQNbQ^[Nѩ94*\i+ͦq>"Җ& cY W=kѾW!tM CߍUEYYyCZ^I3YDžsS}qV',M\V.nESQpXOƪl"j*sqU*_Ư4;niIGGhχ_d|9X|cf<-_ik|!Ɯ-nw7<_|4_?`>\;Ed]cx}`eV׳lQX(*G_%B֪8rҌR=*9rYn#MYr[J|]G:/Ƴ^ƭ{ឥꚝ/B5 ۗ{닩figyd٤zJap$`>$a! (УbӣJ8RN(Fc1 &nfӋo}}'}lx&?1Rbo.RH1z<7y1և2+g '$ABRd7ռ*VY'6Y?G a~eeK5O|Uj}]]w{gGpQ^)۝{NG*'U⏎+/>3_Y27KxljjZkV<+#W^gOE|jSj1o|<ɱ8⪸|J\N_K(x1sϳ<|u\ .]`Ԇ+ͱFWMa2ʹ"~ZJP{%(Un1Q+I9F/qޙ?gt Fw^O]M6~V GjzDŽtˍ6Ķ:i4Zi^Ac*1K^\5V-7GOGG+#W)Pc+RwNTU8JM8K5Msk\rM|_w|9O]Cԯ~ |%[|H}7EcM*|;xږaᝎkS g;Oi9oR-.E8n"K+>XV:Mxlm~1 %^Xl)yzؾV;6YtJaOYQm{<ѕu5N*DU^I{/޼#RHV? |xSg?O ^oo:db>L]Qkхֿ^m]kYldym,++E/_O)bFЄWU/% SV\J˞ۏڭƣZ}]^YzTzLquo-:Zla}j iZzv߽[-otj=򲵷Mͭ՟2|Ҿ#\F~=|Mu z?K״So¿y#V{_|CKKǥO BXYZڦ-\~j񶯭SյM^-|{y^\ѯϦϪc޿똭[AʓHAK\xEuּa^<{?=Ǻ?|bRxP|UZO &?&l/[ DAfCߵΗ-ᏋntK;RMfpj uǟf}o?~6]7>t/QSor3GHcǿ|+o>#Ǎ>mL5½y ߄ SrZH/|k2Q-r|O͢Zx]g_v ;EMPZ4r~F2E>1ohڶ -CX珼!O^:W*45 V<%5ozYIl qh:g3o%9?sAi޽T1 בO?5{:y/O#~(kZ7cM࿅^Ѽ_xcI+:|7oe_4˘VB_K- w'G}K\7U]m_5oxvW:?5;+U]mEŬEvI>=݆|6_]~a~ |B_5G^x'}cR-OoM-b[if{A!@P@Olt#|<u`c];ēG<ߞZ˥E?i4߂G?ixN#8c< ו"YKr^}/IOoF>kDRk~gy"-'a%#{/*qMH?jYlRmesȼU= ^ߏ_~+?->-񇅯5?3}|4񆳪x+OgTҡ}"8k>%$VǾxWdFb&͟H`sk/5߂ Eu]7:<]g;h7?__ɥi!$iKH> ?l7x{?&cBХ^C#j7&Kuڗ{?Q|$м dMCHӛN׾'i~{mODׯ趗&69K+P$J74ڦφ~5-CB/>!o.,uM'Z7kOm͌7!m_0eh5:_h\E$LKo7~?%^&-J׃~:H")_Bmm[U5 yOipZm*vC⮹E^<𷇮tZ~S|C֞1kŋx>?hIeC=խ[׊t]A!k$PkE>wax8o?%ӵ~j5 "o1Ej VN!@O׎Av귄tP?IyO'< 嫣?'GoCG[o;i-Ccxsw>^#׵+ᄶ-FQ i ٵ+(߲q72uk.=NjN20Go`R$ ȪsD}[O^=:zcO@3|Y¦HH@1{|-. Mc+P6H R}Ϟ?ikͦxO/}_ᇀ>$x+_maV׎$x~k;y,vNdYJZOdp6"?|;_h 3;k}X+xƻSO{^?oh>#[5m۵_hKg:NSCk%гtCᶫ=>ց6/,2[Κ.7U>iK [? o*-^A-{-E|S}/?>!Ӿ/꺟5oAi#VZznZVB_(gi oz/!_jTV_tksPL{b{hFEiK:"R-~^_|>ռimt?41IAo;[ȾTG̾&5Sju s ߈3\~4?J㏋-΍{y?gxM5KDi"]WA.ݭ Yikh}Uf[3Q/=#u[O|G^?cau tڮRMn)uyd &v[<|bGC/4|2?hm-| ֱxߋY}͠EgTխ/m..5] M] |/s⋭GHt P~f,_ xjPY,iL%"QӼM~YkH;4쉩6W~/lfd ]3>euY~KS n4s y#cel'\t+QRӫ[FFxbQI\9TU9#U(TU¸՚e{ؖ:XfOjc[[m$}qږ{{9_ \]:|:p)F]yrJ*R)N7 nc:'9WuO Jt8l&6mN~ib5;G &IZ*ɾ#XG_??f٣Fksޱ_ii,[ '<'2ܣ|PjSlnm?hEx:Ïs?ߊm}E5Y4k""Dф eP\7c1aW3fE|N;bCSY=LN%dVW9e*Xz>ʚF1%((Nd 1U#de}Q9<[ A{~ouHt_뚟?gmlSaJNol|;.fxӊ2>*7G2/c^.BMvQV`#R8ژtNXXaW,F)'?iӣ.XbN0m5S 4fԛJ*'ïmeeg߅[9|$鍗j[\TtVO"ȶ'>⿬hqNA]/^ӏ̡H;֧Yկ]F+nJT-*t6%)98Y{gEhzg< [ׅx+mƳ|յ/Oվ` ^9A{Ixh:17"Oz_ ,Vq F?fibBxzvcN9ॕxbeO*~*xgF\\1>*a1pq)šTL,c[}>Ԭ i~um;QNoJ}f¶"k2ľ'J/ߴ6fq'9+y 63`(`p Y:tp L&*1` :Xԩ(Pq)I~r|JW+FƯg\[7K[G/xRּMZx"淠W%uƚ>*viQXSI׻U9o'?%vÖQMĿY-xgl__{r|)߁Gfe~Kּ!a[[P 2as Vy}X+a9k <|DG[WGB|w4>^[*CLMҮMVa4SiJҋ}>=a/>&gi?ѼexZz{{`4=OŞ#$ڼ7,.;^`1XxPn^N<4gRzbThV;I%%dN6jIyՇÚo| x[ׅx+mƳ|յ/Oվ! ^:A}IxqsXq'n؞+{ٞ+98ÈibF]S,x)e^)eR,&1ʌ0TѩWSiƟ%?gNp~46o$T{x/75/x[>_~3/4/ Kْ[D~5`i8Yy.;8 q*O:3؅y=* [U!K',J4*s]^T(iN"OF6NK+F4ڷ=;m?oxcZwl_?Z_ izA>'֮b)Ek{hl/n<;h- M}kqӊ:g||#5[++.8J^)*0<b?b.&W:tRF/z#RK*QznrjRtť%OiZD~uijVP݉c]>IRGgGø`ҒX<hCusxQ Ao ǑY~~>ExO~Ξ ю|A}^t|֟5ǹk'ŧ±LF!=<ſ/_j,>hu]iW^-z s _]&myhjBO#= wA߉fhgi*Ox6_| CO|,:.h^cjzmK:nma)5m6I|/-|}F't_.|sk[CKN zxnH"֯5]SX9ekr}Qcj+oZ}G-/ij{>o h4f ZEm"3I6x .._u8uAOjm'xJh5ož8/YG.]YhС)|ti_9?o\߃(κӢYE*=ۭtZ? >fA_xڿ$xP}Ok~4b?}:?)`a.,d!'˰h>uZXdá_6zu(uORg{R|m}-"yuYt/UMWZQW~ҶZwukoG|.frmOSQOigZ^]\JH`YFʮo뭇"x⿎~'x"ĺn/L<O[^y;xz^ -~Q\U9_8CW`MyƏ|Doex6|<>!Ef?9? ~%ME#ƿhk#K^jhT[8^SaxkYVֵK."S+ORR|/n]7Ěg~$Ήo+&6]IsZjvf_t?Kz'i%bw߆>x/??im _ٺe?`>&/nn~|s[O)!MĿtx[/gSL5|)DwֱM ~[s^64{x~MV-&+RK|Ȅ{tYh~>"D#mk^G Y4}Ek_u"R;$is$جpJG2Y+yK([G{]kߌ|C'L_ ?j3ƙx,:<뺧xm3.YԮo[_'vX߇^w._xC~ .mTT+Yh6jS薶j/kӼ6Ѵ"ps/K:?[ \ݗ/DسZ@ȀcYj1p-[ɯ(6>b5t]h~3xnE욗>xWH4DKh5H8@eHAΎhTJڿHTy,m.x|xV/58x=FO5+O`XեGhUxZS"54\N].֭_|?WX޳xᯊ3Y OH-MFK{uYfpJ)AN_V 3) G;xKޯkڦֳၪ=y]h[ga%/4,WS6vrhv2Jȍ::0*ځ IE.[uzyvGφQHO_k^/xo~ OO[jx;~.ya?oD.ҬE5iqr?)g1/IecP/O-گnNic^l_B񿅼Kjxgiiе:IMqݯawqm<3G|R$-BN#_gƱ_~&ݵv$W:h  7W.$ӴKrVxt"յdV࿅?῁\7z{j؟ |W~"w|a\Am"I-g/ڿK35xOT 7''_N7'y}?`5|-{y|XE|\_|fg|_=&;4wMb;-R."72 Oӿc𾏨B5_7->xq _ 3mo4vq}I F:<P~K,cڝz__nև{_u囅P@~zܞp=9zk' 4 }(ӷ/ Nߋ0܂;~/r 4 }(ӷ/ Nߋ0܂;~/r 4 }(ӷ/ Nߋ0܂;~/r鎔hmaoYO?q4UDo VZ;l$+ }(ӷ/ Nߋ0܃KN|GuI>wEkl!P@?^q?::z$؞ހ(J1cvǯ^h ך?A>AۜR1OA?J>K@% ҏG}1ҏO>6Z>KA??,QE o?nA|܀?J>K@)| Ԟ`v A> >;I˿|P@P@P@P@PKc\HPK.A%OEBPS/img/obiee_impmeta_metatypes.jpgJFIFxxC    $.' ",#(7),01444'9=82<.342C  2!!222222222222222222222222222222222222222222222222220" }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz w!1AQaq"2B #3Rbr $4%&'()*56789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz ?sYHTN$*S?Ұ:A _,=6vVօ`fgjP\  \  `f8Qq!NzGOF$y%_)1댂02A{u\  \  `*fGiЂ:9&q#'fYwaF3Ggh'W_?G'W_?XP["lY>n spsPVP\I4`'3 2=2h'W_?G'W_?Xf.ovdqqGө?OQQA:ƒ̥ <$fxK(xK+ tK& J*?Z#ʮJ8yhxK(xK+&KM\Il6Q:`SThdx扞8֌/9x@:/N?焿N?焿° :+_nhbG'<@~5-ܤ ,da`zQdG Q W%W%oq ʋcǰy N?焿N?焿°YbII9@"cΦҔE4&H] Hh'W_?G'W_?X'IFxı<`s棗Ox9bw =d//^->]1xjLX-gg )C=r  Q VD80Yy7W$Mds9֋-fISopQ',W9#h'W_?G'W_?\ۣHP\~'c? 8R 9;I<0 'u@ouq<%uq<%Xƭj<[^xí:FS$q:[_DNv=3Q@5:A ?:A -(n`Rϱ7϶@uj- b:Yuq<%uq<%ualZAGI(ڥ0@^j5xUO%h3Fouq<%uq<%.4bpE+{lK(NG:gR&qwq34{//vO{2,yx;8=GBj@u\  \  rTQ`:N?焿N?焿¹*(o'W_?G'W_?\X//J,[ Q W%ExK(xK+uq<%uq<%QE:A ?:A 䨢u\  \  rTQ`:N?焿N?焿¹*(o'W_?G'W_?\X//J,[ Q W%ExK(xK+uq<%uq<%QE:A ?:A 䨢u\  \  rTQ`:N?焿N?焿¹*(o'W_?G'W_?\X//J,[ Q W%ExK(xK+uq<%uq<%QE:A ?:A 䨢u\  \  rTQ`:N?焿N?焿¹*(o'W_?G'W_?\X/}$88 1yz}]zGCʌPgZg/􆉦5? *rs׷CW%?u? ]F?:Ow3+d907 {֟[-;Yky2rMiXxTc2ˑܠO#=*?ƹ7SKUxG]r: K8W-nb*Զ"{$J0Yd<E >R8tkQc , ^8JL\9 Ies PS_,궑V*'d`Pps]vC"C3:pPsǽc'a= .5~dž>cn?:. .k a6D@t=z(N{\K]& [gѡHQ &JyV( $bA$ zN{\K'a= .,mnaC*sБ3,q\2crkN{\K'a= w@a 5;[J¥Nάvq2}/mnqw‚26@:)ND?G"v_t>/k+(䍙Y`pOn㡫Ft5̯,$I2}Ǐ:V"v_;s}/Qt*6I5u@3ry#=ǡ;P1`*[(;@ǠZkD?E[ >vkh& c]1ͥ _>yfqkw;s}/Q([[V,9 ׊x!X ySD~h~23Z?k w%xc1 ׈Owq[cNV"v_;s}/RVb7t#~c|`Pjp_Hnn]@tlqzZ?k‹9Vuiތ @zz Q4vH2UQ`d?kv֐IcPwQy"2'ןn(N{\K·KhC@m#p<55m~1:ʐ,n  30=k_;s}/Q()b#G ʸ.NHNk=OF`7g?k'a= ??@c[ksVf~  w˹yHs]_"v_;s}/S./y6nHOjܺ'Ҕ\̪GCQc?kD?J {,5/($CU@XI>)7qׁd;zN{\K'a= .Źƙ%K;1]j(JȮ;s}/Q)Q]"v_;s}/QtEv?kD?Eu'a= ??@qWc(N{\K]Q]"v_;s}/QtEv?kD?Eu'a= ??@qWc(N{\K]Q]"v_;s}/QtEv?kD?Eu'a= ??@qWc(N{\K]Q]"v_;s}/QtEv?kD?Eu'a= ??@qWc(N{\K]Q]"v_;s}/QtEv?kD?Eu'a= ??@qWc(N{\K]Q]"v_;s}/QtEv?kD?Eu'a= ??@qWc(N{\K]Q]"v_;s}/QtEv?kD?Eu'a= ??@qڭ a3L8D?U?)mm}jC _[ۋd{c5R;f{."1.q>.hɿhDM|FAcwqmSwlVIAYnQf󭠂8LV$DJo 'o???)Ѡ}+RyH(z:t]wszj!l@\;p3nFȨW;T\|A4O4h"ZJmyuH-_KtqJ.@Ȋ `=3N_Ewz /;UR8j W/'j8j/.ల.6W;UFI;PU[[/`\5 4lH?ժ(W:cO>˻3cbn@"ko?&fWnۻ8^1 PHqSR_Eiqe&0FT b3⤼{xY_U'L #I@"=Yi5Q@Q@Q@Q@Q@ V[F#r8<z,࿲25'Ap{QEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEu/EnVQԿ \AKq4EE@ {/ANY ɠD@TO\ҿe^"-{]Et҂g=I8q>QA(ݸ( x_;-QYeyl@2獽릮Fv4tm_+B&M8vt%OɧirĩYR%g dn*\8S8Mt)je>s^m U򣈡p?1 .Ü_xѵ;1*ZsWvÿb)`qp2)rޢn>Is<G3J 拭{Ox&\Ǐ6D㇌ER#x9sJ%7>u[wK5hH`h$v$r M%έnlhex]G HpIKbEfoil/.3~'8ٝ.7}Nv7V?ixcq4+ L0ʗv>N09_aU 5:ƥcH@CA21T4b-KJլCXG!Kn-nv++6^nRd2gʑH?+ IhmLK8Kf[s,xu.iQYO-5R6GH֒v"+`cJ(zTNHB@`^ER61R G+FQH((([C ^=t5joO é[wOn0 #iTA$`r:ϫYhچ{q5EY:IX1%k#iz}"u AOYde7bda N76H-.DiDH>yel.#b‘ L6D7 SEs,{!(3v2۷'OL<iV֓- >b#w@޺:(ƁˢN}-M8 lJNfM#O}i Gd%0H}.9i2i ,.|ea-t$2n z#k}Uugpq~<2 k2aeR`Fc^Ey1iW>Qmq.m2#33 7\C4ƟCi*^(tI#+Q*(crJ,eUbV#yťG{P6Iai0*Ӱ`W5q|m[Em:9.Ads`oQE/@'68./p oL‘+CoP{ijz"IvѴBC%S! IIpwh5լ4v-*L--f3~R2eOJjV q9䁨4!H"W>Zdea=MA-G?j]<#=G鞵vi ]H ŏ4>+RJh_̗;P0#mt<*,/,y@bbaIDykj-nlԵm~eERAW_q' nm9iǧEk620jp Uފ^)*y0OigiyvQ&Iڪ0O'ަBo5+)K-+yN S[szL ?vߺ+(ocQa~N6zW;tb]F$.fXȥଭ2 H³ AiO%[n&!H6nby) 0wTQp9ivMH!X/{],5Y @zo k4& x5amNcnH1fAfǥQ@ WxwLP.ocn 2999( (Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@aEKQ[u/E4Cq4EER?QPAEaZ/Zҿe^"o&$`|ҺhFm\dq*OI }U_4M ~nxtw+hF<㙙 B2L9Ƞդd"n4g07(>_*3͞;:)󰱏uk[3j|6h崌!v|鍽=nUjLdS$l<$H>M. x] QR0((({V׺O] sڷBz:0U_ Ų%֮<52u&7w# 2R(( Xך 4=o*;wpbeUn;6#Wz]:t;ig;vڻ]v<TVgXYnVɀ|+R%7qEf.bZD$42 g]̫Ѫq \uTW0Ă@@}986wijd_(IPIb21,G'pݢ <́ȡt(TA#"H((((((((((4;^fRJElc ~ ҷ_xk| TT:֍pⶌ`V<00ҹ/4z.IkE5p~_`U7o*9J2Ώ&rwй+.3JŒX6].,Z{<S Jf'hmRh|GU< ҭjki5OZG2Yܫ_WN_V̫̂_WN_VWWN_G?ƭQG2 2?Ə];Zde_];ںwE,ʿںw?t5j9Yt4jkjs *jkhӿ/ժ(AfUӿ/_TQ̫̂_WN_VWWN_G?ƭQG2 2?Ə];Zde_];bM @Y0+.?'tmo.shH8TzxjmJ2_V!2CLào2=_2VN`ޢΉwAX /]"I=z۵mQE5 >YLٲ:WPK09~bF^4oZQn4 p##c@Jl,2E{l'+2"wg< e4Ai*20`?ym Q@Κ&5͝HǻQH((((((((((n{okw*% BC?5}?/of$x]UnC pGsjqoy^ k 0;`|sE!.\]}ͧ4qS$K9"7X/sNvًk9'e2$K{vKQtP%l(EYKhNҖA[fl``Q9ஞhlZrϗ;}޼ERt;n%dhܮ7Y##+FFA#Qڻm+A_?WxF["0QRG5f4(6_ܰ829m4+mLY0B9dX2SJݙc^ ҽHMrVzgўK8+?iæLlE͑t7Cbڝ~fYZi44Tе[t6ɱ +^2'[oNJjsFbE9`XP7Nݘm"*/Ώ(X)UfS$N{dM%Jܷ;Z+g;g Ƨmʑc|cujbhiݵ̎Z2Cݞ=<<OGwEy9ee_ hK63 656vXc VKB=ceHB2QRtwWޯM;O+on%GThl0=4v)j66]r( b8rÚnhHzx[gO$:@-`9AмGOt^"o D0f FFR+]54uWWH?յ/O+_7b/zqקmໝ:Kӭ4oK 10MJz9βC:<sY1iz|(K`Me#56OFZ\iuw}ʖ4hO< e 1Ir8z1s ;:ڥӞ(   AR#{?¾!uKi\};,Đd #hlm-C'Kw@Ņ923z(@QEQEQEQEQEQEQEQEQEQEd[[jψ5;;=Z;(lcEyw #2Oi2pݭ(ڦ0be+' vS%͍pZv[Q%,txËukpc= |R=Tn {S|kywryw4qJ|5`3ַOUƛš-ڼ/7vmqWo9vsZ-<-Yjɪķm|_O)ywdֶh[\,zeiڌ>u\2Zѥҭb1]y%r>d@;2ˆ y)FM5 \uxsBlaay P ۉ$_ hyp\.9O#hu$ծɶ֓k=spIٙ8=Ҽ5Vβ+n,JRv{ p (¹WB\Bz+.?'PUe<aGM_ª{YۻF|gf_kp]x^wxX퐍 Gϴɂdj?EUe}=;Y׹@')''Oc53IuK?7>_'X3΀.1,8Ϊ Q@'l}O2.la|8<*j@S&+x$y(cR#UU$z)QE0MN ̊цI = VPC4WG<$ȡD`FAuS(((((((((b"D2rII'wߥ mKQUӴY/Zcin#Ҥ.9V|G~{k#Hm-qn)_2یNssWa&eRHx8.Az.^_%yy%9_-=;+9顋XkkIk DEw&ۉ2( c n?K}jީ5΢ͭ G9 ]zImcaCܺM̓pޑV /]Nԭ1hR?YB-(0-ȵu}oͩݮ566A^ms( NIa@˭ެڬ,rr QP7ž/.u%.]Mca;A K13/zZh<˜qj-PkZڮj6v)XaFW4e_0>+^cZNK˘"[Pĩ Ϙl`gn*Vwae,\UB_EscǽaxB;$3y&þMݐ8 :PXOEQE(((((((Ÿ7xZ)+V`eA`(dL_YjM[A{Zܙf9L*ZM%x{SٻO5Oyi){ ɷs89{idI)Q'=?C4K<BȥIG(A"G 6=͑[9be`sЩG[uMRDTIe$ܬ2#޹'/ty{yVTFQpa8Сiv$Ѣ@(֛.dJ{h nwf 3IH-E|S![K=&/.]6qʍgd>2W77OicKyCcvӮ>ŝF=7L]p*⹆k+MMU}>qRXIBn%%nFv5["+71l9lY 0]qF ȇsh^n5S|IL,&Ȓh@ STi]lM2LXGX 0B2ɴ.O1ޟXGymc&p9FjŖͬK25PєNpKp<+Oؼ-:Ofoqo.cہ01coU V4{ٴ[GH^M:{TEbegFB0B?BoAjn(s"`U_+ָm{MuM'3("K_)1NR$@b[,H56Ɨ:]Q\Koh%f k, *_!Ȼ[^uqy j&X 1kX,%΍&hf n@5 v8r0C&ӯmue`*e]@B D;)_kHb`*1ҦMѣ[;,a=X8^ܦUjy$Ś6=_JHjiXy +'&G7v;;PZ-സ+ٹNdzݴh%In. DAHUxS2T\t~Q-maZiXT"A>S>\jO<\is]kjx6řsW%IV T)Bcٻ`DYH Ӏ8c$>HVCM6m`5- $a6# A!$&4ۨu1"Ix!VXKr϶aJX AȻ9_dF'j8M^ztk+ I WfbqV\ɂm/)i4pdG-8 C!QȂgEVc ( o 7{=a?'9dHbyd8DR}_L|=AX?_-KAW[;D|-jR)/V5\LTIZ5j @j_}줹akiuw%L\ P}l4XQ^[k4gsm%{T\[DKش0~x$ךιwIS/-VA%FG#8–!p>Q\ω.c7PnaaҮ7NEmv'o'U:xW[.b&7B!#@g;`N^:PoEyMUbnqsuҮl^^@6Y~HT V=CCiK(ZK{V` Ua%6; ,aEs3[4hcombM]tB\]#3LooiCijѦw ®I&1qvgEr汦Acw{Xn\#i;~xo_]m֥GxMB35El(QEQEQEQEQEQE s ĉH7yd$`:g`8'3QEQEQEQEjoO 롮{V׺OGFJP9VUi#dF9ji3&lR8|ϙwmPӌQSrk б'*GGa{&n"G\a_SQ@=6!I-,!$% Ђ9RS"ySJ'۾Id_q#1ur˂6H9:(xSG"vhK]E`@Hle6Iz 9i^HVFFDF%Q ^>Q(" 0NggKg,͕ʎ-s n` .fU+,r.F78/6Xm\qECui+oE9@28uc' 1I,{67^1'8(#^Үh!#/$23c,l13t?ɦNguVX)Uwi%>\`r8n@lm~&v qXuKӬh6 qBT(NcEgh~:s  +d%QW|2y"rp :2HY]y=v_ib8D>|)vh2]#i-Exg2.7L2gbPU9RUOUע*l  䑤wlv%8U(EPEPEP[;VԵ&j¢3O~6/ĢaoXTв %r 푟QX6Zfky,QL\VTSt;P3xSGX-Hc"0%ԫ 6>P r*֓cyߴAa! c0yvϾ4Juf۱L貦I"EV|sEQE((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO éV(,(((((((((((((((((((((((((((((((((+?(_ܬ?ȣ/):**hi: +@_*eizU=AhZӿ/֍;[jkhӿ/ҳn_WN_E]誟ںw?t4Yn_WN_E]誟ںw?t4Yn_WN_E]誟ںw?t4Yn_WN_E]誟ںw?t4Yn_WN_E]誟ںw?t4Yn_WN_E]誟ںw?t4Yn[C ^=kjkkPUwhQw?Pӳ!+2Š(((((((((((((((((((((((((((((((((c"\:rZ?"n?֏wTvF2 oE}]T/GQQ@)?1~My)?1~M2iI"5.;UP2I',y)?1~My)?1~M/GQQ@W6^q}{B0* $<޹[FOYKWBty)?1~My)?1~M/GQQ@)?1~My)?1~M/GP+t` E-K7Q7TTP{k.滎/il` qqc֯5A7W97OoOo𨨠 |& <& OoOo𨨠 |& <& EƒHҶ0=N>OoڕͶ_O:BXaSҤ?ȣ/(rc"\:rCq4EER?QP_7toQT_7toQPX0'^ujڦM WIȥ7PF PS]ⰿ~%mζWS|rA:h~&y&u&7*Xf=;D,l8b>+]G M&GPi1D*(u/^M7¿/iONԿy\A4 Ƚ=?U0.QE+ ɔ%n*U|dnꭃ-KO:pXdhR&u;Yx ׯeƠw6Ϋg%؃*hE,"dߞ3WHvo[Bm!B"•G;*^С-NoڠI6FzP?ms=&o &K{Xr8##ަͰ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ٶm~(Ͱ+o@ofnm§< &2qv=j K֗m[ߥ ?l?/Pn Kѿ?F5a>VK?F4oOѿMifϕ_6|fOѿMS<4o/Z_ٶm~(Ͱ+o@S<4o/F K֗m[ߥ ?l?/PT⋋y!x4p)BDca 晠[BKwXN@'ON}OKfϕ_ )7熍kK6|ٶm~(7熍hͰ+oGm[ߥ 7xh_&l?/Qa>V€3wxh_&)7/Kfϕ_ )7熍kK6|ٶm~(%lugNM="woJ+ШGzf1Zl-噮eu$p㎹ӎ_a>VK?F4oOѿMifϕ_6|fOѿMS<4o/Z_ٶm~(Ͱ+o@S<4o/F K֗m[ߥ ?l?/Pn Kѿ?F5a>VK?F4oOѿMifϕ_6|SЬ/,{#ιiP Hl[dKmUܰSq) ;jm[ߥ ?l?/Pn Kѿ?F5a>VK?F4oOѿMifϕ_6|fOѿMS<4o/Z_ٶm~(Ͱ+o@S<4o/F K֗m[ߥ ?l?/Pn KMRڮqc,zJ$ɴ&G9[ٶm~(Ͱ+o@?(_Ͱ+oY0bDDS~O Z "xTZZiI^Y#)aiO*5}^m9]%幞ռA%}jO%?/ɿmENCl4y&[æHG$]'ll<YCmnuȡ@I5)?1~My)?1~M u.YuFQ3GnҢ[߂:W%ylBTӿNkxloi<v`J_J_y'E:v9mbjI~/grBÄ\eWQ<+ sӒ})]nCt4$#t^clSӭa%Οoy$(#2z$a:)ӵOh+cU5K|;:&,Mt_ jwd[59u#I*pCB 9+ma7JK9/H7E8YS~b( W>t\пr~1\ s(1 ]Hd=T>{+"a+oLYtr0YDS^7Q7P5h$Q]Mj&!uEY}h#GiM,Jۚ 8eUҵ|& <& yky0AS5ʶ:0 ]K7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7Q7PTTJ_J_EEK7U;h|)R9>> zd,O=_ >oGϳE$p'`nrTAAnq47@>WptQ.l{v#QoSQ{peeJ֫i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟G(JnWs[h+4i_o???Ɗ(ҿ۟U5;빴庝VxPKTCPK.A OEBPS/img/wm_rdf_rules_index.gif&GIF87aR,Rڋ޼H扦ʶ L ĢH&JܥWs7GwhvŦ7fxi))SYF)gY ZIV[ zȄٛj(ʛ[1ZK[rL{ ˬ ʹ䙜}EjݷI^p8N7n\ڈѼ] <0… :|1ĉ+Z1;z2ȑ$K<29Z| 3̙4P7B:{R>DxHҕ&qt$T!SXHV,n\cy {l(aprrW֝rF^5)r $/bW[|! YTl**8w3 ѣA{$muj]YrMb$hffr!8?d)t#%o.\P Kk'ro}:~]1z\v+,ꛧOP; 8:1;@O~%_i jH c!cw؈ґD!g#=Hߊv|)kd,*O}%P9%_@U#L8p< $Y*^Q\o->mɎuZ'N{<`18b="=1םPwe+P>ך^HY(<^9hXaJiN*I 6)j-鸙l)4í4ª ;d"rd,3mt>[jb{DmhKnd ,Ǯ4޷*o{Vx.L#Z) Zn3&q+&9^qb0~6f6+m.{?ʶ0?*eg-+)~<3eOL9t/(g5]3YDkZne^l7vQ3 D٫=nyR1w{(.'K詮( =vt8mMR 5 乞p(&v3{$RJȞ'Lju6$8խ'-FK4(5W'?)Z%`$F,kW %MJb':-gHa]Ӹ*}0Ʃ?!! WAd(Zhϊ|??–Px ?֜?5OR !GC|TaS~C'C|8ƒQH?–Px ?֜?5OR !GӇzt Ʃ?!! [Ad(Zp[$??D?J]Ï(+=Fft+ٵ %Q b-,Z5Om85JeLN]W .%xr2G8 9{j;y~yҎ_0V!N2{o^G@mm}<%p}8zkK'b1apH?G-A>SیQ4>J9m=/=ڎ[ky=G/{7<N(g(q?9|پyv#AQ(TLsrͮSیQ4>J9m=/=ڎ[ky=G/{7<N(g(q?9|پyv#AQ(TLsrͮSیQ4>J9m=/=ڎ[ky=G/{7<N(g(q?9|پyv#AQ(TLsrͮSیQ4>J9m=/=ڎ[ky=G/{7<N(g(q?9|پyv#AQ(TLsrͮSیQ4>J9m=/=ڎ[ky=G/{7<N(g(q?9|پyv#AQ(TLsrͮ~:/GEqrr a_x7Uv|o -b$La@berK =_IE.l\IݝH0wOuKKm"37_߉96-:^g[M#@ЭHR-Zl6r ǥެ}:G|?DϏ|F.oY5 h6̫iph:L6Qg˵ٚ j[-?Ȼ5n>񏊼vo>.ֵOhïk.43j֯!:-BS1 X)+L E7┞,4? Miz=x? >'|S~/5Fl/_^{[@eoh5Yx{ |cџ[%H$/o xDF5޲nƝg g6zW3'^mk>mS><:z':?U/k;ᎅo.<<0.c;"SoCĿe5v/x;6'i; 5VM+PigOѼ.4 jrODSjkkǟs?_ռ-i\jzƗ+[Q ž+5lj.=~ºec~Q=q \Zx6ZtLIvыxDV]ܿ/.tZ6yéiZu_[ak7:i!е-$c](U.g7f;/30;Zs??bZs??bZs??ʏx㛑?-G bo/ٳMAO)=_̡9vG<]6궦;6&( 3F؊u])U)ʓVOf|uFSw^Y{}j; p/δV%k>:(ӭ*Swtn[-'Q0K2Z0jQO$5_yQ~bߢ.^{M]O@fta|岴Ym՝ i+m0b)_7匹odl$iIqXR[dsc*3(e M ,,%'a5ҪN]/S9YIu'[|do:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodo:sb:[[ҒM_9?>A;X\_kByǾqJq[ǖQAc91Pr9{g? jq[Jנs q=\}\+s1|c|]6*]v9;cKodqޘοl>}/71'ƞm[~#^,Լ5xKCzcX5-JDu+-wiI̬"+[a5eeD:GuA.'x_Oӿ&x/Z~3O&x讵]P 54xz+ mNm{ _ a{O+vZ߇>j> xg^/_gg?xVŔ>&Knt++GE 5m6ytO]~n>Mh񵽟gռeĿ6^KLO|#񯊼msǾmxúuޑxRMnwmvl]Cpω^,mtO&iοqR/~80>0|Dͬ.k/c燴h-< bCo`J݃X|Qs|\Ǔx;ᦣ|6.6_Gvj_~xZյ%KimO Yi0XMsmMUoOu;,gV?xS[/ G5e5 hWPtڟW.5C}7=Uoa={u >,ދ_|w~Y[Z_4x^KKj?ǁT|=y8  x >=Ưi:[\='~&k?]Bj/>GX|4[w|M跷d.MϦg O~8_~V3̓tz fZw֭Ciƞ:&6z‘;|i+>,OoҀ Q@v ]p+6i~vAh%ݻywT[sbzeoc`ו3f'&Lomt6Z\N@ nw=YbrTU0M$>oΊumʪkw.Ekwm3W}gH%Rv{alDE,ًV3 BT%wk_u^tQG}UԵ=S]Zwz+?GhN{*.&kU$c)pEuCÖS5^<ݛr]:XVn4t|F=]ln4Նp6-d=FzWa}(6CKc'o]y ӌ~4rxʬA?#/;$~ۏoƎT@$qp1G*rE|aG4r@$`Ƈ_ ʾ]!79PrEq\c5eʃ?WrE|oG*vH8G*AȐ w_]c1?9RrG1vhAGq~4rK$W<~_G*DF~hq^1?9"cs"@$a=tr$PyhK<$P w u1ʽ,_ XvQ9 zcz$W<g4r`y ӌ~4rxʬA?#/;$~ۏoƎT@$qp1G*rE|aG4r@$`Ƈ_ ʾ]!79PrEq\c5eʃ?WrE|oG*vH8G*AȐ w_]c1?9RrG1vhAGq~4rK$W<~_G*DF~hq^1?9"cs"@$a=tr$PyhK<$P w u1ʽ,_ XvQ9 zcz$W<g4r`y ӌ~4rxʬA?#/;$~ۏoƎT@$qp1G*rE|aG4r@$`Ƈ_ ʾ]!79PrEq\c5eʃ?WrE|oG*vH]#99^'|1 xaykc[ԙ͞g}iD~eX.RcfXwWμnSkV \{7sc1iok>n_|1':?YiJ36I^u<8HU֩:%NWk˷ti0($gۧn QA9#<3As&jNXAϋ/WQ_mAcX5čFzTʌ *nOGSMi}yV~oC$Emil NqpH-o#`#$gF p}k6#mrg99=N?G\2{R-<:d'9Ӟ2q$sQwK{pss?\{yX.`z2@ !x+WMgBKٍO~.oiju:}'W7mŽݯ,.mq nDu `_#x99Ǧ9'KX]3E}켄16; 9EqXL ;{SZ^P8sך6{pB׵;;mc>1.6]hiI&BZu"݋xG=na,)%Q8Q}ﭾ.~Z7Fq6H``As5|5O?㬚ZUg?#\wW]O$r}O˃醩#ɋnך ~([oZ OhNJ0wXyvM]oV,‡=~`c\UUWʟ&W" T_xc'a_&?jMȉ # xZ]UVɋ.1xt>hïq_&?z^_ˆkL^''X_Wɇ5O">.kzg>(?NqNUH?{OO9! #kao?cb??_ˆ?QN:ܟGC|W!ɇ5}?|=~ ~'ZOeWɇ5u/i/DOQ?n?a?>ՆU~[oD?D|]L>'Xe.ꯕ?Lk&?iQ{K/V˪T1c">.o?b8r}?oc=O&CGǛOH|N?Oa_&?j/D?D|\ F=|Ea NUBǯ?iˆ=~CcZ0H؏O">.kc'_?cWȇ(ݤmǧO?֜7SQmL>'?OZO=_O&C?ۏO?OaWOɫOO >|\/\~1r}VëS >|\/_l*v{OO! #~E~'ZO5O">.k /> Eo(cW_s{|5^'>ч5Bǯ?iˆ?QsOZpH?_{KۏN:ܟGч_a6#~C >._ޟ0 >0ǯ{{Z7"._~1r}V˺T0Y=~i#i~S֬-Wɇ5O"G />8w* !c_CG-~qSܟGՆu~|sz6‡ZpH?=Dᯧ^',?֜=[$?_ˆ?^?b?o?cWoD?C] c_N}a_~ .kc'XeS_‡?n?b?> >1coD</L^''\S{R5u/D?D|\L>',kp)Vc oD?D|\ǯ|N:ܟKjϺu?L?_ˆ=~C'P m?H?{M[_ˆ=Ec8k OzMȇ(Z5>enj^ȇ(kǧO?֜=[$؏ >.kzCq_V}a^ŧ!  ^'ZO=_O&CGq5v>?Cy5u/itq5/} ׷Sտȇ(1|OOi-=o{OO! # 8>(8֜75O"G׏Ox\SJOzMȇ(>'>oa5O">._~1r}F|c6#~ȇ(Z5|Eq_V/Ua]jEwz5‹Oa_&okO&Dpc GT^'r}5Xm*v+nj^_G GՅZ{:u?L?{M[_n?a?o|Sc5u/iD>.c"o?_cW!  OR?L?{O">.kc'Zp o,>.v1xt>C|Wia_~ȋ  Qa?L?ZjDsz5‹a_&/zMȇ(j;N9Oa_&?jMȇ(l3*/CZpOe[$/z^_.v0|Sc=M_‡=~qS֜75O"'(Z6'и ҕo?,p_~1r}F˚ݏ$ &ҭ=<ȇ(Z5‹OqF*vcW_rEwz|6qXe.0y5xkDsz6\UUWʟ&/z_SɿOoN1ZO5O&Dt_cXe.yS_~_{KۏN:ܟUa=u/i/D?D|]5/֬/SQj=3'P {{*݇$sz5‡Z0H?/D?D|\L^'?Oh>+֟Mȋ mܟKi/uWO*cW_rLٶ8QxwKr009s%CO/roǴҼ ;Yoyd;1w9:a1׾8MO ޾ގW\]6p{4K)(.Y. G#sЀF=FA8yO8^F9Kin :q=2Adi8dzƓkk}W7emw{oo= f)k +Nla,n뷩lۤڙ[bx@ E< !(^d}c7#RE{vvWwx--n.6x^15p!Lu׵#O~H郍ڧײ2p: ok:uׁums>|Oi_xwNo74?HmiOl0}(Hw*we#oNk8L^t8im741!tu 3|Q}- ܽ?sNAk$Q'wcuq7Pv>a%0zgd wMk8΢vH84m'o=tߩ{\8# G;O^֗tяTdD;8dú` q8;i6?=qjD8c䶫p[^Km'Rt ek K#oOkݿ%EF[][^Hci.ͥq]SX_ZC0Ym-挝K ۫V ko飍ڦS0uTZ9#hv8r3L#N^a{|?T {^9 hy pH /n]yGt>mS'?brpFGO| <Ù,1WTK-7*l~ߪeգ[BvA%}j'W<=8z1=]__7`|=mT1$c;iA}<ٵO5"O#{KC7ڟ8Ƿ}.a:61 KӞW}ޞa;3Qst۬&+[]<ͪccCo tzq_muzy5TqgKzg=9zy7ew48{˴ s5Cwo |rHo}T_AoN?=v vӐ1؎_ |5 _ _b.[i:heEݭw30wMŪqQwJ=?9率0GF]hKpwMOkc4/ Gtj {]q// ?{]?%Ii8*iڷKyGtZ{];J_wyx{NT?cF_Kȇ3<;?=7V쭮o7ڧYQ~41kĨg7ݪ`t^xix骷0{NN?=ʋ۬ ו#oN9u1q*Kޟ>ftZǦ]Dv~o4G1q*>oOO3]biQT|Kȇ3]'#oOk &ÿGtj{]q*/n k;ǵTKA?qڧxĪn/#n1T*// GяTw;FZ{]yGtяSz]*T|/o+yGtZxĨ=?9GџTGOQW!a?TuwT/wfս?.W0wM{Ĩ]^&V>?;ҡ;ujޟw;7j{J5g7a8ݪ`{]q*=ڠG]J۬&ko7ݪqQwJޞaǵьJW=Ok{uwi8tc*4-:=?9OGtj>eEw!aSkcTKAwGtZGяUs>g7g-s ӟG?_Ҿ6deq?7|ico xZ4ZK\_\Ekb/#$HGU ]6=%5~E?G=ƿxW/<1tyts͛X>m)ZNIxb[YmO[mnO/u k:nmks1FZ/uHЍk!hh΁K{ڐ c?\O<},s,۟eN1#5v1 =Yq48Zi7zrݵ̦[;,Ly<$HIےm $GTuG w}\|ίn◍n-ӭQ<5j:tR$u(-%689>L$ e{%GexzLxhIj0Vg}syq[ 3^yc8V%.Wg_5g᷎/xT^[XM*knd_zBEJ@PyqM.>ڰ*@r:k7qR)K(jxkmΔ7vkJCf&dBmo'Ӵ+(Q'YP N;ᖗecxO -4+K{#2CyŘA_E+\5s^K]'yx:wt{Fo?Nm\',Ɓ] D[#IZj_0Ogx .?>7u}P֩D[+ψnߢۨ#`[|E5_|-a .,xSxMjQKNNI0g Mbխ%zX5 |6kU G]Cᶍ⟆w,eyj^M%MJ]kXٯCW^l?3mGÈi~9~?/|Oiq$xI=JoJ G]Zw6D^vOg> ]~$'|w?^ 6?\'>>IΚcA xM'Z6-3ϬY.k߱X]ۋhgծ* * G#_#-Ҽ9Ys$+˭Abiķ lh<?h Ү>(ѧּWŪw5ʶqtg[ٵѦ5 ?CCO{{F_ k'nSI2cDh5Jim#L]I%Eu?C~?xXզWT:}$w./uOj6;?9ӤTO<=b44о~o|q_{VOǠǫO:`cϬɨ5z0oo%ypf$ _!~<&e-ʼn3p@ƅ _}?G!Ko}kS7_+O⏌~,t=k1'D}ŚzG[DG+{?xoR,Jyo@; 3=q#E<qOxԴmv}}֚eޑí~*FMpo)H)lX[_|Chh~OWBJD&|/eO([_yRMB-C K;^C&> m&tk RgA7Í&y-}_h߆Ƨqo 6'nŶz<6֮Wo}CsEm%O%BLl8im~o.|/kG-oJYSWP=an#ѦDt;k.,\k៊^5{h ~(j>&ڧOh:OQoBMS5h&f,j'|w_<3C?~"ŦǮYĒj=ukֿlƯh<;+2۽ʛc`ء;_ڼ<=wasu]OK-YMV,7, 2޽Ց_?t=z(Mf-ӵMC]=5+w/NiG0+]"u{[io;ğ 5ecOŞ֋gX[F-KL0x7当V+V(6w˻S^5,xi5?kzޣ z͠-;D[ȍA8fhc#񷊴>o տ-?]P]Vͱu\Sd׬Fsl-Xtז{i U\lǰP@P@P@P@P@P@P@P@P@P@P@P@P@P@#1M랿 w<;9O tǁ_8Vϱ nJ+JRaV lK/+-5|uxgzM xmx!oZ;in.-e"k]X]v-Gq4$gٛ981}0[`q =bTB3I?Oky%j6V9muu-J^ʳ3Y8NaާnQc_%ٙhZ|Cè\3u.>wk:}早Z8WZD\6- 7*Tb_pC=+OIo|(5hsxkYf!C:,F6fw@ [{7 4j-sz[r_[1E i0ڪD1Dff$E q Gʺ=_[s u5nCm,'u>8xD[+[fawYXV{Jα{i0LKouq.o#OV߅'O\oϤXhk-yjs5l>{_M ;dO?,i \>-ouOxo:Iom'mᶻ ¯<{iZk+j?fo$Ψ$do^ /xT?}Mӳc[o xWK O?hfKBH [+w[ o_/':W%jj.$ZΨڶozTEƹE} e.r6,~?%;knI$跚k{R M/\᱿yj72CeIT|@sh~)o)sDQIM37^5}'(7٬~0Eiܲp g|e׋?}Pב/bMNլ[Ek#ӮΪIť1˩\4|o#x_[,4{ ˩iE<1qնZokSK).uƈl|F_|u5xo,5uiu?? ZyshAr(EHeom:~xU߁]M)Wyo~ۦKybKI?Ή?5 xS(񆳬hsr&<;q[[Y"I$8?BǏ~ moSfC?I6FgZxC45׈.u=:e΅̍+%/2~hW~6'?cZ^5K}+V+k’[}eaN1^s4KL7Ё ۤv~>*}NrA{Hu榏wL=?=yM{?!?Iz=0<C54{ay_0"FG4=F1U_Ons{wH4=A*ö00{_0\' jha ?OןI|s4'?s$:i?O^SG&Hu榏wL=?=yM{?!?Iz=0<C54{ay_0\' jha ?OןI|s4'?s$:i?O^SG&Hu榏wL=?=yM{?!?Iz=0<C54{ay_0\' jha ?OןI|s4'?s$:i?O^SG&Hu榏wL=?=yM{?!?Iz=0<C54{ay_0\' jha ?OןI|s4'?s$:i?O^SG&Hu榏wL=?=yM{?!?Iz=0<C54{ay_0\' jha ?OןI|s4'?s$:i?O^SG&Hu榏wL=?=yM{?!?Iz=0<C54{ay_0\' jha ?OןI|s4'?s$:i?O^SG&Hu榏wL=?=yM{?!?Iz=0<C54{ay_0\' jha ?OןI|s4'?s$:i?O^SG&Hu榏wL=?=yM{?!?Iz=0<C54{ay_0\' jhgŸ]Ά k̂9I:wXPd 7rܜ2e 5uϽoMK߷Xc#I^6jP{Z_xdi? QiϨ?C#I^6jcK}C֯OT{_ P?%F@7mƗ=_/?4kƨ4?!j}/M_5G>kW~K'zo=/{Z_xdi? QiϨ?C#I^6jcK}C֯Oqo*aA˘i֯JVdi? ST?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧?4mƨT?{ZR4!j_#I^VjeK|?''z[=/O֧_$/O\sVā@'$G?U?$V#I^1͕ŞNq F 䵿{%#I 8ح9€|s~44j-=[.dHJ.IRҍ4G ~jy+/#I^鏱[g$ zge*{ |Ǧsu/{%n^X/O+oL<ÜRT[~#zT^VVӌ8">it^b#IfxVu`4(_W~I%ךߠdiCazemA<ϧ~x R>`_Qr?40i?C n8#9##iׂ*TP^XZOi$ /O+qdq=4{*](C_^_#I^VjeK|?'#I^1r8O=T4я$?kSO#I^VjeK|F@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?C 4'izy[#1qpx8쩭SWڢȉ/O+o5G>i?CF8˕?|(TB>T_o{˧Udi? QϚ?//O+o5G>i?CF1;0 C8i?CF@?ʗkS~O/F? ϕ <*v҅?wv:k{oAi? +su?eO|AnX_Ajt%mk~ ?4mƨT?O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? Q?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?//O+o5G>i?CF@?ʗkS~OOT{*_=O? di? QϚ?/Ϳ/GO_Y`_mWW3: ( ( ( ( ( ( ( ( ( ( (ȯwiJ]~͟ WӗϊPz_kօ?P|8t-'M^MͰO!Ww#~>e,w72[stldK \}uZRJ\We{ٝ԰>Nѫ(o.e{_ݺ~2Oow_|s_Gmu{[/j^V=6M'#`چ^h"h\1"'*a*գ.Vl<-JX].kc}?,o?iozc_F \x?ƒXҵse;}7l[^'KwUC$[cЧu%m{{SX| K IN4~ΗO^w};+>~#߅G6hw'Ğ(ҭBDnYg[4>8CFp\*GSdr`jiS̮RZkk߯o3|O|u/^?E/,f4iƽJ-Ӎ.QmgKVWR+9_kڿj?6߲5C5?54H /K7oLJ~Fib[ktqsX:UˑQo]6qaX+GSk▉];}?Zό> | o?Ʋu>߯Dޛyk|%g/_? IStou=,?4')- oH4m*'|/$--wn2ҧY4iE89oͣkW8RTcN2%KWwW5ďex~g{qax/?xTOl$f%Y%o=Zmdu;y浴.aB5)ԡϳmZi?U䜩ԧ[oK_%m|W< 7Qc/Kk^ )xB? xzZ|8{{yM1#iome]]Cq*!:r|VİXѧ8V99B7ym/sᗉ& iZw:dfyuY4wr*xR抣{tSWV/uܮ︯-˺?(~.h ~do]?š{Ok⧈+K־x_Z{?cѯ]7ڝއ-bEot+SON>{5qtJ:;s]l**э)iQU}ywQW|i ~׼Yo DZRRmt/V7DKyO [IpܭR D䢯.N_F# <+g(KV CMMuAc-/kXhpk}5Kf[K_V6Z]\;6H=֨ӝG,dhi:!J:s8dnQ-__ _tMfzޅX&Zg$7 cR՞42-F"+oto\_][OVT#;k>){UoQ_|/} jW>0mt-zUoAm..i}# Vo,Q\kb1T6:4zin睆Ĺ({Ek/~~#|G~>Xx??'MkBoo]ǂH4&ueoHo7ٸ?:r6שeMm(Qj0揳VWϧ?(O7ſ<>5J i =IϦx_O6-YJ$;٭,:aF3O^Z˕*Up5R-N[^Vt{/jP:O~xE Ro/aҵ> >ko"9QZCg{X+tW/ky5-*Uo0èӗ-/K]4No7__pxwW[Ϗ'|W|_H~!xi|jZx3Zi\V(oZl6%myQ**FRtͧ,Z?߭X?o8:JrdUX* {Ga|9~ ~ڟhxW IxV|cox =sRTSq3Ɠ.*燻%}"?('T8rNz.i+|+o ve(;C>:Oq'|75=ؾcw~֑ 6iieAdzR%nk炫N愠˲맻sNe?dw',OoDe_ʽw Ɲ5^|GI{|2|U 7kl~ j~(&/MkcM$ׇ8o-jQZٵ[+ []%?'KjQS1ۗ|ά' '7iecڞ x>.OǏi?qmI59sh5t}cvy-lmV/ Is%VtB.\R[- UхZr5&(֛ۛZo]O&,|7Y|(i).^V|6ujvWi/ٞIbLm&kNu#JPT5}Rz7DԗjFI6M]ivո _~۟ M7DI|]1X߆-~46={=W:l0>-ʆ:SJZx{%wy?OST9UՌ7V۟(տ?߇>?g G_Y~&lko֩jzum4}F*X^n\V.,e:pP\䏽}}=4Z50\gnow`8}|a C3U O)m ~2|%NO\|i|]vI7mX…E:TyrQ-݆$Ò#9i6{[?W?s"> "] _1.;rVtfӴۿ?ZӣfԞ{.ɢv>Xi*rRGaw]tCV\qS/Kg_:;ѵmO޷> 7|GM>ֹ֠Ѿ5zU྇AONʋ["wVX'OEhʿКZiB3K٧um>lি_ xwzGo|9k m<^['Uu=KX|>bu 6 MSRc{eq:ZMt:^YYJ1u)m}ݗ.j(_ ?Vc,/e:X &}GZ@P@P@P@P@P@P@P@P@P@P@i|[>-/ڷazF 4MR6ۺ%9xa Ud)qu$֚ZF*WI+ҋO['NE%~_~ÿĭ3dw֩bt}Ɨ>+־!L<'4Vfw6Mx4+m:KFi0uJ.7NɽyyTIʫMEEo7m4gGgڋZ;Ɨ妅7|%  twkgn.5X<\g7[ia/ .\:98k˟添oSԝ|4qRy7⏅ F\%?u7qKeU4Hlu4W%[iUm˦5TzJK}ҕ&u'O34OmqjZO8U^oonnmxu~*?fCkj5'G,YYԭ..,ΥCu%NRpnOvV.{nv*Zg7EYYȥt^G4l >9~ϑk~=f:]CQԋVѬ/潇\= F}6R3xux':E^zrdx%y_ʣOܞ`(?wiI&+[?G5/Wÿ5-:T[|K D~2Aas=E<%L㾳G›\~)I~4~YVQGO+ϒ>Ѹ+CM>?ÏK^ex@KBXjַgIn4ymt]Q,OI+4.>{鰰=?QiFrF|V鶻5C> |K//?":M~w< 7 QQ&!{F׷.u5R#rWwmo&X_|N|kgu#`.پ&?-$'C/VҝJWRRKySkrkkoz?_c⇅%}jVtk_ìimM:_`7?[msMt䁯"~upu# IJ6Omn\:)b*fGq^;0~?< mj N}mǁ|AMKQѯmxALj:YgšfRXKn &.``vӛfqr1p['U)7f᫽ӕ| fS^|UӼqc YN_ZVtj׺%"u#_L.3hz]պ٢)%q:ؚ4|~{)'~ץթJU+]ϗ~]xYd_|W |_~;j/ xVT|!Xx:ߋWᬰ Z5B@#y#h. X}Rup5+Ӵe&OYj◹깼oNxʕ4fۄRRZ$ty|G1?/<k @5m.ok&ÿ/5-z|?>:" iU-6*QQӍ^^:bZ 4BQqo6]#t/~# x*:d\,φo;2l&~!VlۦO-SJՅ)77zq{O$VӮYsѝX6kZ+G텩WeBuŕ#ƾ *N5ӡ|KwaxX.t/U񮮰X[hmnl3u\&F4yM쫹;kUKU>]!敔U E(RJ*ii{*pX§/ԣs{yAI~9~X> &Njt x\˭wQ4iɢp[ƙ}ogӉ|*QQE4VMwc MEVR,5+_}6uoAj-R/bԬ6ǩi|%qzZM" `pM rF/=mڬm2<2e}SRFLO+͘T;d`ƕvҎ#_7+k>߹U:[D炙crG?5G~z=eTk骽o_Em5Kn4}iqy  | V8:g<]aj>ޖ]eoDŽ__kkUZ_ۿ dRnp< S;3qV|UҺn%g I|swK*U_y]?~<췓~'5Yݳt\謝kk35tҕ^V_{|;k7kƓQ[ï>*iz|7OH<iS75+⭖M4MSF{l׉E/jSԝ:vX{I{OJgbR U(N4W7v4ꝗGC/?/|Q]_B~+6P𯁼U+?Vmg>iYMU,lS\`\\fQ*8IJ6./uݻ%uknre5qj2r6(khC>35(cſ |K GF>O6n*@ 8Fx-cRSkr4]Q\T省z_: ucO ]ڏ|+6EFՂxv}bDbզRﯪZ470i^ZYuI-aF wmtwS+Ԋ+*ܲ4ehU+cZbOZD6_ڷ5ߊ7$WV.ׄ5Oh2 Ce[[e&j/T5 aԱ8){J7sm:/uti ._mʠӓ#.h͟jR:|*nֽ_#UE\/8(_Ow# </u]gIտ=+Z4K j/So"{7Wl:5X5(kgkKN[suJ3Өgeu.[it;oiu;F7FD5x!G]YVO2nxl!}nD.4 osmiZlƝy*Zk{~{& b2鸺-?_Cs x?i{OZu42^$V[ϥjxNWqK8DXF)E”#%}1cx\>JpiW}O~q8r:Q?{W .eye[k^KY:?Ӄ8FLc{Yw%絭ow<|%VwOSj8 ( ( ( ( (>\? =@|=wcET_e_^\.Youw"Ie_kz OZ0xwN4/hwI4ĵ-|*UZWBiv\GTjkMys>ne5.Eo3߃/?> 'ScmnˣJ$vn8gۮs eJKKF6Io;ޭv{u[q،`u3ǧs]Vv+%QKV L8wN髾;]+E>Z3C߿gzNxedOk|qg`;t'0#6^zvZm}O)~'ƯS[R-#ÓC:xB:->ݨN:J44sG\#^eWtOfOS\rn)b=J0iT4?wh q1 ;[ykM: z`w qzQi86-Vf3';`ccKM_sY-RVq9'I;4kϱ9^}~>Kݯ먙ͧ9<8䞦mfeeMem{ac}/˵koݻ~}t~];_m7 ;|k\ӯSۧr-4VIhuoZ/;oukӶ1c9\w_xg`ppO#0)YvZ$]pZ|}:iivZm&lzq8 Gy@//u[y6V[uo`Lz9lcA A? pp{g@q( (zqzt< ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.})W'!70kֺ{['д;UlE~G,"HfԵK(V;nXޞݱЂs␶Zok>19~Rzg_7!?4~+ZJO}jUm^FWU9Z^_lů|D-o |J}CD?2OTK?<>ևIj"͗jڕV+ߴgX|5ǚ~_ յj[y  '67zl-wZӼw ^EjwVEրr?xUu֬&v>u O9]oHq(FI4/xFg(ok/? ii>/GF4/Fn/?$MgH#!y+],u9? kOo[.h'Rp~e3NﯼAVX!д;CXGM/O u+Wv>Ѭ[}_>xⷎxTk?ߏ>xkǺGuo\A(𭗈t^)u} ڤĢ5{E+cg;/m? w ]_Ě}Wu-AiVwڎR^ahvM|t/xO##Pմ⟆<j$wo [\[sC𤺗G!_XRX0x&[Y"9_^@ -4o |cυ;§𿈿>Kwxc^uLj`}ZSmwui57˧gK]ڕoz^+Gs-摣EkFDC`qZж˥W< w>^tm2⷇};I/7ck4WM~Xj;w^ůx\<K~/G]KQ#o:5KI_KM:ƯZv/u+ԵMnnH&6 O"~ ?'  ?Ga1[5I#>Ko+ P<'xS?@ µ)п 7SIH__Zhvwl&%mo__:6.OuW?<9T  ZgWվߣŗY\]5`k':׍~"|"y'\߅>$ae[<WBW[ҦAⴸGkq-G}|tkbAN^>ּGIx>__:-oΕUF-]KݱCZ> } s.xKῊn5]3m.M>C/Σ4=)ٵ{u6Ko++w:/x|3Rp~%Lf*}!/"h>.MkmfWcJ6Mt73_"}M ïYk]/W=kOj6Wmoq-ƕyՄӉ㏎ ~xK>5մ ?70[OlyK῅_Ce[[\[I4ӬB_+OP ڧ=^+,Z8\i?xVzi[›`-K3#9&~ xE)X_|WotUS}}xl?#ĚH+|=!@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@we ;k&ZE \g= [:_,# Ծ[&ߎ|'FL+FnZ|b*׃5_ |#SZ| ̈́=π-w6's>r[[Ye#av?g7c|5g+YiZïzkῌ=6hƫw3 <^>%V޴v- Irm2]mcCtH"5/HucRNJ~ؗڍ@?= jڽ~FoދOc{UAGze_Kֽ'[x>4-:~u]fO>kՇ56G[hfд]5mj*CLOaY{m~2//'n5׿f|oz "O=i.үnF!%V~1<3sU|A}hG\D_\KZƝYO uK 1ӍN3Oo>~c"Mω>!iwH !{CWjEi^-CG6+S|yy࿅ |g௄I|a1uD<%mj hk[Mh ܎^Il"zw.=;> <-⯍z|&~3㳚-4/_ >Хk^m#ksfQ,[ۦV 蕭`Oxa o&{.ſPJ_.w[ZZ8P\ /y__*_}hs~iCKik1&}T|o|X/ |kKau1|:~?x⻟:¯[ҿgxN?|*յk'MK#I|lW2/8桠I~#r]j ib~~1$TS^g.n|:vlm$m ^?Obu~*w?.u_&~> ~U}k3Na<=hF=œؾem[xS?fj,ǟf 8`[$V3L_}qWُu6:^,^~/+\i~ZW_ږ2YEA oGV2=;Ox[P.?L~̺ 6P)៾}JOVk߰܏i:|r!gK[-z5> xNiojTkfwC.k}0LecV[o5Ay\ˣ i♮@j'DS]=Ki]Xh2xA;wq$)w|~%!?~=}2?uk(xN5KK~ 7OϧkvMq/%;.ſp%q|ߏn^ikxkKp\N>e?)֟iֿ>|DOw[^K? 'ǤxN  Z^ɪjZM略Ro|uxO//| n>Mɼ}x2t,'~h/LJ> gۿw %O/W6H??gήFA?dK#]-G7|7Ү?kχIo?izmi*2V'~ 9-xf}žlZ7>)j>2SLM G7"ĽGVC6FKZxُu6:^,^~/+\i~ZW_ږ2YEA oGXeo;Iྗ|%|1g9'-?_|UKҵO6akW~x7_H7IT-oa6)j^Dφ_<+_x?;Uk|EMWěr~w0IA>$qZ?ei? ~'|5.w&պ> "״"$I= ijF R+N=Oƞ&.z׊,<%?h-UQΟ=jglⳕ/o5=I|>7ۥyk7Nxῄ!/?< 7>-ZJaRa?n/ScwǯOuٔ!>8xԵ :_^CPگ`}~#MDM`/qiZg~߲'ku} I|=_'tf(2A{xBbguJM"oAIٵ=IP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|?H>#^}u-?"ˆ.3xpz9n[N9wL |OgsI6ˁ=8 o 'קA3F ӱ錜rONހ 99 vtF{<9ܜ`rOO3}z38>ӯ덹^qӶ1|pGך`CӦA888<翯l;c9^zs`ϱ=1ӧ>NO΀19vN@;1zt' o@sq;p@:``g~IAo ؎N}s9'~h v#(? NvӯOn ⍃`؎`zudQl1 #߂qcNǠL|r<{ z+[zp~SӰg3L6Vç$ hAh4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~??ƀ@O?h4~?~{Iq~GSsϟY[AKm, 602%BZa2o,-?"˅.3ʹ>$GAO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽdAo@>$GO?G6}O~ͽ|o583z~:tɿ@_ϼk玠 ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( :c? `q ( ( ( ( ( ( ( ( ( ( (>$y_Cd OHrΉ5PP@P@LpO^@*{{zԎ>j߁ׂT! ?'8׊=t|| qǰϮ9n@#s89:{c=rr@#yA9N9 \|4toAa[mf{{X0БLZ.h`|y-ᇆ>x־(|lռuoNG~/jz^_iǓcewxIQ}?T-6#? dls{P{}~-szG%|lo/4o | ?J={OY߇g-+^#[kiU;yq%m7K/_B>ZGk;е_O-QZx!`OjEσh/x[Vum&B,T_R6xOr};,7b]Ew[ҵ;/x*(<5|Y ]}V/.oص}FZ 4x"j!$|9w}-[oWV >1|_ /6 -3_"_x73|"mRϸ?|% K@s> /6 -3_"_x73v>[]GBƇxoCZ6hMtˑkiCyFݲň}ou( ( ( ( ( ( ( (şccoT?N[mgZ;-cBԯ4R=ٵ ͳH} qVUІ.N!V0Úg?8l6CV՝ %HNp(EДdUP>>vd{/<qzoCyֿU]Zlf6ƻY>Q5`Ʋpszoˮe,-ijV_K7}ji^IJ/N٭L?|vk7?qG!˿ g77 a _qaˊ?վ]XoV9 qU[_ZÍ\Qr"kªEooCoC\oVo-?x|=B?s&;i[EaU0Oů80Ї.,7+*ɋ k`OL(V}[2|nQq迵qXy_g3 +\ dp3@"`I],mhߗKEln4[Uߞ+Yn*wI _qaˊ?վ]XoV/*ɇ/Z~-.(V{9waXl5UoL__֟C_:׌_e'o?]zhWZj,6c[JXwKWqӸ?Z_.P8@N2N)|=BDDžZk{^vڲlg q*|}}[g1t8gKw,VkMn%euvkp'tuĕm_giY?|vZ,/1 q3_ñ+఩Y޼v˷mnbZܼ۹+/k]j1X02?>2(\7`e*ڼ[*oWͶXq5ުmY~:_1/3]c$ g {OA˭Ѭfg쭴oiY,7ٶ5iNN> _qaˊ?վ]XoV9 qU[_ZÍ\Qr"kªEooCoC\oVo-?x|=B?s&;i[EaU0Oů80Ї.,7+*ɇ/Z~-.(V{9waXl5UoL?|vk7?qG!˿ g77 a _qaˊ?վ]XoV9 qU[_ZÍ\Qr"kªEooCoC\oVo-?x|=B?s&;i[EaU0Oů80Ї.,7+*ɇ/Z~-.(V{9waXl5UoL?|vk7?qG!˿ g77 a _qaˊ?վ]XoV9 qU[_ZÍ\Qr"kªEooCoC\oVo-?x|=B?s&;i[EaU0Oů80Ї.,7+*ɇ/Z~-.(V{9waXl5UoL?|vk7?qG!˿ g77 a _qaˊ?վ]XoV9 qU[_ZÍ\Qr"kª ?H>sLi( ( %6B\ZS%)4/wA.1WM>Si,ROom9{w|E%C4 ŮcXдQ1,|eLŤ]Hnc|F +H~Cil_"jۭX_f]I.`oo\| I_j6OLx|i/č--K}/u6Ie GOL"%6/1xCN|;c+=ſo.w)G?Ƒ~#@?14~#@?14~#@?14~#@?14~#@ZzljxtMuRZ9t#MZdP@P@P@P@P@P@P@C xm < XrW Wy%Qs?>s̿P?i32m'wz/| w&У~Z5W_DԥՠԍŨ&[Y6@牭r Uc1U+Ѝ VoSJ ,U,MMr~jS*~E x<^:ЭUՅ8u/B)ahMNp])k>"׼3[OooS^WW!Z5Z--}bqՌ/UFzbjaࡉ"Ic%R\JN4/IFU%9^&aWBJN4hBFsx?*Rt\ܭWS%#sOѾ"/l~8|SV|D~"b-[^IE[It}RkbG<#:9cgغs<8/lpr9a3ϟ U[PQ.~y*~#%ּ)bㆧbԡTfZQsӍ))O[*^;xo '_OO;y_ៃ&JV}/T[ SW|}i5U?|cj> ^*SWmxNIt-n{Il=Nx ʥzxG :4o V^΅Hի:u*M8NUi'XSBѫ9b0Rq8Sj!U\gR2SRg;-=w𥿵ҿ'ğ/Пh?7 Կ_lg9amCêP_=yQ-;K.^G\ѯP_\4aj*}jSI e3#'}4ޫtxKl<7]:BӴ[:Ny X𿃾&^Gu!foH#L5x=:Tĺψ C7^lEG\'|HfX1Z0)5:FХU G,M,<'VaqS"^jaUJwyԕ)T\QhZχ| ꗃ_]?W۟1uhvk`~+G#2 s=Ǫj:"i3XNu>x{,*Spyv])# b?% PTn-ƺ҅,v.R'G nvx+?-KѬ~'xKӼ=uIa6xFSf6m%Z%aߔVcZ0*WpBJst8Rq\*XLJ%:X\]hS9PiǒmMMH(){EQu75߉ gé4?ޥ\xZuo7WEPCA:7E}-BX:)TXEU1og!Ssw2藰[?<'w͆xoÞ2wj3xkZZŮC{Zwvv>"Al]^WieScR:)R*sICEƤPjsgR2k xXԧ*tjRTHӝTRI1mIBRT xźVv֚Nea4VN"þlMi^aWw%Kywsr.<%<8zs5T7ʕkTzdEJyԫ%Nw p5XF8ׄTcF"iA$N0QsfC/ e?[x6-Mzff}Jnkq6R'NlI<6pZy|8Z.e+>5m8㉜hѫ8'9hN'W4ua N6*5 Ҥxb%>Wta)ԧeN adk}kLxK~*8 xO%$fYuǹ|Ayi뻝EkYDUKi/"񖯥kaII.O˥vi)ki^j:|[{y ڎMC O*zR<2TU*|*NU*_F3eԞ=եNQՖ"pVԔ*ui8*rFܗ|KniQYeiAuUV%Ka=SKE{=GMPn`cp W*RiTGFdBSUhԫM\6"xJѯI/kQBNԋM8֣uRDӅhƢWV.i3t ?oZxIΗmeje录kiriYSKAKi\؉b\R0J9¥9Jtj\Z'9z:U|2F#KӥRgzisr9FVyKOωu[߅_xWu?WWT]WQ?K]?Z'8[,REvOW,,_ 4?g ^`)V',$T(G9\Ռi^"-,F&r:xSsQͪ.]|!g;u-g~(ߴH].J%0 |1^I}>:9Tr׎"3,ㆡ=L=9rwQZIG"uiOS([NjnjTSU9qt#Ǹ‹Vq -Y=[ {(g@Qo1?_՟c~E?Ǹ‹Vq -Y=[ {(g@Qo1?_՟c~E?Ǹ‹Vq -Y=[ {(g@Qo1?_՟c~E?Ǹ‹Ve ;kOο2oP?p( ( ( ( (32z}ጂ@d ( ( ( ( ( ( ( ( ( ( py8(gTB{ωn{ ?_m糶!ִv_^t2\Ϋz6mȳ6g:5Ss(׼p4<l yԧF'4?Pג>5i| |%}E~"<bO&o C_ş?J?++a??_k GE~"<b?q63;KO {a(_O5N7`i| |%Pro /?5Y"]?MA8C'?=0W/˿V'Aɿ'h0ôgE@yw X97mfv,Q_.Xk&o C_ş?J?++a??_k GE~"<b?q63;KO {a(_O5N7`i| |%Pro /?5Y"]?MA8C'?=0W/˿V'Aɿ'h0ôgE@yw X97mfv,Q_.Xk&o C_ş?J?++a??_k GE~"<b?q63;KO {a(_O5N7`i| |%Pro /?5Y"]?MA8C'?=0W/˿V'Aɿ'h0ôgE@yw X97mfv,Q_.Xk&o C_ş?J?++a??_k GE~"<b?q63;KO {a(_O5N7`i| |%Pro /?5Y"]?MA8C'?=0W/˿V'Aɿ'h0ôgE@yw X97mfv,Q_.Xk&o C_ş?J?++a??_k GE~"<b?q63;KO {a(_O5N7`i| |%Pro /?5Y"]?MA8C'?=0W/˿V'Aɿ'h0ôgE@yw X97mfv,Q_.Xk&o C_ş?J?++a??_k GE~"<b?q63;KO {a(_O5N7`i| |%Pro n~_fο2oP> \g>: ( (>t45R ᷇<_kuWdҮ>x\uJI}3PˬVOV5tKKJ_-=;^¾)| >ߍ|Ht0{cRF|?j1_m籷5}#eSk |V]_h?ďxkƾ#>~h)׆Ϥiؾuo_j |d=Ȧ!Y~?j.9_ '#X8w xLjIDj?yǐW偰_ůCԾ!N\->뗑[ZǨKYY8=滰ym o$k_>>#xMDeEneޫ$%+p6w{P 3$xE/|?i煼;j 6]jҵo{YΌZZg$zro ~kxW/Q'>(׿t# VA22)I>Ww;ď&@|;ún[OxoO/5uO).L>(em̧>WR$YþNG>,߇0xH4X]w9UEz8;.G@_٘c#0sV] f_S2?gse G"/?/dO`7lMSc8C9:׊,ʗڵ fS|T:z dK_x}N13Б#r .?l0_٘coc9RAK.$sdfz ?9Nh]mo Wڵ#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,xǗ,D^6ڔ\n)_ ~/~η__;ߴ֒?|+{;O_#tzf:mCqi,VLѬִ׷y}~|X |F|5__;xv%׉o)l2\x~Z_*/jڌFH}Em-/M6qyX|_4ž"k=ω>"5?+sQޭuCojZwo+)n 4*[}u;Wo;Xmψ xc>f_<+=_vYg4Eei~-%iM.}fq޼Vkytu/Co>%O~տ /?|>_&Ğ5Q^^ x%oeI4A~CM-/O^$<}&=a]| {5k^ K =_|sg3#}:^I5~áʷ$G?]1O>|g?we¡)OXw??&8 Ku?/ş\xO.~߲O5^>1c/qk!bwW߆u$WxKG'VݗAc1Eǎt"_}%?uXv ūM]`<%gxFѶI{Vm}-co<]77OV'ö~w<xs gZ7?|Ivs8|#kl<]˪ke^Ct ֵߊ|FZ~̟޹4ֳ𷋵sUyqi t-{{M:&ܡkdܧ!  O|lrpH ^>g:v6 oG|'X"VKe;ق)ml[_jC9Tz/+K9?>m?àxO3_w~!h=a B_y?:|[5l,,mg -<ϵH<⿅_q?~|5KBKg_WòHkڅkX~;֑DG}c[:{X.y5lhxSBNeuoh+ _j>þ-QMD ΅6k{H47Wfw5ok;{oe}>kt/6OOz3Zh?| ]Vռ!u~Os\x7Ěԇ>"w,˄՝Ke[-t>&ԡ_!;m=4iқAm>{;yOmI_M+}CQ|1F|HUFYP-|-x(Y?~>՝͟DGQ5 ~ƚíW^u-G]z[z,cb%ٜ_i}{+v;:W -?enhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/A+vx2nhҿaih/s>5O j HnlumC_戻_|b8{!Pq@P@P@P@P@P@P@P@P@P@P@P@|?H>:tɿ@_ϼk玠 ( ( ( ( 1lqttt8ӶsOE`aQ? /]E(w`~_0‹WQ? /]E(w`~_0‹WQ? /]E(w`~_0‹WQ? /]E(w`~_0‹WQ? /]E(w`~_0‹Wg=3$y_Cd OHrΉ5PP@P@P@P@P@=N>|m` .Qu]w](Ep)a%ΕjVvZ暗VYڍ}uf`5h,ݢky{yX=?8gnAl]w](Ep .QuS,,/'ծO`S#uY,4Z47Rh0ӮfǗ]2~V.PHP@P@P@P@P@P@p,/[|B ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@P@P@P@P@xEK Y玜#MWjo)Qѯ#Gwןblήou{uqs{sm^jΡX7׷vpOsqOȓnowLg~4ekhO i sm⛛ Y SMxĖqk6d_fj7vz|..\UԹ_0v$+ 1"/uv,4}Qͭ3ܴVVJ!Y\& *Ri+._ R[[ K`ϦhMs Zi:J^֟hM=>\Snװls>-jugjz[Tҭ0>i&+(|W}Z]Nƻ[Kg5QoWmt/1}wo|3_m}uꚏ+ |]7O45@m ICq{z%4]Ǧ_<9C{x+;+W)w-Fต9ϋ|;r Kp^KvK$3o_69ƞ5!+#}NM?*K4Z:lGus%6>:~=>\b(ێJP=I'b%]ְ?4S` rIZ۱LZ~ |#{k_|N[HxQE|}i /&mt~wAC=J_}C=ӧx13QOo?Qא omoP d9#|jY|VNZggN&e]xF9Կ^6ngզ!I.mG1|tԠ~ |[{_ |NomZ:xQEӼ}w ;?j߇R\WM l927 G~5/uh:Fc^?h4{]cV>^ lֿ c~᫱c-cKdz̫-u cOKƞ%<4>|M#:5|$}SH񝥝xo⮯_}EruI.s?gkZq|M{~߿qT?g? SO\`s' kCy U}[P@P@P@P@P@P|1U*( ( ( ( ( ( ( ( ( ( ( (c/G3^W_7(o?|P@P@P@P@P@8Υc pq ;QWJs[*:5|owúlj<Ե-Ɵ K[yƟc>(o;˿S c{] xCU;5Sx[5ߌF]GN5隣nunHZi&_fi? Y}⎗gSŃcއKO |j]?Zľ-a&-gѼUMRzΦ2ReuM輛'}&_1xgM5 u=ŝ&MƈͧY\˪i sGii52OJ6݊vm8 ߆?:σMcK_ ~+x O#ݣX4 IW/c{jW_:nVjncOH岾HTI#xniF#%B/8q饥8%Љ-=\:M@A0s=Fp3sM>iy[**shW^13'4z׌%x/oZx]~474 XnKW֕jZC{b7uMWNYBKG߇__@b >"E_i~ţ,G ~3_c,LxCSj_ ^=su6ΕeD<+OxRx}k'G[:VgבT^f;Oi| x{uqYsoχ^}xQ3L5ދׇRmQl4rQwQw=ƛx핶kFtit~$4E[R牞??L--# W+4f~'wx8OƘnއݕP@P@P@P@P@P@p,/[|B ( ( ( ( ( ( ( ( ( ( ( ;2T|5}u-?"ˆ.3xA@P@P@P@P@wI>xNwHMgYt{M>9d4 Ro 1[L]<h j*IJ./K.W h?zK$pGAqOnS﷑\뷐գ\?g0opGNϧ~VM'54L}p.nգaϧ2'MŸv@j0 0I 뷐տ T#pqx—w/kuZy1FNFx0).OOD9뷓@j0c<] A 97"aOku>|1OĆ2^xO}aci[_mۭԑȏ*c7F۾C?ڷ=qA9&]_*eCɋ[>ק;A1t$'4`]?ڷǁ8'GP^šdBV?-1gBq1Oÿ뷐V8>@?uz{ ks[AZ8wą^AE/{Ys[L_@j.ߦ?.x'$t$îOaMus[ɉZ?z<]`gNzy&뷓'/_xCh__ c,۾˽J]yȣ̫WFV际P@P@P@P@P@P|1U*( ( ( ( ( ( ( ( ( ( ( (c/G3^W_7(o?|P@P@P@P@P@?O0l=A@ ( (zoLq!( ( (oL801P@P@P@P@P@P@C xm ;( ( ( ( ( ( ( ( ( ( ( (QAwLו9M`.}_c>YR'֬<{KmcLF," nF`أ''''''''''''''''''''''''''''''''''''''''''''''߱d+sLi\7qƾx ( :{up~.h鎙 gԁdg?@B9٠Ǡ9g׃aO'9v1_:~S@sXcp0HL rO$Bʒ0Ӟm A#xr#9 @P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P߱d+sLi\7qƾx ( ⯋\\Cm= 3Vе>VZG¿mo|ki#5+<) |0~&}a TΈI~ eo[?z|Et UO׼͢.yXi[P.a[Ol&f쥂r B(,h'vaCMm_toiw>{KB]NI|iǿ5D [i#>"\ԭ,vP2Xwr$"[Y ,|oÿ?Lq*V߂7¿t_Χ^׵7SWҾw:]Ŵ2%eG|DCE|gٟxC_u|E6؞9Ӽ4[r>_n4uY"]"Ka$-= ?|' kǂ<{g?43xO j^ oj4A vx7V݅bT -: ca5߶C<⿆wc!w5'Ķ^4#Z]xYmS?jc6OԭbY.m+er~kt=6?g"/QXxk_^1)U|XYL񭿄|Kx_>chVڕ"IGȤ쭱\/_iMi^>vY~(/hVv-$F#M/<6 H@տPmLj%񷆵 ^yo_xAŢk^&>;/-m4|KiΜxsIN6V%Im ,gg?,MM-h o,[[K; '>9doM;\jVVǶ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (}qpz}2Hp ;2T|5}u-?"ˆ.3xA@P@>e¾*Q࿆_oωZ,9lt?E7 7odfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,}sogPNy] #7˿]G@_ٛ _Qe .Nx>'0Hifmwٴr "lS–KY.y>dfŸ2TYw˿'l0_٘`ω)dB\܂˿]O@oٛ~I9]l~ p|"ldK.Yw?8/1S7>'H<9 _~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿ifa?fF?Rv@`˿]G@_ٛ _Qe .}/?/dO`0S #,88K9zAK.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]?lSRJ.YwGfo))%E,#7˿]G@_ٛ _Qe .}/?/d>dfŸ2TYw˿h? 3OK*,e~̕]9[p;RO&jX\hP%G׶i#Gs"h$tE[Mm+2 c6aVn\onScHt6SKׅ5i/~èꚄ׮}g8ddA5{-~_fο2oP9p( (o<~??=|&9y$9  O\~KL (< 1|5?4z>گ+ {B< {gyy _UHö5YTM'ZM~%9a]u['Sú5=;D{|ʷ `+_6Q?5KIaռGU? ij_|3VPM A֯f|+-ciztK|wQh~.M?'s^w5xs\/.oV?>-<c#mN;;Xц'U$j>%'5 u NW(h05 E׾٠gRm/V7}IY-b^I_~>|AO \-gW+~xFj_&mUn~ɛEtmzr-I\|G-Kw/xCsh14'^(?˟>.|[}]2Ucx[ϱot[pj Jxo/߀~xoiYyZQힸci͞a֞h^^\y[şH'S& HӴ}Că> wĭk 3cH5?<)g/LkWVzeվ@>IK=4M1;K l/i?W2o|;@noxzxJ> 3h=E8C^:s/o {2xź}g ľ.q%%Mק5y-a[hx3eKo]Z8~_>+xCV7.?O[?}CZ$? h6l%%>ɴ{*]m`q Z$gtw=Ʃ=O ߆:c7QI5OXQW:O<2!tl;Eu|9@h! wx-/|Sa|sxPoVi߳s er[㼿"E;wFmG&=Ý/B Ú|`sm6jqž5a87#Q/><1Tx@Fîx]wECF񆉯_.S>[izVMv?oK$?j-o⏇x3:?;T?o~1IxŚxyvk'Ȟb.]v=s5H_+PH)kt9uo&cx?;Ԧ toi)Ú:֜.&~G$( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (/\DīPF?&sr:)/rEGu|U~Ѿ s~+CZ5Mj Wf . tkO:]GVxe)D0JJN)u#EU#~>, eߊOx>)\AsV^D-n.^$hn"X5R7I{$ky\|[|x|R=?ðrNON8 '=Q!˧m~_fο2oP9p( (o?p???SkTˎ>@P@p?t;o/5\xG ]#_BѮ{{?Qgee!oIm]&n"CO:OtxPgV?uU AuW? ؖ>'#_YI6 ְx;D|}_~,C5ǂl>%]|;:6|1'kw<;k>!&$!5hn? mN+K#o!9ikX> ^uߋGV:}7RL:O xn#|, W6).UbrV Z~!OkC>#: 6!ᑬymՇr/.o݄Wc-ை#_Buou(9}-/Ve[Ph6vӴRi 4x?_ V6 {2-S0x^MF-5a4q6x o|M_~i65 Wj>kV^>}7|Adҵ=~@3}.m> ^uߋGV:}7RL:O xn#|, W6).UbrV Z ^9wl'`V <jKG"mA[X K]jnڥdX奭c)|ɠh?e»㟌P|^(M/Pƽa [=K<j_=lo'7{b7& nO *iPx~0׉|Gou~ՅLju}rTSN=jxEc+ ֵD5{|Z~-߉ڵ爴Tvg<7+^m|, 6W+\[&{[kJy9=<`r'ox·?KK>&؟4yxkw%R]t/=nѼ;\U}Hx6 XP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Pol xP.,,4^[;;;[nguQyd*X*D% Qѯ#W s70|L=[dFRy c*j_ލ6{k̜ rZӞcJHٴyWAOXAjK}KH=A%ʹrJ\WU°!zbگ+ {B< {gyy _UHö5Yj~g kú=Bzww6zv!Z{I{aypZVc=? _?Wafxhjo :d<+B -ߌ <}F¿ S_3m7k^0ּF_>^mEU7z":4Fu4> Q_x{s  |G-xG|=W:t w6oþU%صio8~?/=?9Q^_?#ռ{ciΫS:J7*JZuɷyVkBkÞteu7oX귳j=R+ !Ҋx݆Z.kꗩiEa(|/ |YΉ_ _hԏu?Kk;G>Сִv~/DP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@ouڅ-vl2o+mm';|7qEN#[}.y יhKpۢifowdu>z>--nO_.w8Rۈ(&ܧ2sKyX^^Zߵc/G3^W_7(o?|P@7ş8?eiL ( (sxySZ>yo ->upG;+/ Kj6t},g)|kxڇ?joj fҼ cNıH?MՕɷnWV%ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Yy KZƆjEZ?'j~!4=OdtЯoxKq+e`5/"O.pn>[^=/Gsx[Q!|6c;E/"V oDk Gω<Oz,.KZlWVh]:ƞ!4NRKoOlV+Z8fA|/L|:㯍 At kl5!΃u ]ٷ07I`t> #^>Gx<-ռcj^6+:JI/6U6ڍ9ikX<?|G} - ޾|}گt]KҼQĞ"-In۫k[ukXO|zoOň~MīwF5_Mu7LJ|AxG>$d:ƭ $W0Mi}dmZZ4<jLJtf;7RMqV\?<9oXv"Z]^Pv/u%V--kw~ԼCoQOkw!G&r[- M@S/=úUψuW=/EM}@ (8`띧$| ( (1{0Hn@@Lv=1g<z 6' ``P@crsے?/dN;ppx=׀;u᳞x=H| ( z Lu@R:ڀ ( :{t^08'nh (CGC{PۧFzsnqȟ9>{v |϶ߙ}{mAm7o7o=oͿxv~lJov>7|o/O?7ߪ ;2T|5}u-?"ˆ.3xA@P@3|YaZ\v" ( (ȼ/jc]#O-b״!im|wuY4{ <;l^rӽU|?|S'N/L`:j?Nlåxm}쓭A͞tKb Ǖox?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzZVUǧxw(O&?|79Fj;i~#.|A{gG}gsghu;k+7=Hv4@!o|uOɦxWA[ZxCDž~xƿ |g.oJּaxo|>ڊ.n,9< k>-N Moh|Z׎<gƑjZxS_ڙ֮|ť_/o_m~sž.SP׬>_ VmWǾ>*_Cea4d:QOx3Bu+E-tR-(7U_WO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo ǔ~~_>+xCV7.?O[?}CZ$? h6l%%>ɴ{*]m`q Z$gtw>5OG~ xo~1ɤxŚxvk'Ȟbm^6<)9 zQMsn@of|{-?VOC4+ g[Ҵ]bM/Rҋ[ʷ iŸI/>so-{v]j~"tů-/\i6ƥo5a8򜇁l~">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿>Y]j)|  :گ||Tit0fwazVZ麥ZQkyXn<}N폅K7\~gs}Iҵ< ;|_ˆY_j_m2]xTVmm?'[yT+Y~!U{?w WQ'}},i|eǦx?F_?|5}>{Oi* 'E\<%KY_y[_&m5J;`jr?hSIJijxo?֞>#q_^)Kҵk^#[|E6n=GFOvM#`q=v?j>zcx7 RmSǞ*մ?xnx.7RK 6,6[K9|En-R}K௏n^]WZ߈!{k1k>w5>~ [c XyN[ώtOZuϋF|_[i>A;V1jVvl5=2j qizh >)|xw&?tU| 5 haҼs} I f:Mk|ʷ> iŸI/>so-{v]j~"tů-/\i6ƥo5a8|~4/~3^|;<-L.oaڧYmE-@'oj(x >6x/QGxk.I~!{kǾWi"z\K(Oka5ˡ:c0FsKomcv__E|+;@Eo# xY77so-{v]j~"tů-/\i6ƥo5`q|En-R}K௏n^]WZ߈!{k1k>w5>~ [c XyJ?iT4<;K_ l/iÛ~[xź'|W4 zLh$1MB5`q^x{ǿx /'$Oصo{oKݸ-$1xI/?On-l⿌ m<ˋxu#g÷}i. =%4ok''H?=/0Þ5.-tCuk? &Xov4@;Cq`]xTVmm?'[yT+Y~!U{?w WQ'}},i|}ÿv=SZωZ׌&o=VMSڧ=NAjz;olM?{OଚO \xS4WF[`K>go3M m-}B'ŝC῏.^🍼o>oxT{o3Nk_,w/M vkV}-/Nͼ E$}{cr1FH<@h/|>(G=y| [V{:tOm+߇|m*KKj[i/oN<}e_wEi6#xΗx?|Es@0|HXX]tWTյ8bF80t{Gյ_ ._~30煵K 4mo#6/m zW 3VK[wYg'rE_xsƶeż:wó>ga 犍%x]w>/>!ki_Qw&}߅^S]|9_jPӴ_i>5WP|B"6Ҋ~`=x>W/G#xG:'~Aw?t>?,>!gi ?f>qG үMAdÿu?gx7~ tDOÏH>nGŇ/#szۮ5hX:U,վw⎳o\_?)q msXx.\~uƭ|6GJ6zwYg'rE_xsƶeż:wó>ga 犍jt<:׍Qs~`{:|Cx$|cIΣbv=so Sz?*UzwyW/o~-tMm]xs^|CkZWi SK9&?z\嬉qcCFF^,tnoۋ[9?"<9[k2GH;lZv eFՒs ;N(_tm&_|Oo|9/9h7+ Njڶޝ[VUoF%k7Ozkx/OxGѤ7z 6gZK"-'"f|:>x^;i vvz+xWxs1մz~_Dl/-|"5oME-]폅K7\~gs}Iҵ< ;|_ˆY_j<ǘ?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzZV*cw?> L]v?s/SxsC< xI/xjw^o=ƶ>.g[5toyV!o%𷂠&=SO',]?[ҵk{ [7`|2񥷄.t2k3xm៌xC߉5O'cmV!4-YXZ __>'Z)PI_JbռGUO j|=][5д-n[oxC’XI6.\ZwE_ ~?[>/'mOO< J|;þ7`m|YEjolK?  =߉Ѯ_8Ck)E[N_'¾>Ӵ^6? lj5s{3Ŭ|oOiNXmS쌟lyV"v3xTEĝ ix{z<3ǣ]iwN5Է3_=4I_YD|vs\]?j-o⏇x3:G;~ښ#lo{>/Uwm>[m3Yw|í~}R?3nڦoZo|3E&D~_fο2oP9p( (o?p???SkTˎ>@P@P@p?t;o/5\xG ]#_BѮ{{?Qgee!oIm]&n"CO:OtxPgV?uU AuW? ؖ>'#_YI6 ְx;D|}_~,C5ǂl>%]|;:6|1'kw<;k>!&$!5hn? mN+K#o!9ikX> ^uߋGV:}7RL:O xn#|, W6).UbrV Zw~!O?>#/?xq5}VHeQJ 蕭cпf^ *C(5(w]׷bSn!f9/5 Xٚ _u14X.5|F"-׆: '?vCf 'vlZZֱ Û?~&v oNg?4txr-fd5Eu֡;iV^&KZZ9 fA/Og҃GaA ; Fq'tNЎm`rֱ1c0N03žxNB$ nu+?O 6|1 xx_Z_牼UcĚkyZ]hobb ^9wl'`V <jKG"mA[X K]jnڥdXrֱoQM1|]p uƫI8ivҼ[}KXּYw-|O/ s*IMsD-.ٴm oa[Ǯ0vyx9slM/O$ktكH56_.ǤKOΥu𯃼;c}kx i|?IYO}qC]Ef>Rc`ww P/|oj70ߍl7򰜮kXῄ~MηxU?zR5kv_C\7?x\-o5>KmCX|鱁oQM1|]p uƫI8'> TTms U,fM5W‹ èm nu+?O 6|1 xx_Z_牼UcĚkyZ]hM5{:߉|?h~SY&v .]ÞYYo,t;`-.|]uNUjc+ KZg)|kxڏjcmjZ/-XgXxC#Lk?]Y|ukX ? [Ҽok*nOC6jZƵ|Or#o_$.Zk{-.} o`Џzkψ|{h(cx/\R S~*յφn-VO4i^K_ .+G+Z_x{ A#vMF=[4[M.J~x 1Ynb޸$t,?A7>4[~=W]>tSŲxG|?gkx{FKmCX9cȾ|΁y'0:w|=𿉴:vO]Ϋ&e^~'xA\u.E"[ o ڵC<=/Gsx[Q!|6^xxY|4u?ڷ>yS([|މZsm“jx_:Υ {޿dJvcHm=޾\CڧmĆ/xX|Wc-:uYɴj}#%6p:ÿeX}hus:UƧ{h/5_7mn9BCv~;UV_+? {ƺ濬dLl?>d[E޾ G~^i@a|:B{k?kc<9]Uľ"},$w^\y ֵw;QQ:|)w*x #^>>O^o' b~6{+^ ׼?o Wu-uhmS6nhht_:ţwnw->C<Y&OIڟ|7ҾYx;Csip 5jPyn^ukXρ? MojZ|Hl+׊uG~=2áox\I;KhSa7COn;woMFܞTҠ'a}? 9 m&& zEb Vխk0Fz_^wl-_gefsp)׼QeofdҼCzάl;V|W ^|QxO#mu<x )kDh>Ў#SbyHeIx+jֵ{N: דs)mu< %ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Y=KZƇ wω<OZ *K<x7M񟈼Mo%acYyXcfh4W~2]ï:`@ּQxx^oL7_Es't9ikZj |+@| wZt}WU3uKFm[OO\׵yfm[د|r>'/ i vW?ke|?T4jsi,':wuy餴;ǬF9_?fH~|TO=OԾH~_ x;ö>< /M:*׼Qeoii^!-^W[>+.7þg<iM{Ei#:GH'%X5-A#ך?IoX}s~N~"6<᷂O \|I7~#N׼iǣ<5c?<Zsa/OEs=g zO$focS}7-7maxh c/G3^W_7(o?|P@7ş8?eiL ( ( (ȼ/jc]#O-b״!im|wuY4{ <;l^rӽU|?|S'N/L`:j?Nlåxm}쓭A͞tKb Ǖox?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzZVUǧxw(O&?|79Fj;i~#.|A{gG}gsghu;k+/|I/o 6 tZꗚxkLj|?yC*.i {Tmlu aH-e~~і#9&t٦[ꉤUij?,4ښY.urZwVFhoyVl~ /IR4P σ|+Zǂ5MO+O x`Y;S6՞x5oZ^_ }v % |Inj?ao>ZwE̾T/o,bb9_m~?/=?9Q^_?#ռ{ciΫS:J7*JZuɷyV"f|:>x^;i vvz+xWxs1մz~_Dl/-|"5oME-*x?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzo+ʷء-g:'|-ţR> /IR4P σ|+Zǂ5MO+O x`Y;S6՞x5oZ^w(O&?|79Fj;i~#.|A{gG}gsghu;Wy[şH'S& HӴ}Că> wĭk 3cH5?<)g/LkWVzeվ@;||υm3'774=C,qjFuO쬼kcnzU#ZGA6*O|En-R}K௏n^]WZ߈!{k1k>w5>~ [c XyNⷎt?h:_ٵmwOU+G5C?m?]_y9|H_EQOxHgSTw l>17ڍMYzLjzj|)!wg]G1i .gNo ohz 7oYP<9N: gYxZߌ{}"F֎m*<@h/|>(G=y| [V{:tOm+߇|m*KKj[i& q[g??SD?qR],Suq&Gø.4ϫj6'O] txMxZ @06C*IuFfR i~k)E[N_'¾>Ӵ^6? lj5s{3Ŭ|oOiNXmS쌟lyV!lp? ;N(_tm&_>Y]j)|  :گ||Tit0fwazVZ麥ZQkyXn<}m ?"_Xxßٞ%<ڿ[}OIjc8?yp[ $u|Vӵ/I𯏴M/q{dk@ӖT#',n^So;}1dy#9omcO6<);G_S[A~}.h/!2MJ)h7VƷhŮjbyV|~'ҾExG־+x½QnsK7EUxotskV _-x_T-#ʟSPCo8si+'>.xnm:> ?n>\ľs0x6:4 &}kV- mxSs֢ ݾcͪZ'K &LJ)hWvηhŮ^Ǖo/qj_|}Ou]wZgB>E\<%KY_y[_&m5J;`jq_yZ/ߋ6m>'7z]Mw^4 ċohuEyH[SN-a*c{5Q k<XxNW~(h0^ ҢǤϦB-)<;ymwRB? /og[M{BLխ.'{/ /o/xSWo1'Qk3yw72P Չ?@}=v!@0:c8`u?tpI#Ӏ8 mP@P@P@P@P@t埽ddpG^; ( ( ( ( ˺6okp<7v-"|o{GM'>O~gC*vkwݿ|3}_?6ouc/ϻ*GYmTwMo{}Xh?"Z?C~QAwLו9M`.}_/u4z|) wl|/x_gŚςu?UuxҼCZή>OW/ŝw;Ua~ Ԭ'΅}}S^_ ,C|@ͥxu {touܽ{+[coٳcw&/v[žu !x}^+z^;=Ʒyz+jx?QՕͭc_~x~B6oCJ,3aˣxwÑk0|-厇d,ŷZiVͩ>eچV6m{cq$r[U5dÏ[Vx{Wu+'mVKj6o rְjv+ϊ<_cмM/Y__7> hMäjzlO6 ?/o`mZֵ|I~/xߋ7 qW_ j ]oÏ|I/uZH.n`SS|sqپN4? )Ԭ\x;GՆ.E,Ʒ:W.'m*ݵKDsKZ|>,R6~7RJ W7|%IYel4 Knm.c.Ktoua<jLJtf;7RMqV\?<9oXv"Z]^Pv/u%V--k wω<OZ *K<x7M񟈼Mo%acYyXbYW7:ޕ_Wl>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm oa[ǮL{c1&`?3|9'鴎h[|X~N7SG,#< Ru!YiRh<67M񗈼O;֬,u[7rbgXx!§y7S¿>m[:?6^(Ğ"Y.۪oZyA_mwSyw_t o%_.~#?n!Q0'~|>m`rֱ g_x:t|=xczG|acrY;H~ϩޥae؛yXc?fH~|SO=OL~_ x7Ö^< 7 sxAVSsx|EZήl>}-o|Qq^7xexOe[xKo-gYf;ö&_CG'u_Z]h!ІVktЏz[ψt ᎙ZKrڤ:U <7 ۦ5ğ{s4%ńPZܟ=7\ #f;'w!蚍5<7[?AO^%ׂ~sV!RMNM:OV)ֵGL4}#Q_scOzO]~RQ?/// kgOO74kGDn5YAh_JwZ'k鹽vk+.{{/k 6W¿=OM65äzlO>1 ῕kZ|zoOň~MīwF5_Mu7LJ|AxG>$d:ƭ $W0Mi}dmZZ;x C&?`%|G_'ͤżkcnsȣ{v+Z}~N~y/x__ڷ}wEҵ_|:DscZM R^^'6Uɮ |zoOň~MīwF5_Mu7LJ|AxG>$d:ƭ $W0Mi}dm'--k:ãh. _x_S<SHѼ[ rgڇRCs}54 2jV/8JV6nhx;D|}_,C5ςl~%]|;φ5_Mux7LJuk>!&$!5hn? mN+K#o rֱῄ~MηxU?om/ž6ԵkźM巉F5 w+ԿI]sDZ]ءw!蚎nO x*iP^1׉Gou~?5}ze}J]:EC+ ֵD5{|Z>-߉,t=ST|/|oxOÑ+]b?|Aͭz6PV}!h ~3XiAtm(%z~4VnW[LHArh&@奭cGGmWY'OZu@ggO-? ~;@Vy |]3[G?[t}8ӎ[y PHP@P@P@P@P@P@P@P@P@P@P[cx7M_Nv}~g>s}Y=o#[m3^;_|n>o{>/U_:ٱ#ߍC6}o7ͽ>}4^o-TAwe ;k&ZE \g>: (>f''U6L?C7¿|NSeTūxmΫ57R{ᯈj+hZ<$Lm/V]?\[ /B"f|:>x^;i vvz+xWxs1մz~_Dl/-|"5oڊ[_>">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿K)/ |YΉ_ _hԏu?Kk;G iŸI/>so-{v]j~"tů-/\i6ƥo5`q8o>9< k>-N Moh|Z׎<gƑjZxS_ڙ֮|ťw 4IN_i,uU0~ /.هJ?&'[Ku<6Wŭ*`+_'Ծ Ο>΅u|=>xKǺs\#pMkv7ՄuTyOD3ƞ0jj;9M.pwc$s-Au|Vӵ/I𯏴M/q{dk@ӖT#',n^SolxT,g~!Z%5?xWᗊ|kk׈MQѓG]F7S+_'Ծ Ο>΅u|=>xKǺs\#pMkv7Ձǔ> iŸI/>so-{v]j~"tů-/\i6ƥo5`q(?|?SD$.|5ԼixjKon&k^3Y5O jŞXi_g >0ee|'fHٷl:c=:O֑ʾO |Mּg7 5?j<%q=O>*Ҿ1^gn|a;/O65ȷ-7 Lu>>_j~/xo~6ut$>ѼUQh9|lxO-4+ Zu;O65f1uzu_"wQJU3[?m[Ǻ?6)?<k!~򭴩-/ūMm\E8y~)Ewk3M{o:^x]5"c[cv>)^mR;Vӵ}jʷ?iV~|+Oƭ~MS x÷|u/?#OANuh>(1^.ZI-orw75bf_} t  Z AXy:G!χo|gӵ],7*6O#}7u_O?|>(0~^6EπnK⟇|'o:_Gټ'O\jgdtoVJ uotOC᷊uZi^8_mͪx{DoƚCⲒkjq"Ŭ qcCFF^,tnoۋ[9?"<9[k2GH;lZv eFՒs ;N(_tm&_|Oo|9/9h7+ Njڶޝ[VUoF%k7Ozkx/OxGѤ7z 6gZK"-'"f|:>x^;i vvz+xWxs1մz~_Dl/-|"5oME-]폅K7\~gs}Iҵ< ;|_ˆY_j<ǘ?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzZV*cw?> L]v?s/SxsC< xI/xjw^o=ƶ>.g[5toyV!o>|A#kI[S|ҾjuyĿ5[__Z-Rzbxw}4kxj$Vo[keirl rx{Qk|?>?a u #v{&ikޣiگ?t"xeCo^z]G/:E?E٭&# ̺-CH7#ע{Vӵ]KKfJ1ZYVw叁|¯}h~׵ :]:ַ^~XjSM:7C kNg&~G$( ( ( ( ( ( ( ( ( ( (MZ&/]>snqȟ9>{v |϶ߙ}{mAm7o7o=oͿxv~lJov>7|o/O?7ߪ ;2T|5}u-?"ˆ.3xA@P@3|YaZ\v3aG}͗OKOe𷃼;c}s>/$,.|⫭{Vf>Rus`jv+ϊ<_cмM/Y__7> hMäjzlO6 ?/oa6kZYW7:ޕ_Wl>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm oa[|sqپN4? )Ԭ\x;GՆ.E,Ʒ:W.'m*ݵKDs奭`On;woMFܞTҠ'a}? 9 m&& zEb V+Z7^c&>'> TTms U,fM5W‹ èm nu+?O 6|1 xx_Z_牼UcĚkyZ]hM5{:߉|?h~SY&v .]ÞYYo,t;`-.|]uNUjc+ KZg)|kxڇ?joj fҼ cNıH?MՕɷnWV nu+?O X|2Э x_U Ğ$4Kk|Oe[6a鱡Kן4(VEW᷂O \|I7~#N׼iǣ<5c?<Zsa/OEs=g X^4W~=W]jѴSŲxG;_ ú6wV~uXɯxQt)/5E͞N!~=遌:wq'cMfA|/L|:㯍 At kl5!΃u ]ٷ07I`=rֵ,+⋍oJk*mhҼ[mKX|Yw|M/&^tx\-.YivCDX|[YW7:ޕ_Wl>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm o`M":#/7ߋ&Gڏ=w:ݖzE[MsUtST kl-67j]|?io"_G^?"z$6_:ƣnaпkkMSvIQ)7Cпf^ *C(Ү58w^FI}/e7W03kq֚mm3 ou߈ڴ:*SMNu|y=id_Cus6ixF v~;UV_u}x0OxT^'"j VmTމZ:x C&~>ֿ`%[U5dÏ[Vx{Wu+'mVKj6o!ikXw!蚍5<7[?AO^%ׂ~sV!RMNM:OWVx?g7ğj> :,|W@ Ӿ x+0^v|A|M释5^ث_]ρ? MojZ|Hl+׊uG~=2áox\I;KhSbӡ|(/‹2X1|]񾓣!,<3;QOywG lx#LJ/a[J^#拏Ğ Y.ۋo!]Zְj> #^>>O^o' b~6{+^ ׼?o W[?y ox_W&Wx^%^$owZmF}7;sm“jڵ/JucQ_|aw0_55g&cga;$(M|?|G} |- O 7 hMäjzlO6 ?/o`mZֵ}tAq}-x;D|}_~,C5ǂl>%]|;:6|1'kw<;k>!&$!5hn? mN+K'9ikXX~N7SG| R}!YirxAbǂxF鶖3r-;,uK#o+r /_uw_,R׊>#kMCi!n`Qn;B6--kXWVv:~9|\>SN#};CB@𵎏nIͫiik5-krj{G__3e1>!|N G_l[Ꚇ8|Y[Nm#PҮ<=4G{]XX7gã/SI'Ÿ Ooxv灥|^IY\'[}_WZ,<+:}ޥ`?cQw7,m<=ih <=Dx\|C[D_KXh =tGi:+z{c7~#N׼iǣ<5c?<Zsa/OEs=g w?眓Q+am8sm“jڵ/JucQ_|aw0_55g&cga;$(6sR_C'G!_G49 nV;P@P@P@P@P@P@P@P@P@P@VwX&ngӮݟe߷8OvO`ighW nO}ϷUfm;?leH7|Pͻjnio|Ϸk ?KGo|?H>:tɿ@_ϼk玠 (,0`MS.;Of@P q6^g]ᮑ ~1Xku4w>n[;Z:EŅQZ?H|9qV*ZB(|OŚN}--/{qkg'$_aNvL>x]<?> Ӽ 7cA˨^i/-4 Q"_C#kL?Z]^Zn6[r mxS:΢ BO 6:||TOm2-% n5sKu{]7TKBZVUǧx{(K.?|79Fj9i~#n|A{i~}iseh+k+7=Hv4@!o|uOɦxWA[ZxCDž~xƿ |g.oJּaxo|>ڊ.n,&%uᵞMVYOp\h-դյ }>,LwxHG`G7U]Vw叁|¯}h~׵ :]:ַ^~XjSM:7C kNg&~G$( ( (Lg $zp ( ( ( ( (z|t埽ddpG^; ( ( ( ( ( ( ( ( ( ( (*.ٿulu۳;focS}7-7maxh c/G3^W_7(o?|P@7ş8?eiL ( (8W:`^eCY&OI 7ҾYx R+K 1C{+m{?q X?ixiox0m4߄6n976O&?/`rֵ_~x~A6oM u+?5aˣsÑk6 厇b, JmRQ2\o`rֱh?4+~..=׊>#k7h?|F4ۈ??|:vto+]7Ýڗ|M?(Kxai_z=gxwJ>ȶ XP@=n\= & P@P@=d߁prF;r :c^ q93ԀMA?==9$ (# ܐq}'zq?ǧ\c펼1ۯF8 @& P@c`t=Hzcj?Ǡz^P@PۧFz5[?vxzV kkuQu_ qg#IDl'S?[ώtOZuϋF|_[i>A;V1jVvl5=2j qizK=4M1;K l/i?W2o|;@noxzxJ> 3h|ǕzE$?|En-R}K௏n^]WZ߈!{k1k>w5>~ [c6S)/ |YΉ_ _hԏu?Kk;Gw5>~ [c XN<q[:4/ڶIq~*!kCWf./>MS@Rk/_(G'<|3?;6 ~GFM&,u=c?Fӵ_ Xh>Do/x?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzZVU_WO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo ǔ< `+_6Q?5KiZ Z꺞 SYu/x梺mxRK 4eˋ].赾Cpiz/S-Igt .&/kEŴ:ǂ|Sڤvwk0Մʷ> iŸI/>so-{v]j~"tů-/\i6ƥo5`q96<)9 zQMsn@of|{-?VOC4+ g[Ҵ]bM/Rҋ[q[zwh7l|/࿍]]O_پğ;FNwyXYB\6V& lyjz;olM?{OଚO \xS4WF[`K>go3M l7-=35Q k<XxNW~(h0E\<%KY_y[_&m5J;`jp? mo|sx|Z5#O%#N 7x,#Tؿ3a]Y闃VK@h9N;>ӼX?1a_P];EM?N 6ym.m[7UWO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo ǔ{ޟoo:gxTEĝ ix{z<3ǣ]iwN5Է3_=4I_YD|vs\ ]NH瑂3[|o+"k/^_i/Nc _ɹ>7'u,6FOXܼ ?Y4 CеE\<%KY_y[_&m5J;`jQHǺI ]w:kexHնxżS-X<]|AŨxUfG?!m*=3=? _?WafxhjoNvL>x Y.Ax]?/i:E@ŭ|W񭵇qov϶{;Ya2`jt<ÿv=SZωZ׌&o=VMSڧgo3M l6[O |Mּg7 5?j<%q=O>*Ҿ1^gn|a;/O65o8oV1uzu_"|;Wh#<|,exocjd<-xJz|Ua|b2(w__K;ym koon6[oxę>, ||!u$^"߄m H|/x:~њsZc4ZhW>]Ҵizvm`j)'cc둌0=>ZD G~'G?>!g?ڷtm79|Sjx@|Co _;[iRZ_VN{|q['/S-Igt .&/kEŴ:ǂ|Sڤvwk0ՁǕo~>/WuZſٞ>o<-_hGm )|Q{oxcҼ=]隵Z _ok''H?=/0Þ5.-tCuk? &Xo!`=x>W/G#xG:'~Aw?t>?,>!gi ?f>qG үMAd ?k |Tx:Cx+k}HNvL>xl&O#~`=x>W/G#xG:'~Aw?t>?,>!gi ?f>qG үMEY+|'}o~"|__}xMG⋹4/|O?rT:.N晨Q[wYg'rE_xsƶeż:wó>ga 犍jt⏄qJjo="Z~1m/ĶDɫjvWNl<}NwQJU3[?m[Ǻ?6)?<k!~򭴩-/ūMm\|ʷ [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJ-o+ Ǖo75bf_} t  Z AXy:G!χo|gӵ],7*6VK~5|~$_懨x^x_Ve5 SÞ(izl|5]J$k]h& q[h Tx¨/ j=xUN] %~mIi-Zkm:<}e_wEi6#xΗx?|Es@0|HXX]tWTյ8bF88o>9< k>-N Moh|Z׎<gƑjZxS_ڙ֮|ťp? ~?[>/'mOO< J|;þ7`m|YEjolK?  =߉Ѯ_yW׃"k/^_i/Nc _ɹ>7'u,6FOXܼo; ixN4;9^.xǿG?<#iMS߈6![_߂> j7i6>f!6ZGI'X6/gyXG*ZFM{^Я åsk 5 ߇ņ4McxNM>>Сִv~kz"B ( ( ( |ofַ wnϲۜl'vOk'mykwݿ|3}_?o{o}M'>O~gC*o{6[62Ҥ(fݏM7g﵆M%7>$y_Cd OHrΉ5PP@Pd~0֩}3H ( \v9^ky=AoºG=+O]k~B6ڻh0M=E_K>G/t:`ګxGg%:/-Xl,|!4OG7FƓwuermխ`w|7&.Xk|JwptmpcUO$Z~xw(׏|CM6I|CjAu{sڜVFBrֱ|~,ڭ߈t Sng<'t+GJYe,4 Km.S \3ۦ{:x C&?`%|G_'ͤżkcnsȣ{v+Z~;UV_Ag_zW~m^-cxJo-cZfi?5xS'5_촻fѴ?6[|sqپN4? )Ԭ\x;GՆ.E,Ʒ:W.'m*ݵKDs奭c?%ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Yy nu+?O 6|1 xx_Z_牼UcĚkyZ]hM]`6sr؛^Iycam< %ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Y=rֱῄ~MηxU?zR5kv_C\7?x\-o5>KmCX|ij? l)]I'Ÿ K_xvׁ~@WUλ-<+:}֥jϥ:~ _~&Doa%M*x:/ÚJjriǭOV(}oa9]Zֱ,+⛝oJMŶ ;oi^-񾥬k^,;Ö'oj$&Zk}l67c%ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Yy KZǾ:qy8>AS/ٿׇ~"ʱ:աw//g,S]\Ef)?[=gklhhtoZŶ;Vcօy}K^^!@~ |@5 kdou{+m`[bYW7:ޕ_Wl>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm oa[h?e_|uu(>x0m4߄6n976O&?/a9ikZǧGR5M>}=O|1< [ٮ?jy7Xk'km{s4%OQ{ ZAg_zW~m^-cxJo-cZfi?5xS'5_촻fѴ?1n<jLJtf;7RMqV\?<9oXv"Z]^Pv/u%V?|SN׏c ^Q =Դ_[3>!"4G/h(6 ֱῄ~MηxU?om/ž6ԵkźM巉F5 w+ԿI]sDZ]屡Kן4(VEW <+q'\Ɵ7~$|Niǣr-a_ZAuo} ^XY$v]QkC?Ꮒis⯊{[>dfh4W~2]ï:`@ּQxx^oL7_Es'tا--kX7¾)<2[2 ږx_7ơz?ivҼ[}KXּYw-|O/ s*IMsD-.ٴm o`M":#/7ߋ&Gڏ=w:ݖzE[MsUtST kl-67j]|?io"_G~G"zD1yg<j߿eQN|)mz%kX>ρ? MoS3|uB7}j9CMAwk~!?k_?" 6]x|YEϜouo􋟶^C߻ٴ[yX4/ٿׇ~"ʰ:j,tN)4=;]ץѥ_Kj~;_ "osG5o=CvZxWx?T^8Դ=GO =oA]'ún⯷O j+ZOAqͪyX+ZA=㟈~8,Do'MW^Q1["헐7m+V8~?x s | q j/xJN]%~$mwZmF7--kN1|]}Q'kg4xsgw/Ef>+Үψ<[ޚ[?y ox_W&Wx^%^$owZmF}7:Sj|#h7~|i8vFԵSy/ x:5ggvi)enJݍsm“jڵ/JucQ_|aw0_55g&cga;$(l&p>j> #^>>O^o' b~6{+^ ׼?o W)6[ij)!^)5ws]s_m&6vڟcH-M޾%ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Yy KZ v ߉5 מ-Ց~}'j?M+mh,l,|Uxtw.k_2f6~?x s | q j/xJN]%~$mwZmFNZZ%ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Yy\CڧmĆ/xX|Wc-:uYɴj}#%6zt9|χtu?x/OM' +x?U'6+ x/H7,4o .io?x'"SKnT6ܴbgH׏C?Q^>O |8?ohڗ x_zxͶdn lmZZ/'7{b7& nO *iPx~0׉|Gou~ՅLju}rTSN=jxEc+խk ^9wl'`V <jKG"mA[X K]jnڥdX奭c̾ ? þ5o&؟4yxkw%R]t/=nѼ;\U}Hx6 XP@P@P@P@oulߺ6y~n:[sD)md 7nO}ϷUx]{mighVm]{#f_wT~7 ۱6}zɿD~Q߱d+sLi\7qƾx ( cT2iP@P@^g]ᮑ ~1Xku4w>n[;Z:EŅQZ?H|9qV*ZFh >)|xw&?tU| 5 haҼs} I f:Mk|ʷ< [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJ-o+*cӼ;]|mU@>Q#P?O5o^4> w #Ӵa :M ˂^>$_xⷅ|Nk-uKJѼ5F>סx\_ؽiz6:0\[locпh|J_:{ߌZBx᭿D|UTYÖE mM,Uu9-|;[XSӴKn<}¿|NSeTūxmΫ57R{ᯈj+hZ<$Lm/V]?\[ /CӼ;]|mU@>Q#P?O5o^4> w #Ӵa :M ˂^4 )0a| 9G CyѼso7|l4oٹҐ^\Y}6-c_k|sѩ$)vxgxf i<0, jLN-/C|/@cmEj>$7_57`tf{"_*ᷖWڱ1h Tx¨/ j=xUN] %~mIi-Zkm:<}Ə~3k u < J4φz?= <+H<9j=?I/xR\6sjȦTϕ< [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJvq[P_k|sѩ$)vxgxf i<0, jLN-/C|;]|mU@>Q#P?O5o^4> w #Ӵa :M ˂^k+<[ώtOZuϋF|_[i>A;V1jVvl5=2j qiz>YQLZg|BÚm{M}[Ŗ~85OxS}vV^5ַyw=*Hu|Ǖn'_>">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿<q[:4/ڶIq~*!kCWf./>MS@Rk/_(G'<|3?;6 ~GFM&,u=c?Fӵ_ Xh>Do;||υm3'774=C,qjFuO쬼kcnzU#ZGA6 Ǖn4> Q_x{s  |G-xG|=W:t w6oþU%صio83^i ]w8kgxHծ~):xugյ}+TWxHG!F L]v?sRsC< xI-w>oM ?Xxg[ooM@ʷ8? y:MzMRV?:xnToK"?ykwxM1YtrK-o^_WO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo ǔ{ޟoo:g7'u,6FOXܼo-e.uo~täKuYOMaj0i@n;G*< O?!gx[Ux{:= ^%^mKi}-Z{}6'U!_&/<]>>Y]j)|  :گ||Tit0fwazVZ麥ZQkyXn<}m ?"_Xxßٞ%<ڿ[}OIjc8?yp[ $u|Vӵ/I𯏴M/q{dk@ӖT#',n^So;}1dy#9omcO6<);G_S[A~}.h/!2MJ)h7VƷhŮjbyV|~'ҾExG־+x½QnsK7EUxotskV _-x_T-#ʟSPCo8si+'>.xnm:> ?n>\ľs0x6:4 &}kV- mxSs֢ ݾcͪZ'K &LJ)hWvηhŮ^Ǖo/qj_|}Ou]wZgB>E\<%KY_y[_&m5J;`jq_yZ/ߋ6m>'7z]Mw^4 ċohuEyH[SN-a*c{5Q k<XxNW~(h0^ ҢǤϦB-)<;ymwRB? /og[M{BLխ.'{/ /o/xSWo1'Qk3yw72P Չ?@}=v!@0:c8`u?tpI#Ӏ8 mP@P@P@P@P@?g`׎` ( ( ( ( ˺6okp<7v-"|o{GM'>O~gC*vkwݿ|3}_?6ouc/ϻ*GYmTwMo{}Xh?"Z?C~QAwLו9M`.}_/u4z|) wl|/x_gŚςu?UuxҼCZή>OW/ŝw;Ua~ Ԭ'΅}}S^_ ,C|@ͥxu {touܽ{+[coٳcw&/v[žu !x}^+z^;=Ʒyz+jx?QՕͭc_~x~B6oCJ,3aˣxwÑk0|-厇d,ŷZiVͩ>eچV6m{cq$r[U5dÏ[Vx{Wu+'mVKj6o rְjv+ϊ<_cмM/Y__7> hMäjzlO6 ?/o`mZֵ|I~/xߋ7 qW_ j ]oÏ|I/uZH.n`SS|sqپN4? )Ԭ\x;GՆ.E,Ʒ:W.'m*ݵKDsKZ|>,R6~7RJ W7|%IYel4 Knm.c.Ktoua<jLJtf;7RMqV\?<9oXv"Z]^Pv/u%V--k wω<OZ *K<x7M񟈼Mo%acYyXbYW7:ޕ_Wl>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm oa[ǮL{c1&`?3|9'鴎h[|X~N7SG,#< Ru!YiRh<67M񗈼O;֬,u[7rbgXx!§y7S¿>m[:?6^(Ğ"Y.۪oZyA_mwSyw_t o%_.~#?n!Q0'~|>m`rֱ |?k|L#zzi>g^Bl<=^cZG|aaxritH:W 嗆5?,n<5(4izsKlwAg_\kzW~}^-iV[mKY|YNEzI]sDŖn6t!64#^c&=h(e+RW>*յrR+ ~#UMmf"^{FݏM==y9Nǁ;D|}_~,C5ǂl>%]|;:6|1'kw<;k>!&$!5hn? mN+K'NZZ(x[sUOahW6n]|]GΩ|Mf,,$d:ƭ $W0Mi}dmZZ;x C&?Dƣ_ h<۪{E۶ oDkOlz`c9c4_ |Ok|Woj>ޡK|Y'Vt]_. zH_XiIy.m4 Swk/xߋ7 qW_ j ]oÏ|I/uZH.n`S`k^Sy3|J^}O~O?C"Fo5_j | !mjt k6| /'cmJoxŶɥj~ ,#Li:>khphc+ }7/;D|}_~,Cuǂl~%]|;?||Oux7LJu^!.$ uxn? }N+K#o rֱῄ~MƷxU?;oi~-񶥬k^.o-|OAŚ?Gךe۶a>o- '}B~.M񾉨1׍l7򰜮kX4OW7ŭw]xC<%OM|7i_ ,C  W F=RR^J3A¿>|]Ƌ(~F|Ya?0 0i7I奭cGFmWY'_Zu@/<#>xsgw/Ef>+Үψ<[ޚ: (>f''U6L?C7¿|NSeTūxmΫ57R{ᯈj+hZ<$Lm/V]?\[ /B"f|:>x^;i vvz+xWxs1մz~_Dl/-|"5oڊ[_>">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿K)/ |YΉ_ _hԏu?Kk;G iŸI/>so-{v]j~"tů-/\i6ƥo5`q8o>9< k>-N Moh|Z׎<gƑjZxS_ڙ֮|ťw 4IN_i,uU0~ /.هJ?&'[Ku<6Wŭ*`+_'Ծ Ο>΅u|=>xKǺs\#pMkv7ՄuTyOD3ƞ0jj;9M.pwc$s-Au|Vӵ/I𯏴M/q{dk@ӖT#',n^SolxT,g~!Z%5?xWᗊ|kk׈MQѓG]F7S+_'Ծ Ο>΅u|=>xKǺs\#pMkv7Ձǔ> iŸI/>so-{v]j~"tů-/\i6ƥo5`q(?|?SD$.|5ԼixjKon&k^3Y5O jŞXi_g >0ee|'fHٷl:c=:O֑ʾO |Mּg7 5?j<%q=O>*Ҿ1^gn|a;/O65ȷ-7 Lu>>_j~/xo~6ut$>ѼUQh9|lxO-4+ Zu;O65f1uzu_"wQJU3[?m[Ǻ?6)?<k!~򭴩-/ūMm\E8y~)Ewk3M{o:^x]5"c[cv>)^mR;Vӵ}jʷ?iV~|+Oƭ~MS x÷|u/?#OANuh>(1^.ZI-orw75bf_} t  Z AXy:G!χo|gӵ],7*6O#}7u_O?|>C|"=ܺ ~k'{M{U޹{hi'.4 iz+ZP|q^Ax]?/i:E@ŭ|W񭵇qov϶{;Ya2`jt⏄qJjo="Z~1m/ĶDɫjvWNl<}NwQJU3[?m[Ǻ?6)?<k!~򭴩-/ūMm\|ʷ [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJ-o+ Ǖo75bf_} t  Z AXy:G!χo|gӵ],7*6VK~5|~$_懨x^x_Ve5 SÞ(izl|5]J$k]h& q[?/;?$§/mWǺO.*4{xsÞM-صiۣo8_yZ/ߋ6m>'7z]Mw^4 ċohuEyH[SN-a*c_k|sѩ$)vxgxf i<0, jLZ^ſ|3hH|3Zù;{^q/ VŗVTޘ M Ǖz]x;^N/ve>x"?^$^ O? 'k d=Kt~.|0Hb޶&㳚 r ixN4;9M]?j-o⏇x3:G;~ښ#lQAwLו9M`.}_.O_/>:폅K?x@N'Y[yWtKW̓Klo?x?+_>(OB6 мe}|5jzw4 xhG<$2$h[ѿڵkAg_zW~m^-cxJo-cZfi?5xS'5_촻fѴ?1n<jLJtf;7RMqV\?<9oXv"Z]^Pv/u%V?߄C߉75krxoƶ SJĞ;K;{+/KuθR%6&bx__ڷ}ux{FҵO;l|7/&Xi֕c&-GIO 6z^;|ZǮǦ0xߓ3H4ٚ _u14X.5|F"-׆: '?vCf 'v)KZ; .5+?MŶ NZJo-cYf7߯xzI]sD׾e۾ oaylAg_zW~m^-cxJo-cZfi?5xS'5_촻fѴ?16ρ? MojZ|Hl+׊uG~=2áox\I;KhWӠ|/?+h~TխeW{eB񮹯?6;O$vD޾Bxw/,xJCӵz_OmY%F_&ͭ4sZh^q|4/ٿׇ~"ʰ:j,tN)4=;]ץѥ_Kj~;_ "osG5o|7:_YV_mZ\Eq&k?4K|W/k!M[hм#mos;[c?gk"_GCQ7Ѧ+(sb{6 o+7:_YV_mZ\Eq&k?4K|W/k!M[hм#m| O /x:4_^  =Q:^wM|U>mSzܟ[./YSo+z%kX<=/GsZQ | oDis /F?|{\ͥ~G~m[Z?6i^)Ğ"mY.۫Cm=奭bw P/|oj70ߍl79]Zֱ~"h~M?dK+_֮_gx,3N%hxgzU|]7[G?o({b}w>_W>)6[ij)!^)5ws]s_m&6vڟcH-Mo_NL /4bwNxDόG=5z5񇏮m*_xWoxid/n. ukZx;(x>[?y ox_W&Wx^%^$owZmF}7~?xŸ w§{7 ѿu?JV/kĞ!Yn۫Cmڟso]|3MOτ~-O<g^ږ>o<#q>o\N#"#՝i[| RmS{VSCe _Sj>+? {ƺ濬dLl?>d[E޾߳GDx!d#4o]OҼUex *'|}V[j6>m_eQyK/7t}:%|gu= $6_:ƣnaпkkMSvIQ)ӡ;D|}_~,C5ǂl>%]|;:6|1'kw<;k>!&$!5hn? mN+K#o!ikX_w;?u_ź//]G֩|M ͅ֍k_C\>w,rֱG~m[Z?6i^)Ğ"mY.۫Cmy KZx;D|}_~,C5ǂl>%]|;:6|1'kw<;k>!&$!5hn? mN+K#o rֱ|/?+h~TխeW{eB񮹯?6;O$vDo_N#/?3nᖥ}/CJy&񅆍ṥm-g/D[wyXFVPw?QI:|'?>kog-R^[WOyڬwmաmKZ~ _~&Doa%M*x:/ÚJjriǭOV(}o`r`߇7~.MCJ5ǃ XirZͿ+hg/F8Ne rֱOlz`c9c4 <-q'\|!;^4h֚]< ykᇀ >KuθR%6w|6^k:'5yFt [X`? <iu[uDzt,?@nV=?r{y8l-G| RmS{VSCe _Sj>+? {ƺ濬dLl?>d[E޾t?j^Z^k!7~(5X -텦Aw~*:F" }@ ( ( ( ( ( ( ( ( ( (  +PYKܨ7WPҩEdeԔV/u]!^ȗC σc/G3^W_7(o?|P@7ş8?eiL ( :{c1W0sۀ < 1|5?4z>گ+ {B< {gyy _UHö5Y :u_ߍ7Ej$KdmovgKR }C!|M_ x}R}G_S[AI}UTOMEx Նiz.khK[ʷx_~)|=ե'φ>}?H4vWׇ5/zψ/m58.lCֳEpൗe~{ ~!>/,xKVGۂL?'Ohv >&k^3Y5O jŞXi_g >0ee|'fH7-Y4 Cе9< k>-N Moh|Z׎<gƑjZxS_ڙ֮|ťui/T46ԼijK?O źoŚxI6  h|ʻ/j\/Uš_|AKZk7ً JiFb}|9Ci/DP@P@tq1$N P@P@P@P@P@t埽ddpG^; ,##: lP@P@P@P@P@P@P@P@P@P@P@P^ M_ZJ_Gtx!_ºM\s.;#.|UGêDWPf|?H>:tɿ@_ϼk玠 (,0`MS.;Of@P@¹s H r){-[7t{y|W F8#Em%v`I{|}\CڧmĆ/xX|Wc-:uYɴj}#%6zt8?|SN׏c ^[??(Qlͥxcc쑢?54+oNWVw|7&/|XkJyq j >&׼þ FI/u}Zοuys6FBrֱD5{Y~-[J2x:O W7<%Ḏ0Xh\\PgMVQ-X/eklu#9M<K(KOGHyǝ["헑G?+o7VBxw(`uů(Ү58Duv_kXW(n V_Mֺ-|3 fA|/L|:㯍 At kl5!΃u ]ٷ07I`u5{:߉|?h~SY&v .]ÞYYo,t;`-.|]uNUjc+CA_wSww??8H=?txrAׂ8OnӦc`t=Hzcj=?txrAׂ8Onx~91& c`t=Hzcj(=q xSx1~N?ǧ\c펼`u8<ӎ:{cFzAMA( ( ( ( ( ( s@Ǡun'%Ttk=tcZ'Oq~Fxv4K%=ϮOWO0+~Gw遂2HylDm~_fο2oP9p( (o?p???SkTˎ>@P@P@y-YAlbki}_V>M-ϛx&osqacmkֲx\xzwGϊt ,U_ G"/-tcm/!uH47YΓiu{lZ!96<)9 zQMsn@of|{-?VOC4+ g[Ҵ]bM/Rҋ[ʷ_~)>d'φ>};H4r[ׇu/zψ/lHlCGrൗe~{ ~!>/,xKVGۂL?'Ohv >&k^3Y5O jŞXi_g >0ee|'fH7-Y4 Cе9< k>-N Moh|Z׎<gƑjZxS_ڙ֮|ťui/T4.|5ԼixjK?\xɼQ źڍ *+ so-{v]j~"tů-/\i6ƥoڱN<-g:'|-ţR> /IR4P σ|+Zǂ5MO+O x`Y;S6՞x5o8  mxSs֢ ݾcͪZ'K &LJ)hWvηhŮ^ʷ iŸI/>so-{v]j~"tů-/\i6ƥo5a8+o~tj&)gVkSć_MĻDs6OqK>Y]j)|  :گ||Tit0fwazVZ麥ZQkyXyV_>">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿'S/ǟ~Gh$/Ah1j#[sz5MfMԾk.Z{-΅u|=>xKǺs\#pMkv7Ձǔ< [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJ-o+ Ǖo~]6"wu?f{|뚏luϰi:V=agss/ p+X+^O |Mּg7 5?j<%q=O>*Ҿ1^gn|a;/O65olH|KGOo1Xx@Fïia ;^![D"kluC)Eh~l>$ks1k/A5d|En-R}K௏n^]WZ߈!{k1k>w5>~ [c XyNGxM3¿-BMn<+>5[?vxzV kkuQu_ qg#IDlO] txMxZ @06C*IuFfR i~!WO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo)/ |YΉ_ _hԏu?Kk;G">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿'Sƚkχz}'47TT犵m amMT2ͨmMSt/A2hui:?O`חR|>xj$Vo[keirl&t=8~CC c#Fim p>׼S⶝zO|}h:m$~Ïk&gXĞ.֜ڧ>cr|c :d<+B -ߌ <}F¿ S_3m7k^0ּF_>^mEU7z":4Oaw_>">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿ga 犍%'Ohv >&k^3Y5O jŞXi_g >0ee|'fHٷn6[O |Mּg7 5?j<%q=O>*Ҿ1^gn|a;/O65o8oxwv4@;@eXǶ1#`z}~>U]xTVmm?'[yT+Y~!U{?w WQ'}},iEo?_d|7ԒCSx~񷃭s1!mFike|9i\n wJӯey1lcF3Өix'h Tx¨/ j=xUN] %~mIi-Zkm:)Ǖo ;N(_tm&_|Oo|9/9h7+ Njڶޝ[VUOh_~5koŸfx[|yz t3dxAEJwjIk|}~ѼQ4"[[_OH+x̸Q?v|;{=֝0a|Q5dy|QG:~}~-6.o;]> *k5Pvໍ:Úf~ZV"Q[ÿu?gx7~ tDOÏH>nGŇ/#szۮ5hX:U,վw⎳o\_?)q msXx.\~uƭ|6GJ6waF.G/~(Aw/c/Ci/4i(>!h[|h(|OŚN}--/{qkg'$_aNvL>xؖO#~`=x>W/G#xG:'~Aw?t>?,>!gi ?f>qG үMEY+|&?7Ao|SKė^7Uxc x_>'soQmq]BZu. +o3Ӵok''H?=/0Þ5.-tCuk? &Xox"?^$^ O? 'k d=Kt~.|0Hb޶&㳚 rx{Qk|?>?a u #v{&ikޣiگ?t"xeCo^zVw叁|¯}h~׵ :]:ַ^~XjSM:7C kNg&~G$( ( ( ( ( ״u] YMZVa;iךo,䶘ZWfvqi xx@ݔigDv6l);yrNlu\}ח_]A D߈y/_we?_÷gsgڳm *_ջ|~<ojz?tt K_oZ5닍JLԗ]tAp"HhZ[A说=tmHO#i|?H>:tɿ@_ϼk玠 (,0`MS.;Of@P@¹s H r){-[7t{y|W F8#Em%v`I{|}?_>5ӼuCşX׵V7J5t_[3i^'bXC$h~ng&7+ZoQM1|]p uƫI8M;B:KZӘǶ=88 y9|7¾)<4[aOC6ZƵӼ9khk&VOk%ivͣhmM5{:߉|?h~SY&v .]ÞYYo,t;`-.|]uNUjc+ KZKG7'w? \x&Uûk&|þ F|#iKcV ۘ&ⴾ6--kAg_zW~m^-cxJo-cZfi?5xS'5_촻fѴ?1n mL7p?眓Q&xKG7'w? \x&Uûk&|þ F|#iKcV ۘ&ⴾ{y奭c,+⛝oJMŶ ;oi^-񾥬k^,;Ö'oj$&Zk}l67c?f #H~|SO_/>:¾폅1'f?⫝w0ZyWtKWՍKlu '}B.M񾉨[~5JT$ߌ/u_x'_5aq_\-ӡZ(QarbYW7:ޕ_Wl>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm oa[KG7'w? \x&Uûk&|þ F|#iKcV ۘ&ⴾ6}tAq}-h_ïEbuC'58_Cӵv_}kY O7[ ͭ,S~{> ^ߋmwvy->C<%O 9i_ ,C Aŭj,ɪ1@#bV} nu+?O 6|1 xx_Z_牼UcĚkyZ]hMA_wS.EP|ZGa! xim3sd:om>Mgh_rֵNzkψ|{h(cx/\R S~*յφn-VO4i^K_ .+G37¾)<4[aOC6ZƵӼ9khk&VOk%ivͣhc+6(x Û?~&v oNg?4txr-fd5Eu֡;iV^&K'--k Ij>+7{hg| b5BDia_ 3ѬQwugtmխc,+⛝oJ-ź/=_|mk׋u>oxjWcĺky촻w?1cB??>"ix᎙ZpuH5OV> Xk_?[kۙy-|(|P:ekl'ox}>oGxQGy4kM.I~x  ^WY%z\{)WOcwCᏂiw⯊{[><=iZd6vtm_4zJ^R^k~'=/N;Cn-cz{c#kLChA>936O"`=rֱxo|Ss^7xixmm+ž7ԵkŚxr""7W?ό<&ѼEڇ=skZ̿<~EKMsUtWSS.f Yt={?q X?ixi燍GGY~ݫ~헐)!7V\CڧmĆ/xX|Wc-:uYɴj}#%{ }:¹6IOZKWO _%~:NǾ0[t/i1GimJl t/ٿׇ~"ʰ:j,tN)4=;]ץѥ_Kj~;_ "osG5o|mBxw7,xJ}Mu<{q}m%Tn Dk=6 6ݬÿeX}hus:UƧ{h/5_7mn9BAm{?q X? GOFG?>,cg7E/!oW-pÿeX}hus:UƧ{h/5_7mn9B!;[c-<+∼W}Hu x*/jZ'Gz7YWۧwǍOr-o'fM 蕭c?gk"_GCQ7Ѧ+(sb{6 o+gH׏C?Q^>O |8?ohڗ x_zxͶdn l '}B.M񾉨[~5JT$ߌ/u_x'_5aq_\-ӡZ(QaukZlj#z|Iq6/ZCΗٞƟ <9; Wg-woM˱x[Uk\CڧmĆ/xX|Wc-:uYɴj}#%6!}:2(%:>>3u(t0ǂ>xr>մ|E;^YI \6խk Gvu-uhmSm<?|G} |- O 7)6[ij)!^)5ws]s_m&6vڟcH-^o_NG~m[Z?6i^)Ğ"mY.۫Cmy KZh_ïE`uաXWRhzvKK$~2vDkM 66344OW/ŝw;Ua~ Ԭ'΅}}S^_ ,C|@ͥxu {touܽVǙ|~|G|kgxÞ4usQo$,aa~kÏiR 7--k\Ǧ0xߓ3H1mSuOj1O&i:A`\|~x $Qo\e*YlA+y9=<`r#7Ýڗ|M?(Kxai_z=gxwJ>mwP  ( ( ( ( (MZ&/]>snqȟ9>{v |϶ߙ}{mAm7o7o=oͿxv~lJov>7|o/O?7ߪ ;2T|5}u-?"ˆ.3xA@P@3|YaZ\v" ( (< 1|5?4z>گ+ {B< {gyy _UHö5Yj~g kú=Bzww6zv!Z{I{aypZVběVowO宩yZ7(x:_¯ M/P{&P|W pZa;oi_ 'O{P]h[|5>O_:K3rHamZN%ukkjzv~ ǕoWωֿ l4j Ҵo-u=Sv_|5VEt- [Ǟi˧]k|zw|_⋯jhg~A >j~g kú=Bzww6zv!Z{I{aypZVcßw|}?l/x7?ao~/:7u oy7:R ˋ/[yV%p? mo|sx|Z5#O%#N 7x,#Tؿ3a]Y闃V ťzh7l|/࿍]]O_پğ;FNwyXYB\6V& wQJU3[?m[Ǻ?6)?<k!~򭴩-/ūMm\|Ǖo|>/mwã.ᧁuC_ F_agg|Agi< [G>OKfmR#VRꑙkBkÞteu7oX귳j=R+ !Ҋx݆Z.kꗩiNn<} mo|sx|Z5#O%#N 7x,#Tؿ3a]Y闃V ťz|_⋯jhg~A >j~g kú=Bzww6zv!Z{I{aypZ-e~Ǒ|/ |YΉ_ _hԏu?Kk;G L]v?s/SxsC< xI/xjw^o=ƶ>.g[5to8+_'Ծ Ο>΅u|=>xKǺs\#pMkv7Ձǔ~~_>+xCV7.?O[?}CZ$? h6l%%>ɴ{*]m`q Z$gtw>5OG~ xo~1ɤxŚxvk'Ȟbm^}~5|~$_懨x^x_Ve5 SÞ(izl|5]J$k]h&!8ߴG*< xOT|Cחooos@ wʶҤ5rmUK=4M1;K l/i?W2o|;@noxzxJ> 3h|Ǖzi]G1 .gNo _hz ?P<7N icgK ww3ҭ6 yV__>'Z)PI_JbռGUO j|=][5д-n[oxC’XI6.\ZwEK iŸI/>so-{v]j~"tů-/\i6ƥo5a8|~4/~3^|;<-L.oaڧYmE-@_v__E|+;@Eo# xY77j~g kÚ=FَW֗6Zv!YkY{QypZVc=? _?Wafxhjocr|pwc$s|o+*x?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzo+ʷ,>>G}/Z|_=gⷋ/,ui~F[:G'>\/|֧ڥQ9=-fv ~#Ow!Ṵo\V/Aysgz%ϋ/kXa\ ,t/ y::q [x?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzZVU_WO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo ǔ>|wQh~.M?'s^w5xs\/.oV?>-<c#mN;;Xц<}H|KGOo1Xx@Fïia ;^![D"kluC)Eh~l>$ks1k/@jK=4M1;K l/i?W2o|;@noxzxJ> 3h|Ǖz4{Gյ_ ._~30煵K 4mo#6/m zW 3VK[h7l|/࿍]]O_پğ;FNwyXYB\6V& DP3SH=8A@P@P@P@?g`׎` ( ( ( ( ( |ofַ wnϲۜl'vOk'mykwݿ|3}_?o{o}M'>O~gC*o{6[62Ҥ(fݏM7g﵆M%7>$y_Cd OHrΉ5PP@Pd~0֩}3Hz dq8$ ?==9$ (t;o:E濠kmn/ x-W.}_ëEI@m**YcO:OtxPf|b> ګxj??'WQlͥx?[d ,MYĦ7+ZoQM1|]p uƫI87"Z2iA>74~$VۛkOԮqn rp:gã/SI'Ÿ Ooxv灥|^IY\'[}_WZ,<+:}ޥamD5{Y~-[J2x:O W7<%Ḏ0Xh\\PgMVQ-XW0&O |8?ohڗ x_zxͶdn l--k vk,.|>F= x+BxS։d|-:FhpoFVխkXDoQM1|]p uƫI8e_x>Ҭ/ZoJI/:U69ikXٚ ߌu?W|uO:?@QU^?m8 36O ggV--k wω<aOZ *J@t|1'ľ1Ѽ54mExϩ%acٛyXkXtهG56_.O_?>:W 嗆5?,n<5(4izsKlwAg_\kzW~}^-iV[mKY|YNEzI]sDŖn6t!64#^c&=h(c/ܶjÉ5 K>$|UMm𠱊!^.=7\ #f; G~m[Z?6i^)Ğ"mY.۫Cm=rְÿeX}hus:UƧ{h/5_7mn9BAm g]'j~!4=OڕtlokxOq']aW6Ipn V_=㟈~9Q!|6bxho?l9wi[|w3pGA\n /^ uߊ+oxQj-^u;oxú6 nO 5_:~׊/+sgSwk/xߋ08qW1%Ou7LJ|AwxN_k7W0Oi}bmZZ0uهH5/^)]<%x+Ok*eX#X-m3k>%ԥqxGko Owv7Rȿlx> | okt k|7֬)c}NoŶjzcgI]+DBX\%ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Yy nu+?O 6|1 xx_Z_牼UcĚkyZ]hM '}B.M񾉨[~5JT$ߌ/u_x'_5aq_\-ӡZ(Qar`> ^ߋmwvy->C<%O 9i_ ,C Aŭj,ɪ1@#bV}!h?e»㟌P|^(D5J ( ( ( ( ( ( ( ( ( ( (MZ&/]>snqȟ9>{v |϶ߙ}{mAm7o7o=oͿxv~lJov>7|o/O?7ߪ ;2T|5}u-?"ˆ.3xA@P@3|YaZ\v__>'Z)PI_JbռGUO j|=][5д-n[oxC’XI6.\ZwECƏ~3k u < J4φz?= <+H<9j=?I/xR\6sjmE-Ɵ|[TR+:~F"׾k:֡.HA/ZZϢpe6CijVV%ǔ~,D/\jG:% 5JF$o%k^8YFiO ?gjfZ/|2/;Ȣ~]Ѵm~?OmKмk"湠_>$\kx|[Cx.:+ͪGjڜwzvo XyV|~4/~3^|;<-L.oaڧYmE-@r~4Z_|97]YЮu:B xRb}k}. BMcRX8[şH'S& HӴ}Că> wĭk 3cH5?<)g/LkWVzeվ@;?|S'N/L`:j?Nlåxm}쓭A͞tKb Ǖo|/qj_|}Ou]wZgB>E\<%KY_y[_&m5J;`jq:i^fwx[᎙O\p5IOxVh_[[heI.(|,ڊZm/O45[?vxzV kkuQu_ qg#ID)|/qj_|}Ou]wZgB>E\<%KY_y[_&m5J;`j~4Z_|97]YЮu:B xRb}k}. BMcRX8~>ixw]j^4<5m?71o<uw/OW_i1jxHbFjʽL ?"_Xxßٞ%<ڿ[}OIjc8?yp[VI:75bf_} t  Z AXy:G!χo|gӵ],7*{VKh(|OŚN}--/{qkg'$_aNvL>x]<1]xTVmm?'[yT+Y~!U{?w WQ'}},i|eڻDg-k|OS+&mSW8C +~G,죿OYi[6jz;olM?{OଚO \xS4WF[`K>go3M l6[}U{cr1FH_'Ohv >&k^3Y5O jŞXi_g >0ee|'fH[&O:]I/5?7?x::} *Ψߴf־6X_ 'Öφt:Z^yXI3:c=:O֑x;_%G|} Q*z-Mu_ڞ5WVTbզӮ^"yV 㼿"E;wFmG&=Ý/B Ú|`sm6jqž5`q[`韴j? l]}'Va)ogOa k:ǟ@i|'x'C:Gm^jzfm9w;1xI/?On-l⿌ m<ˋxu#g÷}i. VK>߈|_/|C>i"lMÿ  ⦻s\h Ӭ|9j:Enm);GYz.|w_xG@tO8~6?~9|X|B<?~? }VU>#^|][>(0~^6EπnK⟇|'o:_Gټ'O\jgdtoY+|'}o~"|__}xMG⋹4/|O?rT:.N晨Q[wYg'rE_xsƶeż:wó>ga 犍jt<:׍Qs~`{:|Cx$|cIΣbv=so Sz?*Uhh~#t/_?Kw};Z3/>:φ<3 |Kxmi1Vz$6oVcӴok''H?=/0Þ5.-tCuk? &Xox"?^17ڍMYzLjzj|) E{zwYg'rE_xsƶeż:wó>ga 犍Ւ'-Yk> VS^״+/0vsZMzw1aM4X"ӬSO4(u8]5=v!@tq1$N P@P@P@P@P@P@P@P@P@P@P@VwX&ngӮݟe߷8OvO`ighW nO}ϷUfm;?leH7|Pͻjnio|Ϸk ?KGo|?H>:tɿ@_ϼk玠 (,0`MS.;On0F~ ~GS2[i|oh> DUֽ+{O3Jwj:smox]`w/1^&SZŸMON&o 5=6'D[z7Vc,+⛝oJMŶ ;oi^-񾥬k^,;Ö'oj$&Zk}l67b ^9wl'`V <jKG"mA[X K]jnڥdXrְ|'7{b7& nO *iPx~0׉|Gou~ՅLju}rTSN=jxEc+խk/TyOOm tkAxZ 3ZI^+kEa^VYW7:ޕ_Wl>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm oa[|sqپN4? )Ԭ\x;GՆ.E,Ʒ:W.'m*ݵKDs奭c??_>5ӼuCşX׵V7J5t_[3i^'bXC$h~ng&7+ZYW7:ޕ[_O>iVҼYKX|[wm|O? w*ԆOk%'-FVktЏzkψ|{h(cx/\R S~*յφn-VO4i^K_ .+G+Z z>$ O?'k4cMKto-b0In޹U䳆Pcx{]{VhV<#z D ?^Ҭd׼E:~߉"KӧsoۿKXޞ;s8i&3A¾ >|]ƋԠ:⏈CŶ~gA.tۘ??|п9ikZ}ῄ~EƷxixԴK_^-񶥬k>,;v&xR:?o-,+⛝oJMŶ ;oi^-񾥬k^,;Ö'oj$&Zk}l67s[Ǒ|Oau {h#uG➻VMnYo^ ukNփ-\˪EĶk.x{_x?4 /C?/=m#-#_wuo􋟶^E۴BoDk¹6IOZKWO _%~:NǾ0[t/i1GimJztsm“jڵ/JucQ_|aw0_55g&cga;$(ӡ_ïE`uաXWRhzvKK$~2vDkM 667:_YV_mZ\Eq&k?4K|W/k!M[hм#mos;[`пf^ *C(Ү58w^FI}/e7W03kq֚mmgklw#9L}K(xj>7"z49ue#>qտ.~y S~fҿAmcпf^ *C(Ү58w^FI}/e7W03kq֚mAi_EWFKQxRA<'>Ktϊ?' o&zh]ޯtlU⯮G_  &?ko-_>$6_:ƣnaпkkMSvIQ) | E,_.Iè'F\<×0ޭ\\K/jO\,zܮkXO~?xŸ w§{7 ѿu?JV/kĞ!Yn۫CmڟsoGvu-uhmSm_eQyK/7t}:%|gu= $6_:ƣnaпkkMSvIQ)ӡ;(x>[?y ox_W&Wx^%^$owZmF;e.Ǿ% Ơ׊|#_I-|@t{8fPF2Iq*oGxQGy4kM.I~x  ^WY%z\{)WOg X>x/Tĝsx?{ƚz<ɣZiv:NX-W0_4-_:J|p7+y9=<`r#_W>)6[ij)!^)5ws]s_m&6vڟcH-^o_No:/-/5?bx|Q ܖKuvA ? zFsU}#OKmcA@P@P@P@P@P@P@P@P@P@P@P@P[cx7M_Nv}~g>s}Y=o#[m3^;_|n>o{>/U_:ٱ#ߍC6}o7ͽ>}4^o-TAwe ;k&ZE \g>: (>f''U6L?CDP@Olp12 `0{pPx_՟v/G&G[UahCY'ol//!ki70xvFk'!LJ{yXYk x]?/i:E@ŭ|W񭵇qov϶{;Ya2`jt9?|S'N/LGT_.K{`|s|4oMD3lQiuyjZ!ox?/ 9OOOW:k5 <7o귳=R>iȴO3A5/EtR- kyXyVo/߀~xyZ湥Qcg͖bZh^\ ?"_Xxßٞ%<ڿ[}OIjc8?yp[VI'ڻDg-k|OS+&mSW8C +~G,죿OYi[{|eo /IR4P σ|+Zǂ5MO+O x`Y;S6՞x5o8%&|_㟆z4 Zgû?qxVV-_WOi1Qf!!moyW{w-Yk> VS^״+/0vsZMzw1aM4X"ӬSO4(u8]H ( (10:F8$d6 ( ( ( (?{wOY=FFGu@( ( ( ( ( ( ( ( ( ( ( ( ( |ofַ wnϲۜl'vOk'mykwݿ|3}_?o{o}M'>O~gC*o{6[62Ҥ(fݏM7g﵆M%7>$y_Cd OHrΉ5PP@Pd~0֩}3H ( \v9^ky=AoºG=+O]k~B6ڻh0M=E_K>| RmS{VSCe _Sj>+? {ƺ濬dLl?>d[E }:g)|kxڇ?joj fҼ cNıH?MՕɷȧ+Z;D|}_,C5ǂl~%\<5O]k qx]?^$e: _}JK#o!9ikX|~,ڭ߈t Sng<'t+GJYe,4 Km.S \3ۦ{:x C&?`%|G_'ͤżkcnsȣ{v+Z~;|J:XWoi/翵,Rcoe7Z+/]m&k]{|3A¾ >|]ƋԠ:⏈CŶ~gA.tۘ??|п KZ:|sqپN4? )Ԭ\x;GՆ.E,Ʒ:W.'m*ݵKDsKZ!h?e»㟌P|^(8q10:t=1׵:cN9 dOAy10:t=1׵=z dgNϷ4t1:`pO>@.?<㎼t'c$H P@P@OϧJNӮ1^: |=n\= & P@P@VwX&ngӮݟe߷8OvO`ighW nO}ϷUfm;?leH7|Pͻjnio|Ϸk ?KGo|?H>:tɿ@_ϼk玠 (,0`MS.;Of@P@^g]ᮑ ~1Xku4w>n[;Z:EŅQZ?H|9qV*ZFh >)|xw&?tU| 5 haҼs} I f:Mk|ʷ< [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJ-o+*cӼ;]|mU@>Q#P?O5o^4> w #Ӵa :M ˂^$9oB½K+s37GWyI /7-[w7n `j3;Wh#<|,exocjd<-xJz|Ua|b2(w__K;ym kool> :d<+B -ߌ <}F¿ S_3m7k^0ּF_>^mEU7z":4Fq3,D/\jG:% 5JF$o%k^8YFiO ?gjfZ/??SD?qR],Suq&Gø.4ϫj6'΅u|=>xKǺs\#pMkv7j8[şH'S& HӴ}Că> wĭk 3cH5?<)g/LkWVzeվ@/?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzZV*c+_'Ծ Ο>΅u|=>xKǺs\#pMkv7Մp? ?lsAͫkd⭟Z>M?Em6aϟsE=.8?j-o⏇x3:G;~ښ#l$\kx|[Cx.:+ͪGjڜwzvo XN<}Ɵ|[TR+:~F"׾k:֡.HA/ZZϢpe6CijVVS/o_m~sž.SP׬>_ VmWǾ>*_Cea4d:QOx3Bu+E-tR-(7Uǧx_ }v % |Inj?ao>ZwE̾T/o,bb\Ǚxwv4@;Cqc<#Q/>ƿa]U4x↍o }A^oU][ŬՒ0~ iŸI/>so-{v]j~"tů-/\i6ƥo5`q9Y4 Cе">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿<-g:'|-ţR> /IR4P σ|+Zǂ5MO+O x`Y;S6՞x5o8?> Ӽ ;ŃX:?EӴ_[0^9z_xC$i~ng&صCq[~4Z_|97]YЮu:B xRb}k}. BMcRXyN?j>zcx7 RmSǞ*մ?xnx.7RK 6,6[K᷍5O\|IмA~|N׼ɣx"?^">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿<{hd|_㯆Z4 [imOnj[?-+LZ^=&m4xѡ8C<=Hga 犞Ւ75bf_} t  Z AXy:G!χo|gӵ],7*6O#|;Wh#<|,exocjd<-xJz|Ua|b2(w__K;ym kf!oxwv4@;@eڻDg-k|OS+&mSW8C +~G,죿OYi[6Uc\gQ Wÿv=SZωZ׌&o=VMSڧ=NA4> Q_x{s  |G-xG|=W:t w6oþU%صi뗷ȧU|2/;Ȣ~]Ѵm~?OmKмk"湠_>$\kx|[Cx.:+ͪGjڜwzvo XyV:g=ڏ[/_Iկi |[Soxv_ 6 ηm=+څޙ[i%]NFF^,tnoۋ[9?"<9[k2GH;lZv eFՒg/Ǟ=~2VwGD<+eׇ5g:x_6|K>E^ƛ ,4k.Zl?_o'uuZ| ŭƕ.o߈75z~ Zżdk^2ft[oM(_o'uuZ| ŭƕ.o߈75z~ Zżdk^2ft[o4>~kA<OU-jN'Woo|J~#|L֮5 ]6_k#6Ex˦5oM@Ҋc1xI/?On-l⿌ m<ˋxu#g÷}i. Ւyÿu?gx7~ tDOÏH>nGŇ/#szۮ5hX:Uȫ%uox]w>/>!ki_Qw&}߅^S]|9_jPӴ_i>5WP|B"6J+cѼQ4"[[_OH+x̸Q?v|;{=֝0a|Q-Y.G?o~"|__}xMG⋹4/|O?rT:.N晨ȦVo~"|__}xMG⋹4/|O?rT:.N晨Q[~]6"wu?f{|뚏luϰi:V=agss/ p+X'e_wEi6#xΗx?|Es@0|HXX]tWTյ8bF88+7\/g_ x7_Ѿ|c];/ݟ^6xĞz<$'[l|h6_!/Lg3о ]_¿Þx񏆭'E%γg6yk|j)lzh7l|/࿍]]O_پğ;FNwyXYB\6V& }qxV}[Q<]w5>~ [c XyNC¿|NSeTūxmΫ57R{ᯈj+hZ<$Lm/V]?\[7's':w|}xc AWQ'v f+7oKdo /l]^yV~]6"wu?f{|뚏luϰi:V=agss/ p+X+??SD?qR],Suq&Gø.4ϫj6'crDg6⋏: o4xgFﴝ'k˩nf <{h5v+XIJ{\)Z$gtw>5OG~ xo~1ɤxŚxvk'ȞbW^x_՝c_%ji5{Bg=ε$׬o'~A5-:4sBZӅ _봉 ( ( ˺6okp<7v-"|o{GM'>O~gC*vkwݿ|3}_?6ouc/ϻ*GYmTwMo{}Xh?"Z?C~QAwLו9M`.}_ Ij,~6Q.ٛJ5!;#DKtk$d:ƭ $W0Mi}dm'--k'λh^O?V xgB/ t^!@Ծ x<@=jPn^{+[c?io"_G^?"z|]ƋԠ:⏈CŶ~gA.tۘ??|п KZ:|sqپN4? )Ԭ\x;GՆ.E,Ʒ:W.'m*ݵKDsKZ!h?e»㟌P|^(=q xSxDCῄ~MηxU?zR5kv_C\7?x\-o5>KmCm_5lP߇7~.MCJ5ǃ XirZͿ ?/!n6\>| 5ֻ߇5M_7]^69ikX ? [Ҽok*m=W|ok׋5;N寉ExZ?ktޟ_@F#oz`?.w-<䜞z0F96#"_ > ?/!n6\>| 5ֻ߇5M_7]^6ȧ--kAg_zW~m^-cxJo-cZfi?5xS'5_촻fѴ?1n0iFz|)Լwl|/xo>/?4 >u\(ҼCjZ֬l[cOn;woMFܞTҠ'a}? 9 m&& zEb Vխk7¾)<4[aOC6ZƵӼ9khk&VOk%ivͣhc+60<"_ > ?/!n6\>| 5ֻ߇5M_7]^6{N: דs)mu< Bxw/,Z]=qk{X/0j:^mn)bC vƆjF[oki:N*x>o hW7Լ%JYel4 G-nWPfMVQ 蕶/o|Ss^7xixmm+ž7ԵkŚxr"#kMCi!n`Qn;B6zt~ ,~3^|DDG3~0jV|7pAj%ֿ~)׷3JZQtXu?@ῄ~MηxU?zR5kv_C\7?x\-o5>KmCX|C_~x~A6oM u+?5aˣsÑk6 厇b, JmRQ2\oa9ikXO:OtxQ_V?uQ uߍKE>k#Kdb`ivg;oܮkAg_zW~m^-iv[mKXּ[[xcP|MKĞ%4K[|Oe۾a[/TyOOm tkAxZ 3ZI^+kEa^{+[`kSuOa7>$|Niǣs&iiZïAb\><ں-_R%V&V3x'ƞ>+xǾ*ռ{G.xOχ|7/h'5?N|]{At.⏈CŶ>n~#L͹CȺCbwAg_zW~m^-cxJo-cZfi?5xS'5_촻fѴ?1nYW7:ޕ_Wl>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm o`M"#:s|=𿉴o|:vO\ƪ喳/x_~'@\u.U -|47j]|?io"G^7"ZD0yg|߷j߿e)8leM蕭`_W>)6[ij)!^)5ws]s_m&6vڟcH-^o_N| RmS{VSCe _Sj>+? {ƺ濬dLl?>d[Ezt8 ou߈ڴ:*SMNu|y=id_Cus6ixF!;[`пf^ *Ck(Үu8Cu~O_[FI{)>3|uB7}j9CMAwk7:_YV_mZ\Eq&k?4K|W/k!M[hм#mos;[c?gk"_GCQ7Ѧ+(sb{6 o+7:_YV_mZ\Eq&k?4K|W/k!M[hм#m| O /x:4_^  =Q:^wM|U>mSzܟ[./YSo+z%kX<=/G|sZQ < oDyl|;[gvqc~^~,p>> #^>Gx<-ռcj^6+:JI/6U6ڍnZZ/'7{b7& nO *iPx~0׉|Gou~ՅLju}rTSN=jxEc+խk'&'ڮOj :_fxG|,σ4^'|W]xu4xC.aoW6*W#鯅sm“j_ٟx.kMgbu?$vv o_N#|~,߉߈ Tng<'j~gJYet^? ͥxj ɪ^1A{+ ֱ uG-ڟ2xՕn;3Wմ' Ii72V++RKZ]-kK>#ď|[մ/?dë+?6wfh:Oo_neFjWV7=5co9tM.<}oVоwi? xNM~IMm^XjZ\6 bՍ@ֱ| RmSV}3xWԼ[cuXɳN;$M|?|G} |-- O 7?5iؚ _O|A^u-uX-S6Z(/‹2X1|]񾓣!,<3;QOywG lx#LJ/a[J^#拏Ğ Y.ۋo rc_W>)6[i)!^(5GK41x\9#$6zt8/%QyK/7t}:%9]X^ G ?/!n6\>| 5ֻ߇5M_7]^6P'X|vgtk=kOҼ|O]]jgK{YjЬ=}/|-˯i|7Mjwѷz%kg?4+~..=׊>#k7h?|F4ۈ??|:vto+ KZ}ῄ~MηxU?zR5kv_C\7?x\-o5>KmCX|音j? l*|]M'_ |$u?/ x&fsmO]k(4zs>O:OxP^|a ^[?(]KE>kBGa_G.g"WV}tAq}-S=fA/Og҃GaA ; Fq'tNЎ9ikXw|7&.Xk|JwptmpcUO$Z~xw(׏|CM6I|CjAu{sڜVF@奭c+.7þg<iM{Ei#:GH'%X5-A#ך?IoXnZZ>==y9FǙ|6^k:qa?~$|M׼kGr5a?^Cuo ^WI$~[{)N+gzwr1y;J2B؝Go:m/5?c߈|O >KwvA!5-vѼ;T}H㿸F6 X?txrAׂ8 z Lu@㎘#H zG c#=zt} z Lu@OnӦ=z dgNϷ4P[cx7M_Nv}~g>s}Y=o#[m3^;_|n>o{>/U_:ٱ#ߍC6}o7ͽ>}4^o-TAwe ;k&ZE \g>: (>f''U6L?CDP@P@y-YAlbki}_V>M-ϛx&osqacmkֲx\xzwGϊt ,U_ G"/-tcm/!uH47YΓiu{lZ!96<)9 zQMsn@of|{-?VOC4+ g[Ҵ]bM/Rҋ[ʷ_~)>d'φ>};H4r[ׇu/zψ/lHlCGrൗe~O7%{?n]RAҴo |Q5t^%5-7/j^:Mw 7)[/2v$ҾN4з|2 ko}Q4j4ug冑vSK%nNK_>(*c/ǟ~Gh$/Ah1j#[sz5MfMԾk.Z{-d'φ>};H4r[ׇu/zψ/lHlCGrൗe~?h #nC _0o~^to* >nt_lͼ>Ku~,D/\jG:% 5JF$o%k^8YFiO ?gjfZ/K ?|o_~xĺڿ|=Ꮙ> G6w:4+YȰʅmLA+G~'G?>!g?ڷtm79|Sjx@|Co _;[iRZ_VN6*c}?_G]Ox3ឿîBeu 2y>O| 㨞:͜ڥ寄F)#3O6<)9 zQMsn@of|{-?VOC4+ g[Ҵ]bM/RҝbyV>,D/\jG:% 5JF$o%k^8YFiO ?gjfZ/K_~)>d'φ>};H4r[ׇu/zψ/lHlCGrൗZ"_k|sѩ$)vxgxf i<0, jLZ^~5|~$_懨x^x_Ve5 SÞ(izl|5]J$k]h& q[WO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo)0|V· K6io\eh7H~'mهKK'>}ϓi>T=>I }OjjA ߆:c;QI5OXQW:O<2"?q|N,k&-3w I!MP6&?j^4<5k|]Ĝk j+ơxo:π4:CƖE\<%KY_y[_&m5J;`jq:i^fwx[᎙O\p5IOxVh_[[heI.(|,ڊZm/ؿ{:+iWvӦGy8=nofxI ]i `7/*-y~)Ew4Mko:^x\HV5m[ìx*+OGjڜwZvm67UoF%k7O~ tx¶wv<"~xWg?O jWº4^uRŦK y3k|}~>h>K?i]|?i urz++xWF/ xP").u9u[OrX9R[A㟈~/+D?< K4y dcy?bտy,p~vk)E[N_'¾>Ӵ^6? lj5s{3Ŭ|oOiNXmS쌟lyMB8t;19`捾A< [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJvq[t#>->/[h:4kz_ ~#-#Y.~kSmR Hm^_;tW?'OQ|];7.t}+I~ |<߉|3=ŗ5,a.lu_h:MfWE[__< [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJ-o+*c+_'Ծ Ο>΅u|=>xKǺs\#pMkv7Մ ;N(_tm&_|Oo|9/9h7+ Njڶޝ[VU$j>%'7,< x#W״ Pѭax5h:Kմ? `kK}5dK%&|]㏆z4 Zgë7>qxV}[Q<]_߄^..xcOusQ Σy 'Jg,,ne.ye}~lyl~x^;i vvz+xWxs1մz~_Dl/-|"5oڊ[_>">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿K)/ |YΉ_ _hԏu?Kk;G">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿<-g:'|-ţR> /IR4P σ|+Zǂ5MO+O x`Y;S6՞x5o8?> Ӽ ;ŃX:?EӴ_[0^9z_xC$i~ng&صCq[~4Z_|97]YЮu:B xRb}k}. BMcRXyN?j>zcx7 RmSǞ*մ?xnx.7RK 6,6[K᷍5O\|IмA~|N׼ɣx"?^">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿<{hd|_㯆Z4 [imOnj[?-+LZ^=&m4xѡ8C<=Hga 犞Ւ75bf_} t  Z AXy:G!χo|gӵ],7*6O#|;Wh#<|,exocjd<-xJz|Ua|b2(w__K;ym kf!oxwGx+BY6Z|-:=RMSڧgŖW+ɬ#-_% ͤ.m-'Ohv >&k^3Y5O jŞXi_g >0ee|'fHٷl:c=:O֑ʾO |Mּg7 5?j<%q=O>*Ҿ1^gn|a;/O65ȷ-7 Lu>>_j~/xo~6ut$>ѼUQh9|lxO-4+ Zu;O65f1uzu_"wQJU3[?m[Ǻ?6)?<k!~򭴩-/ūMm\E8y~)Ewk3M{o:^x]5"c[cv>)^mR;Vӵ}jʷ?iV~|+Oƭ~MS x÷|u/?#OANuh>(1^.ZI-orw75bf_} t  Z AXy:G!χo|gӵ],7*6O#?~<ÿx:&6gn;^!5/*ˢGj~-mA>ioemg$GoRKM-e<}eO&}\=W[_: Zi_ a*Msy3Zׯ1w|[Hڦ.gE6Ҋ`eO&}\=W[_: Zi_ a*Msy3Zׯ1w|[Hڦ.gE6J+m?xƿ s]o֡|1kq|f 6ī7=j^߃eVo?<#jWcY (>ѼQ4"[[_OH+x̸Q?v|;{=֝0a|Q-Y.G;GYz.|w_xG@tO8~6?~9|X|B<?~? }VU>#^|WVN>(E|#|?OqHerh7__5WÚu ;E]Ɲc3Q|?e-+so4: 1xI/?On-l⿌ m<ˋxu#g÷}i. ՒsaF.G/~(Aw/c/Ci/4i(>!h[|iElaF.G/~(Aw/c/Ci/4i(>!h[|/ /o/xSWo1'Qk3yw72P Չ?By_mS]7~[tgnw6y;|ݵNGM'>O~gC*vkwݿ|3}_?6ouc/ϻ*GYmTwMo{}Xh?"Z?C~QAwLו9M`.}_yo ->upG;+/ Kj6t},g)|kxڇ?joj fҼ cNıH?MՕɷnWV%ޣbb<a*ѵW?]k qx^>4%Cq jqZ_Yy KZƆjE[?'j^!4=OڕdlЯoxKq6X\\PwMVQ-X/eklu#9L}KΥo#ͤW^[8džff"(6`xC࿅4zPZxwSgMOVz<LxWoxF;kNԮqn rֱ音j? l*|\I) KGe𷃼;co ix'nqnM_:,̚OtKW՞~jEZoKXZ+O<5xT𗆢:W/c?|Esir7^jPBV+Z~͞.}7E_𖛫d?GSd ]\k0߉.5OSћUmn n5{:߉|?h~SY&v .]ÞYYo,t;`-.|]uNUjc+ KZǶ:qy8>AS?|G} -*2x[-x GԼ=m+:Wuw^m%uhmg9ikXOox]`w/1^&SZŸMON&o 5=6'D[z76kZǾR$?|zoOň~MīwF5_Mu7LJ|AxG>$d:ƭ $W0Mi}d)KZƇ ^9wl'`V <jKG"mA[X K]jnڥdX奭`> ^uߋGV:}7RL:O xn#|, W6).UbrV Z ^9wl'`V <jKG"mA[X K]jnڥdX奭`X~N7SG| R}!YirxAbǂxF鶖3r-;,uK#o+_ ? [Ҽok*m=W|ok׋5;N寉ExZ?ktlct8QV?rNOs=G# o^+>&~^O>jW4-.O:lW^oXh]6~"4NE|݅dNZZֱC߳G$x!¨y'}ڷmKҼSex_|=o _Ӿ&|M=4|2ԯeiVZ\tد<1#0Ѽ74mEh}N/5 R奭k0F~ ~GS2[i|oh> DUֽ+{O3Jwj:}ῄ~MηxU?zR5kv_C\7?x\-o5>KmCX9cB??>"ix᎙ZpuH5OV> Xk_?[kۙy-|(|P:eklzoOlz`c9c4?~ _~&Doa%M*x:/ÚJjriǭOV(}NWVr:gã/c~ Ÿ ixx[Y|Ś ςuâk^EJw:^s{[.|X>9]_%zG=}?zV++SӾɠ[B8tOM!'Eލ Zֱx{_x?4 /C?/=m#-#_wuo􋟶^E۴@މZ;;s8h'cȾ+ƞ>*Ǿ*ռ}C.O xׇ쭼9/&]iޑa&GHҒ[E$\ivȦ0<"_ > ?/!n6\>| 5ֻ߇5M_7]^69ikXfPt{Cwg*;⟈6~6Eݖk#>&> xRoGk魼-qiwvfR7-}7~AC47^m_AYxNmJ?V7򰹷rKG7'w? \x&Uûk&|þ F|#iKcV ۘ&ⴾ6--kAg_zW~m^-cxJo-cZfi?5xS'5_촻fѴ?1n>N1|]}Q'kg4tAq}-k0F~ ~GS2[i|oh> DUֽ+{O3Jwj:|[c[.|X>9]_%zG=}?zV++SӾɠ[B8tOM!'Eލ&խkX ? [Ҽok*m=W|ok׋5;N寉ExZ?ktء/WnKmCX|C_~x~A6oM u+?5aˣsÑk6 厇b, JmRQ2\oa9ikXO:OtxPgV?uU AuW? ؖ>'#_YI6 ֱῄ~MηxU?zR5kv_C\7?x\-o5>KmCX9cB??>"ix᎙ZpuH5OV> Xk_?[kۙy-|(|P:ekl'ox}>oGxQGy4kM.I~x  ^WY%z\{)WOcwCᏂis⯊{[>ivҼ[}KXּYw-|O/ s*IMsD-.ٴm o`M":#/7ߋ&Gڏ=w:ݖzE[MsUtST kl-67j]|?io"_G^?"z$6_:ƣnaпkkMSvIQ)7Cпf^ *C(Ү58w^FI}/e7W03kq֚mm3 ou߈ڴ:*SMNu|y=id_Cus6ixF v~;UV_u}x0OxT^'"j VmTމZ:x C&~>ֿ`%[U5dÏ[Vx{Wu+'mVKj6o!ikXw!蚍5<7[?AO^%ׂ~sV!RMNM:OWVx?g7ğj> :,|W@ Ӿ x+0^v|A|M释5^ث_]ρ? MojZ|Hl+׊uG~=2áox\I;KhSbӡt_:ţwnw->C<Y&OIڟ|7ҾYx;Csip 5jPyn^r_itv~0z?ueg۽FLmImvkM̺hյ;mJT籵KZ:폈a#m O<ozƙ$4tѫjvڕ핍eck f-X]-kK>#ď|[մ/?dë+?6wfh:Oo_neFjWV7uu=5co9to/?+h~TխeW{eB񮹯?6;O$vDa7Cw;QQ:|)w*xL/_|otK zx]Gx[qxVҮ.%{f'.fKrnWV >L/_|otK zx]Gx[qxVҮ.%{f'.fKrkXN>t'R9T]7~[tgnw6y;|ݵ؝7nO}ϷUx]{mighVm]{#f_wT~7 ۱6}zɿD~Q߱d+sLi\7qƾx ( cT2 ߆_yZ/ߋ6m>'7z]Mw^4 ċohuEyH[SN-a Ǖo[Ѻ|}>Zc(l xD"į $+0Gi$/ :ޫa%֒ȵAkG׿%g=]]ចŞdnj|5mQ<).u9K_[SQKc|/@cmEj>$7_57`tf{"_*ᷖWڱ1O+> mxSs֢ ݾcͪZ'K &LJ)hWvηhŮ^ʷi5W5wKs_,?>ē?ϋ/{t _A~#NѸ!sŭ[uyi/T4.|5ԼixjK?\xɼQ źڍ *+ E\<%KY_y[_&m5J;`jr~u›( Ě4-[ nu]OT݆麗 |EռQ] Be<)%cizŮtZ!4?> Ӽ ;ŃX:?EӴ_[0^9z_xC$i~ng&ص@ʷ폅K7\~gs}Iҵ< ;|_ˆY_j_mG%&|]㏆z4 Zgë7>qxV}[Q<]v__E|+;@Eo# xY77xTEĝ ix{z<3ǣ]iwN5Է3_=4I_YD|vs\\)Z$gtw>5OG~ xo~1ɤxŚxvk'ȞbW^(|OŚN}--/{qkg'$_aNvL>xMY.Bx_՝c_%ji5{Bg=ε$׬o'~A5-:4sBZӅ\#iP@P@P@P@tc n` A@P@?g`׎` ( (cIAo  (cd<` ( :{c1W0sۀ ?{wOY=FFGu@( ?L~ #߂qc ( ( (Lg $zp ( ( ( ˺6okp<7v-"|o{GM'>O~gC*vkwݿ|3}_?6ouc/ϻ*GYmTwMo{}Xh?"Z?C~QAwLו9M`.}_F= x+BxS։d|-:FhpoFVխkXN>t'R9xKG7'w? \x&Uûk&|þ F|#iKcV ۘ&ⴾ{yY ~,@giW~3/KLz?|]k-S:E֞4[9uM:O2މNZ6D}fA/Og҃GaA ; Fq'tNЎma9ikX7¾)<4[aOC6ZƵӼ9khk&VOk%ivͣhc+683aG}͗OKOe𷃼;c}s>/$,.|⫭{Vf>Rus` Ij+׵V(~5j+haAD@4.fFwuerm+ZǾ:qy8>Bk3A¿ ~3iAxmx0 ]vs miMGɺChG@{yXt/x7 qW?.?SGZ~xw(4eC9.`Rp?cQw7,m<=ih <=Dx\|C[D_KXh =tGi:+ KZ==遌:wq#c̾x/Tĝs|!;_-z]|: +}iս+x}ad/ۍvKFp[?Щ5Ӡ xZN CvMF=iѭ4'Ay,+/x}]d/q]>K8mMǧ?rNOs=G#>ρ? MojZ|Hl+׊uG~=2áox\I;KhWӠx·?KK>&؟4yxkw%R]t/=nѼ;\U}Hx[|XP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@oulߺ6y~n:[sD)md 7nO}ϷUx]{mighVm]{#f_wT~7 ۱6}zɿD~Q߱d+sLi\7qƾx ( cT2iP@ޝ:cN( / ZHKj5|:[k;7D--~VM"(׭d$>o+k-}!tok''H?=/0Þ5.-tCuk? &Xod'φ>};H4r[ׇu/zψ/lHlCGrൗe~{ ~!>/,xKVGۂL?'Ohv >&k^3Y5O jŞXi_g >0ee|'fH7-i4 DЯ- <}xF S_2k9~0ּD xQuO qg#ZIDl^^F[ş~i$)~xZ׎< |0, jKN-/CIxǺ?~'x)ux᭞M#VYM⏇p\h-՟VmOxWOiQ_Qg! mo8Cū;r??WJ>4? ke.{k[IXN?f,5)kZui! ~G$( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ()jL_ketP gjCAO~CK^ǝmlaM#S?M w^7?cGG_ֽn[ Tnۻ5A3ye~6ij$? ?ֽ;FK[h Ú~MGLף4mnMG\kIK43sw ưywԉhQ߱d+sLi\7qƾx ( cT2iP@P@+?t@/2'<-xWHǷ~ik~Â?YYx[?F[Wm ȫg>ρ? MojZ|Hl+׊uG~=2áox\I;KhSa7CO:OtxPgV?uU AuW? ؖ>'#_YI6ukX_|zoOň~MīƩk{<;kĚlWաW0OCidm'--kOW/ŝw;Ua~ Ԭ'΅}}S^_ ,C|@ͥxu {touܽV]=㟈~8,D Dylymտ.~ysnҶz%kXt/ٿׇ~"V_:Z*SM7]e^mPe˭kBos;Xfh4W~2]ï:`@ּQxx^oL7_Es't9ikZ_/WnM;B:KZǮ·?KK>&؟4yxkw%R]t/=nѼ;\U}Hx[|w(Qv@`om ( (s289#1ۯF8 @& crsےP@z{pr nH8S=8Ӯ1^ׂ#zcy om (10:t=1׵c`t=Hzcj(DZ'O)R|1k!47TO x_ӛQ[օKj' ƁU0s(JSj)^==4_uC''c㞡i U{⻝bHOẸ5WZon[oi\%FY\<*TAj-o}h_c(@{F8$wBQD}]A߱d+sLi\7qƾx ( cT2iP@P@EVu|=imWSKmgsêɤ[XXe#×be`x's':w|}xc AWQ'v f+7oKdo /l]^n<}C!|M_ x}N}C^S\A~,u[ٵ_ DᕄE5[?vxzV kkuQu_ qg#IDl'S?[ώtOZuϋF|_[i>A;V1jVvl5=2j qizK=4M1;K l/i?W2o|;@noxzxJ> 3h|ǕzE$?|En-R}K௏n^]WZ߈!{k1k>w5>~ [c6S)/ |YΉ_ _hԏu?Kk;Gw5>~ [c XN<q[:4/ڶIq~*!kCWf./>MS@Rk/_(G'<|3?;6 ~GFM&,u=c?Fӵ_ Xh>Do/x?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzZVU_WO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo ǔ< `+_6Q?5KiZ Z꺞 SYu/x梺mxRK 4eˋ].赾Cpiz/S-Igt .&/kEŴ:ǂ|Sڤvwk0Մʷ> iŸI/>so-{v]j~"tů-/\i6ƥo5`q96<)9 zQMsn@of|{-?VOC4+ g[Ҵ]bM/Rҋ[q[zwh7l|/࿍]]O_پğ;FNwyXYB\6V& lyjz;olM?{OଚO \xS4WF[`K>go3M l7-=35Q k<XxNW~(h0E\<%KY_y[_&m5J;`jp? mo|sx|Z5#O%#N 7x,#Tؿ3a]Y闃VK@h9N;>ӼX?1a_P];EM?N 6ym.m[7UWO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo ǔ{ޟoo:gxTEĝ ix{z<3ǣ]iwN5Է3_=4I_YD|vs\ ]NH瑂3[|o+"k/^_i/Nc _ɹ>7'u,6FOXܼ ?Y4 CеE\<%KY_y[_&m5J;`jQHǺI ]w:kexHնxżS-X<]|AŨxUfG?!m*=3=? _?WafxhjoNvL>x Y.Ax]?/i:E@ŭ|W񭵇qov϶{;Ya2`jt9{hd|_㯆Z4 [imOnj[?-+LZ^=&m4xѡ8B?h #nC _0o~^to* >nt_lͼ>Ku>|wQh~.M?'s^w5xs\/.oV?>-<c#mN;;Xц'U|_⋯jhg~A >j~g kú=Bzww6zv!Z{I{aypZ-e~Ǯ\gQ$O~?/=?9Q^_?#ռ{ciΫS:J7*JZuS*a>|wQh~.M?'s^w5xs\/.oV?>-<c#mN;;Xц<}3mG}¿j4߅>-)<;ymwRB? /og[M{BLխ.'}x]?/i:E@ŭ|W񭵇qov϶{;Ya2`jt83mG}¿j4߅>-)<;ymwRB? /og[M{BLխ.&h Tx¨/ j=xUN] %~mIi-Zkm:<}ߴG*< xOT|Cחooos@ wʶҤ5rmUoF%k7Ozkx/OxGѤ7z 6gZK"-w ;N(_tm&_|Oo|9/9h7+ Njڶޝ[VU-g:'|-ţR> /IR4P σ|+Zǂ5MO+O x`Y;S6՞x5o8]QuVM o?[izVMv?oG~'G?>!g?ڷtm79|Sjx@|Co _;[iRZ_VN6*c?%&|]㏆z4 Zgë7>qxV}[Q<](7UyW/o~-tMm]xs^|CkZWi SK9&?z\嬄e ~?[>/'mOO< J|;þ7`m|YEjolK?  =߉Ѯ_yW׃"k/^_i/Nc _ɹ>7'u,6FOXܼo; ixN4;9^.xǿG?<#iMS߈6![_߂> j7i6>f!6ZGI'X6/gyXG*ZFM{^Я åsk 5 ߇ņ4McxNM>>Сִv~kz"B ( ( ( (24Mct@qKY3᳜8'4%Ttks⇃<-~ e x}FW/Մ06X$[K{๵+[x.mfxcyqo4Zy']++;Esquu;G> fh>"A(Rt?e>-=о*%]|;:6|1'kw<;k>!&$!5hn? mN+K#o rֱῄ~MηxU?zR5kv_C\7?x\-o5>KmCX|?F~\[w y9=<`rl-Gx;D|}_~,C5ǂl>%]|;:6|1'kw<;k>!&$!5hn? mN+K'NZZ77¾)<4[aOC6ZƵӼ9khk&VOk%ivͣhc+68-3`4G}͗???RS3x|+_Z<|_h\}*|QcxOԵ{X9]w!蚍5<7[?AO^%ׂ~sV!RMNM:O'+Z/o|Ss^7xixmm+ž7ԵkŚxr"%]|;:6|1'kw<;k>!&$!5hn? mN+K#o!9ikXN>t'R9x7:_YV'_::{.S=;^eֱ_`3|u 눼R= Am k]~'jמ"t=SڝT|Яkttٚ _u14X.5|F"-׆: '?vCf 'vl'--kXXfϧ⏆:gak{5 <[\o?`K~$SMmnfB/t~{+[c?,+⛝oJMŶ ;oi^-񾥬k^,;Ö'oj$&Zk}l67b ^9wl'`V <jKG"mA[X K]jnڥdXrֱG/t:`ڣx?' f}'#Y$F<%wVwF!]Z77¾)<2[2 ږx_7ơz?'> TTms U,fM5W‹ è V|6^k:'5yFt [X`? <iu[uDzt,?A7+>+ƞ>*Ǿ*ռ}C.O xׇ쭼9/&]iޑa&GHҒ[E$\ivoۿKXӸ# .7`"v>dfh4W~2]ï:`@ּQxx^oL7_Es'tا--kX7¾)<4[aOC6ZƵӼ9khk&VOk%ivͣhc+6 nu+>Mx#v[OX|[yg-Q/ s*K=oDK}C_x@|E{#uC⦹}UfO~ }k|P΁-[. 2Zloլ|?io"_G^?"z$6_:ƣnaпkkMSvIQ)7C|΃w'0W7ߋ<'|GkZԟ<~Gk=oUtST {l-|67ڵCl&>v,~ |V𧄴_G?|p0i&xJ^׎Osuqjzڥ|5euskpl鱿w!蚍5<7[?AO^%ׂ~sV!RMNM:O'+Z g]~/|NnChzu+?<3_xT:W/aj_+? {ƺ濬dLl?>d[E޾|=|DѾ$8U~V]|!?gKOYfKYϊneLO |8?ohڗ x_zxͶdn lmK߳G$x!¨y'}ڷmKҼSex_|=o _/$,.|⫭{Vf>Rusam,+⛝oJMŶ ;oi^-񾥬k^,;Ö'oj$&Zk}l67s[ k]~'jמ"t=SڝT|Я[U5dÏ[Vx{Wu+'mVKj6@奭cfh4W~2]]]{(>|F$+Zn.: 4i7HtV--k/xߋ7 qW_ j ]oÏ|I/uZH.n`Sw  %?ukoM_>$+׊}C^>*\{zֿdl?MGio Q7w> [|w#LoúΛq/j=G_44‡&Ğ*sqin-тWV'λh]]OOV xkA t^ N\'pl[XkeW9]o񭟇|yx֚ GuG:ſOJejZ-GKѯ4=H7ܴ}s~N~"6<᷂O \|I7~#N׼iǣ<5c?<Zsa/OEs=g zO$n[;Z:EŅQZ?H|9qV*ZFh >)|xw&?tU| 5 haҼs} I f:Mk|ʷ< [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJ-o+*cӼ;]|mU@>Q#P?O5o^4> w #Ӵa :M ˂^>$_xⷅ|Nk-uKJѼ5F>סx\_ؽiz6:0\[locпh|J_:{ߌZBx᭿D|UTYÖE mM,Uu9-|;[XSӴKn<}¿|NSeTūxmΫ57R{ᯈj+hZ<$Lm/V]?\[ /CӼ;]|mU@>Q#P?O5o^4> w #Ӵa :M ˂^4 )0a| 9G CyѼso7|l4oٹҐ^\Y}6-c_k|sѩ$)vxgxf i<0, jLN-/C|/@cmEj>$7_57`tf{"_*ᷖWڱ1h Tx¨/ j=xUN] %~mIi-Zkm:<}Ə~3k u < J4φz?= <+H<9j=?I/xR\6sjȦTϕ< [_𧋧5+E5τ}UUOXM>S Ю5oJu]7TKJvq[P_k|sѩ$)vxgxf i<0, jLN-/C|;]|mU@>Q#P?O5o^4> w #Ӵa :M ˂^k+<[ώtOZuϋF|_[i>A;V1jVvl5=2j qiz>YQLZg|BÚm{M}[Ŗ~85OxS}vV^5ַyw=*Hu|Ǖn'_>">Wt7_|E|.t+C]oΐ^=ԵEE˂m?PXԭ㱿<q[:4/ڶIq~*!kCWf./>MS@Rk/_(G'<|3?;6 ~GFM&,u=c?Fӵ_ Xh>Do;||υm3'774=C,qjFuO쬼kcnzU#ZGA6 Ǖn4> Q_x{s  |G-xG|=W:t w6oþU%صio83^i ]w8kgxHծ~):xugյ}+TWxHG!F L]v?sRsC< xI-w>oM ?Xxg[ooM@ʷ8? y:MzMRV?:xnToK"?ykwxM1YtrK-o^_WO-ŪO|Ý?M#x}uk P{[}s u-f-}nGO!5+xo ǔ{ޟoo:g7'u,6FOXܼocr|pwc$s|o+*x?/ 9OOϨkWZk/v{6h/!2}2(<]k:ޕnzo+ʷ,~>G}+Z'|]Mkⷋ0|+u^ 4~[|E[:G'G6aqqڷK.<=- Yd6FJ? j>/%ww>EΑ>G+>;7|%x]6Ն:Mcj5κ5o8{i'kBkÞteu7oX귳j=R+ !Ҋx݆Z.kꗩiE`q[u~4Z_|97]YЮu:B xRb}k}. BMcRXyCy~)Ewk3M{o:^x]5"c[cv>)^mR;Vӵ}jʷčGĿ>$  0Pmau߶h 2-~l?n?Ev (~^i ]w8kgxHծ~):xugյ}+TWxHG!FmUP?iV~|+Oƭ~MS x÷|u/?#OANuh>(1^.ZI-orw~]6"wu?f{|뚏luϰi:V=agss/ p+XWciP3SH=8A@LO#H2yP@P@P@P@?g`׎` ( ( ( ( (?VV`@Olg،s69F1fi meloVy`K>٦\F{K׶Cwi,ַ(|?H>:tɿ@_ϼk玠 (xJFu|D|B "Tҭl`>t8rZFbk?˚=|!X?.4{A`лO\#Bz?sG$ D??X?.4{A`лO\#Bz?sG$ D??X?.4{Ac'N,#O񞁮^e5ӵZ?j ~6Q.ٛJ!;#DKtk%j j ]o|;k>!&$!5hn? mN+K#_ oKZ^Y~-?[J/I<3_xT:W,C|@ͥxu {touI[ٿ?eklu߇sX?!蚏 6!ᑬymՇr/.o݄WO8A{+[c Eč2ww7~OPDŽh´ x‡&Ğ*sqin-с/$6k8i/qM'_ |$uKoxv灥|^IY\'[}_WZ,<+:}ޥ缿3c>/ŝwUa~ Ԣ:O xn#x0Xh\\PgMVQ-䕽 V߳ <]x'k)-7]|| YizuqXx~'5=OEmR>^o$W5lo ^:3|?h~SC.?4txr-fxCZ[{.~ۗjqޖm6: q}/y}qv~|FK>[U5독Iog-R^[WOyڬwmաmy}qޖ>+_>(Ok;Gx]Oh^o |25=;4#H؞mn^-n_yc?.4|"lx j|},C5ǂl>%j j ]o|;k>!&$!5hn? mN+K#_"ֱ_ 7~ aCJw5aˣsÑk6Ʒ:W.'m*ݵKDsܿV$ kX4OzƋg]~ bnChzu('΅}}S^^ W6).UbrJ{+[`½cLJt?f;7R]ǃ XirZͿ`厇b, JmRQ2\n_yzZ_ gs];u?x/OM' -S᝗!YirxAbƇx7M񟈼Mo%acY/y˶Hֵ`<+⛝oJ)_Wl>lW|ok׋5;N?5xS'5_촻fѴ?6ܿV$81:k#ߨ+_a 0=SouI:y8| G ,|?kxᖥ|34-.O:lWxF鶖3r-;,uK#_y)-kXw?> #^ -*¤Ï[Vx{Wu+'mVKj6Ǽ_8@ޖ wAGOg׊>#k7h?|F4ۈ??|:vt_y--kw0~^O>jZ;/CJyH7,4o .io?x'"SKnT=.ڷ oKZ9 3q56^7O=OԾH? 폅K?2x@N'Y[yWtKWy˷gkZ}ῆ¾)oxixρ xx_[_C\7?x\-o5>KmCm}o8An޿c&J'> ޚuH5OxV>`X#O4i^K_ .+KoOC+_cӿ.AI1:ky9c4{DZ?FmU5KOj SI_gO|,σ4^'|W]xu5|!c}0F%_]mOXN1|A}Q'kҠ'a}>/x7/7 qZk&||I/uZH.n`SkHMkXg-C]|?|⯉^ Ӽ/|+C~ vZ7|3ZP]JYUntyƁ]]o$ V/847-E~AC4׾Wj>kVZ]+;&~) 'Gt /ռ6nhx;ơG7'? \x&V~ƫIGþ F|#iKcV ۘ&ⴾ^m|zZ77xW7:ޕSO 6|1=R5kvkkhk&VOk%ivͣhm϶H9bzw!/蚍5<7X~& :-M'ٞ?iÞ > Ӿx+0^v|A|M]ޯt{?}5 Cmt=&u D???˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG O{H,?˚=| h]T'.hw@=ShG ono-SDԑIeZH5w+3,6sJp-uyM20q.=ݐ񮵋牯<]^Gm..uiwk]JM5_L ԖLin$xlvhynwy}t&6#N>$y_Cd OHrΉ5PP@Pیzx'f`EEnnH'=8jPx-4Yyb:dgz4_oƏK/獷U/cP81b_wR͠LHQh+8p}?%om|HmSTK!,r3ƏGtȍ4]˶<$ ƏGtȍ4] V%UķQ2H|À[ 󃟨cy y\~Q@۞޵;ϯDo+}#@s\WFfAw k[MPwY sc;ϯDo+}#@pf)ī `Bs ߊҿ"7>ƝWh$-6͒{-$t࿆0?ҿ"7>ƕw57Ev0znێC{ Je ooc\{W_>ƏGtȍ4]qIX3ʒ qD/| T(SGtȍ4;ϯDo^]B"Y#XrGNvW[[HMŲ0F@qӏϥu_hwJ_EA7ky Q*P9;pG*kaog& Yda q@Et_hwJ_EԚM @"&I9a|"܈RL_\ӵt_hwJ_EZ(Fx(9[. e2Sn;ϯDo+}#@riGMoo",[*P NNJuXZKYь,hL)+Б+}#_h?NIV%$RkEnc,Cg$ǯ#]o#WF?ҿ"7t%-5l0JĹe,9vbu5Gq&; 9 =s]_#WF?ҿ"7tvܲ.G;0?V+}#_iQ]#WF?ҿ"7p8J+wJ_G#WF. Ew_hwJ_E(+}#_h%;ϯDo+}#Gtȍ4;ϯDopWw>ƏGtȍ4\ҿ"7>ƋQ]#WF?ҿ"7p8J+wJ_G#WF. Ew_hwJ_E(+}#_h%;ϯDo+}#Gtȍ4;ϯDopWw>ƏGtȍ4\ҿ"7>ƋQ]#WF?ҿ"7p8J+wJ_G#WF. Ew_hwJ_E(+}#_h%;ϯDo+}#Gtȍ4;ϯDopWw>ƏGtȍ4\ҿ"7>ƋQ]#WF?ҿ"7p8J+wJ_G#WF. ]kRۑ3fDA(#nsV?ҿ"7>ƕ-O@.\*q#{gzS⇼ҷLxFpRqJwJ_G#WF4֣Y^-ifӤw_2FeZ56T5i3\C [<*HItGtȍ4;ϯDoA߃7LүD ǹp\;G>֐e޹wyuqT>dpǷk*)97_hwJ_EЎgK𯇼?{x%i^xHs99$sPZ,cAJҿ"7Y[3S=cFI#EzuQLZ?"n?֏K +KO,meegxQ2Hv_DV$ðrDU+ Ema9"O*;/IQd1h;/IQa<?,-a<???眿'E\Ţ?眿'G$⨲ V$ðrDUAsðrDUv_.b[_v_yTYZ+kyTa9"O* Ema9"O*;/IQd1h;/IQa<?,-a<???眿'E\Ţ?眿'G$⨲ V$*+MIxO% ܒù>Ȩ.5Kn9ݡUG*o$RNSF7J Ă)\1'c9@|Cco&5 km.Kq:ssע ykxnW|Kpʒ~a:ze4KSӭ{) r$R h/ڀ4rOhfݱ;p*@4A]aΛmKyVE qO8# TU/3O3FC}|~﹜NjjqlZGu%ί$J돘sz >pϪ;Hq#m%HbzddZ4;جuH.Ǖ8Tv´o,ŌP3@[ F̾mK4P yHԲ.B$HT27QDD4\ओm 9༵a1pFNC@QPSkȹ˳˗x E yjfG"qmڧюwQH ״wլZkUg\!hUN랕{?RmBeTц@B3 q05i~> |1l2+T@^(OhRIokZs=<HLbh8 :*dqGU(Xq4$Fgwq>lIUX$sl{hmK[CshHfB9Ò$܊ER]cLkQ7rn|X698(>N''$`cOsyke +)cUc Y|OGqj~hti_)|"qƀԬZ^Rib Ӄ@X˨I{l.-U2"\~c֫'t$Q4 ync?<z@tU+}cLQE:A89Twgf~@TU/3O3FC}|~﹜NiRXs{l!xZd2ֈK.ONGZ״y : p*aqk햿boa'Ǘݻs -Cź=ڜFdG[̬@gSF3u&˧ɨGXMKC7=hBIiZ{_2={9cĈ0,7.c \Ȟ$Ф4ִ{y .rFџq@tU/3gѴ矞y}3'u$J$r#M@ FlCG>(EPX~1GR_VaEKQMQEQ3q4EER?QTtW?Vuޫ{1}$lorxz֎ {/A{!k-kT9I7Dtk?޵--OWү|>WiwZOHn ) yںTk:J7ɯ!G.QHmSܟZY:eK3"ot2T@a)MuA)[Mr[+kw/!X-ShdJ񜚯gF?Юm>-ya|wqp];%{B>cA9 Un9ç^%B1PX;+g8U#q.Xu瞶[? k,\ X%,P+HPm9^[Vژ1^]\Z>`C{JvvgvuTQp8]'N4f}b$#[lH1|,Qojm\kvUK_2O*Q }vqfiof{+97vy6 ?26-Z-oREي{kxkǀ4Ń@Qs^}:mڿmoi7ys|?ǧcOEON$I.F$ynıy7dgL IaJc,Tg4k2^1ؐcogmx~{컛&h :Bo$9bH1NAʎb8 "5UP0@cڝ&km]xPą`Rƶn3cy-W H#j#hbCA W l!sءcDk\#t.FosSP[w(̬qBa[jzlDFnt q!<-HZO{EounǟER ŷCkzBG!owt[EpZB]h,ٴ;˖A8佻r/  p@Aմ >h-2Gd o[#9ij4:4FPI@Y>.1;x?]Vm/c$[\OR*ޱRfx~⬪ V+ߜVʦs$"\ˮ}@Atpy9JF*[=SJ&;vP` JuBl#ha; (3@J)p\\g)p kz5݅{?{M*M7c2y c?0Rp28[? k,\ X%,P+HPm9)X&ao9o*䎜s Ro$hoQW#@]?ɊOIQ;X;I^TN*$uak!naPͶS\p̛GQRS 149 Ȭ!QTHЕ\svz}!UKI.f ]xHp ($mo I-< H C &?(a0IvPme;{Gmt${ac00 UGL?~_Nݹ:z(,GGm z2-'Fr €Xp|6U ݮ;;1#\:[(6ȡ) ig!Yfcx䜍2!ӘxS{tLp["Iw挞ǏN8kUtTx2pi-E嵒,寚8I}B41\A$ƒ"xC+) @~YOmͧæơ-xA-ż#(۰o [Ho16O<4 h>;7sQ@eցjJm$\+v8#%նwtKmo,m@~B oQ@Z"C=bç]-3s| -h".ws>3^ykۉ+g){jhbH'%E(VR0A"@QH(c"\:r袊fhiZ?",4׼; ^Ek[W?U;Y&;A)kB[P:Jj3m"#}Q^3c9ҶjΓqolȉ so+]>ޞ{h.dJm%ݣE ]$\ 8 FFsQ-a ZI7hdU✯&ht Kؾ_2k5%jKy;~ELss8g[@~`8-1l``sUcQ|1+JZr2vl(‚+ēqWү㾻Q$Vn *.7+sr8(adcˡou]%nv{S&?;s6yt]G"L"hK:0dG0 9Aآf;h䵇QKto՚A/猸gm \NG2Ipc7E ;œO[QdeYAi^ng񴤳98!ي K#5)'ԯ&KI %BwdH]sxآ3 մf[-3=#KrqDc!F9®NL>f6f[gz󞪊|VFUޛ{u C(i+k9x郜u}5ö9gM{T%!WpA:(Rk` (aEPEPEP\!tОo 7{taԥ4[$ȑCy$Y$Я<߳kZtLfY|m:`݋/V'f㸸+9f?+N) oȷuG=:F!V>PRO}E1n艹)秥Gz%I[lg9β,RnKyw7 lvxmEq:l/cM>+ȃ1JۚNsaINETQEW=a?'[C ^=u9M-5; $F*6 :jC:H.VgL F;1 18R_\YܧoqE*ddr8=71Mqb\s7QaN@g8%MUbnqsuҮl^^@6Y~HT V=CCiK(ZK{V` Ua%6; ك:L{wy.r !23erc7[$ J:ˑ 0͖Wm\Zo/,K(Z@KwW|`֑N $x˚, C xZբv(aՌL^sCC.ۇ꿲l~_#?iVg9YcnT6lx"1ʍdxf( `(^g`ǟmؼr4֗4M#b^MN[nCfHYi6:t%[kvbYI9m9,y%$fjm$p[7ۊJ+nss@2VB54}]*\4`B0w xX֞c-pgyFNР,q<ˍ3q=3[K*n{i 2gvwzIYo XےaFvłSmާl6ȶr[3$udS]2U#t&< ls:ϨiV7r\[$$2OWa+x#(P(UU:+nMKd+elψdD":zU;|-o}ME|ZsC2C4i$R)GGPUA4l 8E 3I\4:"St/-$BC۔)9QE SXaA<*<4cެi:nhi~^v}nqdgQv+\WA5՝۶^X&IOAVhQ@Q@sڷBzkտ0чS:GX"fS@5SIмC:$pExmܡ8XԿy\A5oN~)]U;FqΗO O tuIƜ4MdSxmO0WkgivC[^^,$=齘ƅ [۸H#"_#w`@^?x?X?x?X?RTԭ$k|0;` #' mjk}oPmiIOݮI@ Eb9oPBHڅ[{ /-To6\8>cGO O [c]rtb\ Y.DP[7H`B$DB%]w[t2iVX2DV w3Y\o"!9G"!9QOr{+(tX%9U!n[iuG%U,T)e2J>]ikl |oK!Unoq Ϸ) `) `3:Nm?_p4[Uki'Tb$? twSz7-7J61@Q}GP$dp\hQhU$_5=?G}& ngҭ+eaq%!W(HNB/DZwZF,B&6u3+fBBm+"!9H|1 '>fUӂyTv^+ՖQ|Hb鴫vxGĒ'۱ e?&GOq1t _&9b ?wk?.D褀hΡ.%EUrjI8TA֙usFt\?R0}h}k-#NY8ٿuq]@'6mY^Cin AmR=2Z3 -!. , 00}׊W|ACf h%,\7}w߭Eo ?i? ? ʂ94*kh{g缱f5 *W/  ?Ko5B,Urr:ӆ-N.O66:3J>P-G>Ɛ M4.qv*}5ed-̐$"k(#ⶌbfc74)Zӵvvך'm&Gwg8gx?^?"`[ZԵВ;:o2ݔ$&?([9o:6s\!. X>'x4wc,jN3A!@y00rJvalm]RHbR]L22B6;p<|cCC/]!.-a:yͲȩ9 ؤ:c^=GOa9G)r~^106kgx?^??CC/]%汨bKw-x\I9Z&o463gGO:J9g܆<回Mv?!. Yr4^xť8Ffip  v,n _kA{֟ e剔B-VI%6͸|!. N&Sȯmz$Pbbx0T/Ue에"ZMqừr%`<lG2s#`;?0A xTӮm>kGL"#u"2Q ٰ&Iյl`=/ .r' tgx?^?ȱJ4tڷRyVNP/`źmCωH4Z--%&ss P>^>q O Gw`'u&yF,DK",Fo)Is'5+I.oHԴ!~Ryw]{1?(w`;?0Amf^#%ŵw3+#Ku \,y_P.Kʛ-I x֝Rf`Y,=7n8.-jŔyY.Rf8Cs {ml`ں>quhۭ$M,jm1i*(Ubc"\:rED\AKq4EERY\۱ mկ/Q_o.+A_?V~ 2KIs"c%ϯZSs;ɯ[b<[\Zf랻<* c~PtW { +ykOcZ skz\[|}bܓ-Yꊞ^?V ]^?V ][3 "a4LM]O9tI`d\F=ͼWvN nFZ)r~ KQb"FI rIu xV[{9&dJflc? 1q"Jyg>ȫOyiI#eWG*pp3}*=7naHc}DŽr:rcO}l[=jhWE?B2I*A'<,d7kn[q>n`S[`$[YC%a{QQW=l.v?lKܾsxV;.^,yJ|̤/|4S5ޞ֖Ғn#<Ų2^?ǒ8 ^Vf,I'-|[g>65{x]veH!\+xS8/<=^[=ZKV0hHO~\^>Q;JbDI-+(8 HQe+x]7:NIo%ʭTƌHxݔ>Pr0ISaҬew40 I#DAYW$3@ɠ Z8xwNiwRbE3+6dTv[}\R7r;nl3d W+8.٧_Y?syM$sQ>yQ#-`ԐxKyj6WZy|V&Y`r6@$K iW^N!ɉ`O]Ki|\!wc'"_F[;EŽئJ0fB#^~Uk SEqaO5a$R3ʪ@ީ,lfx 6Oqү{wTeތ1`s4 t[-F)#Yt`s"Mͻso9Yݬ2y(pMIؒHYdU9EgUcjJuiOuG}<{">4e7;tƲ\Tv[}\R7r;nl3d Ia3MYcI$(dyf1嘜R]f\74Q%!I-)+YHA 6ӵ;N9GkyΥuliۢƀJmYpW,\$` tk>//7O!Yn$qF cV%c]ʻW?ռ/VkqڴsګB#Y2[5oM.崑;h#>Bم-!2H"0h_Xƥkv? YI ,]Wm:>e^9FBf`2dԢm]YS.`F@A]۸X@#]ѿGsOQusFu}jD*nb]gqߟJZf}O  H{|'9꧎}H~Ά XXaUEdkk_-OpTkk_-OpUJӨ^Ν(ao ?i Z\YhPu`BFa$t"<)#w땧[=hSoqh&ݢSELppqilvpZX1j[`gHuk<[5c-ğ(9,s?2LCBQPԯ,.eHa*U;奞u^˵UNК5jdzk5yʻV#9p{VmυK'6fB?++;1jm m+y H<;il2a7U# 8;޷mumrXGpȓJН Doi.sVnJ˽|*y+9pl$-IGSw`uAU΍":FNű2ҧL=2X.rx^qqIs\(d) S9'+펙^ Z?."{Bcp7ڬ284}9\H]v0Ni3\ۑO}$J:ș,䐬xl}Dri^Gq<1 suP1pG@'$aLϔee?;mS3xx`x$;wI%ϐ,G7`:d[[Z,7LsLe#=cDf*pU1SkvfeϹe?tLj9=cEݥVŌ.E @7'[X.gb(n'yeNW3],#i$? _e,|7[wݷzq[3Y<ȩD2 rq@ys*n2%مghf !2rpp*>\r̋/ Ka1[,Dw$JI=) ]ݔ:$Fە@xF:nąG̱ۉ)tOn b#g-ٶ[tΒJg+mcsezK4X34FbP  B|jʒ%c>4cVPZBByUOUR:$y$m,{&D\dӮ)[`!i,YW69率EԧYn pȲI""+>f9ϰL{(gM:C%0ÔónWbuEm.83BRrs[@̒#hVY6\ Hg9dsEaEKQ[u/E$QQEQ3q4EER?QTo_"2A:W,>w?DEvg-Y[^z]GQJ# 2Fxxu+Ejk6v:E͝Α2q&cM wkDld m ;#>5iM巊4DڌK3EdQ+@^9м!Xh:NmȚn-kRlXXp}GIUsx!ulZYP\ieݔڍ֐͞9Ց02r`rsڼ:cXx8q)ϡ~ WTTuo *kRoooKKG :$`o`3FA5 kZIJﵩ-DT,( ϝ` `$p7k~znZ#״˻+;RӮw+vF\qM{Lu-:1Bkf=3QkZHʴ f6τ d=8fjkvB۷ c 'vzrSBۇ3= W-,m!TN'#ՙ78m+NKg}O,2F679`Gj/6MD~EIRG4l1\}Q.ǠcٹX̤Lό)?.Xccq u|-}2[+mFksΘ89Pr0x9W+惪֛ g,Օ ,_'>`/+EE2+.?'WB\Bz>^ ȝ>*?k{Ÿ'hE46kE9{ ^LϨxsBԜI!{4yֽhVy%v0 Č:w>Ep> ԾݨKkqgmZun~3U/< Eq\&vYו<u^vnq]sNPH 8!b G#@rY.Y&k5q/ޖr YeႨp}q66kpd֙b)$8JMW{ŬI,v-BHX('׿oiM}\%xWTdV,7A/ OkN` sڕϨ^ZBͰIq*ƥN2HUvt^,&GݼRIK{ra1<3H pxI,nh 1 *Y r͓Ա-ws ~DbYZs7>lFucwqmk,wbuVcO򃌜{zcӭΙkkAcz׍ Hy}3D7o-5(&+{)p0 / KcLѴ_=|߻g=9Ӛ.uNYj)p])8 ['z匠d6M:HgLA&9 1ʝ9kؚfilg qj1on얺IpfqG nmbU+/mUJG^y WKǣCҵ 4H#h.#79QPɺ_; W<oZXm!:1`VJK] Zz)MWoXXHy&X|J\`6F8996R[ynv͙ی+B̺{^..9+>BXn-$.I pזUco$:0~1Q[F _z kYGmy[.wmݟ*dn2:{g< (Pѵyۯ7z²F]P7&іU=O灓r,n˛甿4nR"|Ap/hL0 # `>Ũ̀z6?#ڮ%%,M:c4mAM1o1eP 2{V|)#w땧|#"OG\?$^ uM?UA,"vfF^ 0| ýtz~ i @+8@rg$g/=*9[#W׮-6+KYR/ sҹ34i|s4)UIZWnskgGېFqZo'a6t(ېyFG~1OUP2x(.Vg`u|I{(dY\Iysۮ9բ)Em+^2'LW!sqϠ֬:'{Myq-}899ע(Ceuo)HmL)O'2ec's3jzm="Yc`$[ ^9^m.i\\\?/*XFGus4-LaTb92Ձ9բ*j\\J)%e)yj> wQy4KW"meAl*xz(1IqrpV֎"[LbONyq֏칱}>o򏗻vq ->K{w$q9Wyd/'$U1O'K"HtREdYme@A$uB\y$ר?g3X~1GR_VaEKQR=( n?֏[**+A_?W%/z=͵x.ȣ8P;J=.Ya@Tʠq֭jkkZu'J|1Tfy":_k}Wנjkhӿ/G׫v iZf]DzC1`7Ҷ?Ə];rMrr}M-U?t4jkjlt[jkhӿ/fE*?Ə];at[jkhӿ/fE*?Ə];at[jkhӿ/fE*?Ə];at[jkhӿ/fDvvmyo ŻtS ulSUO];ںw,ETӿ/_.tUO];ںw,\!v =jjkkQ D( #qBLwԿy\A5OOְu/^MoxSD赦z(!EPT}2I(-vٮm6㫷?t.}[8L@p| ,dVE@X ͞Nzɧb~bvgy|[]Jx+IIC+ {+iwe %/6HT8TWh=aXU}F#V5 {m2i86ZȱAn6S怀 |&L+GoRXm< [qw7vtTdM$\SF7YXA<j7WO$^@Wi}DWc9殎mK;-)nFҭm4շ)c HX̻y`|mݖZXnNbbܠN|IXr!EwU ]YXTLU 9u\zO!f<}w <Ք* 26'ǡh!X<7sۡHj`:J+ B"F*Gҷ*`(QEQETW_7soRW_7so@}]e5r#tFJ秮2N:_7toX#+ljtR_,*B㑞=(d9G66}7Q}2Sf6v1rq\9KR̀#dŃ-^ƉwwZV쒙٘c)7w;įFM<˷1\>1c: O^%']3\~tavY&mX"pFo- GO-A_26*0dD k?O^*nm#]mGYf\ nC^+2-#ʙ$cg\~5B (((I*I*箿oΜB,8ϯ6>Z?"o=ecm*($NrI@dRY,=FGJT,Ehȍ$]ۜvzŸ7xZ)+'?-]gZIMn'E ( ( ( ( ( ( ( ( ( ( ( ( ( ( P T?B9:r+?(_YQEQ3q4EER?QTtW?UE[{ (Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@sڷBzkտ0чS/RUp?N?ZԿy\A5OO֒)QE2B((((((((((((I*I*箿o΋(::j(>19:S<#"OG\?Wͻ)LA5_Ÿ7xZ)([hdQ@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@EqDЅKQ\?!@aEKQ[u/EJ,z(Z?"n?֏K: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO é*?k{Ÿ'hE`_ 'kIz(!EPEPEPEPEPEPEPEPEPEPEPEPQ]ǤͿKQ]ǤͿs_7toJ` ƤpCw:.֏(<& <& nv"rB7qӟ㐳2:luӿA CܓwEb1T##ҝOG\?o ?iv4QE2B((((((((((((((T?B?fc"\:rz=QTA?QTGsOTU% {/Ani_xV-(EPEPEPEPEPEPEPEPEPEPEPEP\!tОo 7{taԿy\A5OOְu/^MoxSD赤fQL((((((((((((oʥoʀ97GP2TG}*K7/o!庁FV>Qڤ]C$H2\c̶ gϾ{}! C4n6s..wb΄4Ԫ]"FO;o ?iȷuG=kxSF+O% p{QL((((((((((((((?j+C'(?ȣ/+rc"\EEUs7GsOTU-\AIgAE[W?U `)QEQEQEQEQEQEQEQEQEQEQEQEW=a?'[C ^=u2/^MoxSD赬KAW[;D|-i"ES$((((((((((((*+j+z>*M HW]*rYЖ''պ:7'Q@}?+J H hc%P2s ?i ?iƊ(HQEQEQEQEQEQEQEQEQEQEQEQEQEQETWtOT]BQԿX~1GR_TG*9**hiҿe^"M+@_*[܅QE((((((((((((+տ0C\!tО:z?&)"v`?Z ȝ>Lע)QEQEQEQEQEQEQEQEQEQEQEQEzM\TzM\P=usFuKusFuIaER|)#w땧򒦨|)#w땧 QL((((((((((((((?j+C'(?ȣ/+rc"\EEUs7GsOTU-\AIgAE[W?U `)QEQEQEQEQEQEQEQEQEQEQEQEW=a?'[C ^=u2/^MoxSD赬KAW[;D|-i"ES$((((((((((((*+j+z>*>*Š(PSF+O%MPSF+O%R;(!EPEPEPEPEPEPEPEPEPEPEPEPEPEPQ\?!RWtOP3X~1GR_VaEKQR=( n?֏[**΃J={U4׼uorER(((((((((((({V׺O] sڷBz:0e_ 'kX:?&)"v`?ZE3^(HQEQEQEQEQEQEQEQEQEQEQEQETW_7soRW_7so@}]U-}]U%QH#VJ#VJ&v4QE2B((((((((((((((T?B?fc"\:rz=QTA?QTGsOTU% {/Ani_xV-(EPEPEPEPEPEPEPEPEPEPEPEP\!tОo 7{taԿy\A5OOְu/^MoxSD赤fQL((((((((((((oʥoʀ97[7K (COG\?5COG\?HLhdQ@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@EqDЅKQ\?!@aEKQ[u/EJ,z(Z?"n?֏K: +@_*Tҿe^"ս[QH(((((((((((([C ^=t5joO é*?k{Ÿ'hE`_ 'kIz(!EPEPEPEPEPEPEPEPEPEPEPEPQ]ǤͿKQ]ǤͿs_7toQT_7toQTQE Ÿ7xZ)*jŸ7xZ)*E ( ( ( ( ( ( ( ( ( ( ( ( ( ( P T?B9:r+?(_YQEQ3q4EER?QTtW?UE[{ (Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@sڷBzkտ0чS/RUp?N?ZԿy\A5OO֒)QE2B((((((((((((I*I*箿o΢no΢,(@?nrRT?nrRU!3)QEQEQEQEQEQEQEQEQEQEQEQEQEQE]B-EqDЅs5u/EnVQԿ(袊fhiZ?",4׼uSJ={V!lQE ( ( ( ( ( ( ( ( ( ( ( ( o 7{=a?'^ ȝ>*?k{Ÿ'hE$S5袊dQ@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Eu6U-Eu6T]ѿERѿERXQE* ?i ?iBgcES$((((((((((((((*+C'*ZP k?(_ܬ?ȣ/*QgED\AKq4EERYi_xVꦕ {/AnB(@QEQEQEQEQEQEQEQEQEQEQEQEjoO 롮{V׺OGFLKAW[;D|-kRUpև5t@12͓JHtVz}7TiI=ܙ|BܞsønYډMډM Vz}7Gz}7EƅMMpEb!]gLK1Ow8ISU!qZX;M Mg=֠bY,#W 2W@aqFpYvQ=bcBOXhOXhXТ/?/.4(Do&Do& +?Q=bQ=bcBOXhOXhXТ/?/.4(Do&Do& +?Q=bQ=bcBĴ] oh3J/@}5X7?*}c^}c^誟mC{mC{[}?}p}?}pnX7?[PْȪ?_\ ]BSQ=bfcim“<ꀩ'eݢc*c"\:rz=QTA?QTGsOTU% {/Ani_xV-(EPEPEPEPEPEPEPEPEPEPEPEP\!tОo 7{taԿy\A5i'Mh@g H>zRUpm.-=[K I"CӅIg:uMۋv(@*A͑2HG?Fo]_i2]]*,-;s\`v)0܋ee]p1L *Is4EU|?ܹi?>.O&-QU~.O&˟# w#??߯|ѭ|1o˜NJH0@2Ps eȾ~b?^ξs},|E?8eȾ~_|9~{:?; ߟ׿\>s"럜}?߯|2d_~~_^sX̋߮~p׾~s?_~9}}{c2/?_~?^ξs},|E?8eȾ~_|9~{:?; ߟ׿\>s"럜}?߯|2d_~~_^sX̋߮~p׾~xgU!}K>^eȾ~{w 11b͹v- A?7cBs%\Fc4g Ry#!_7toQT_7toQPER#F׵j֬k >=~iR_.HmFHa=?1)9Wt ky,m[b5#  ~n?(_z4I![ILn)" ES4W]I4aP[IstGsOTUΡj-# yJrhWrh|?ܹi?EU|?ܹi?>.O&-QU~.O&˟ TU_˟rhWrh|?ܹi?EU|?ܹi?>.O&-QU~.O&˟ TU_˟rh F|+^ ?׷GO?N9w$G$qKBߜdӧQ c/?OnӶ>C? /On0K}}=;t c/?On?N1=v,|bE %H>ޝcOl~^;c:{c#۟ӷL|>1"s{zv鏐=cz{t폒/>d&$_n~OoN1RX\Lc^%f|V4+y/Y,2O&x $$`?i?4Ѫ>-&kBck!o,1`e8z [uts!. :R(fc"\:rED\AKq4EERYi_xVꦕ {/AnB(@QEQEQEQEQEQEQEQEQEQEQEQEjoO 롮{V׺OGFJ A*O\yT5/^MbXkWMzLי{: \Qs)ΧW_/k+V:5i; "Okʎy#aYU[̀H.=BNeKD@`pzW_/kW˨骲CsmttR,cۺPs&;Q'/gU-<1yJ9,>Bs t_jf\y\%Q7^cwCdU%P,)rq5g}U`26 m#>;P\yGڮ?缿ٮv?%ۿ/EF>Vۓ#*o \o ~bI{[{/l/fnȲIi<`#f )㑐W_/k6?- ordEm$Dj̬Cޟpv4EYUem X%B U4}{}tխІuؖ!,aI\V8V v)d&W 6shlb-bR]m.;FHY/hU5I7& UpU#8Y/hU5_Ũ.D4vޱb G"Bqh S]n!xyf%"'$a@3Ԓ \yGڮ?缿ٮ^zXOwvw$^չ1yJ0&66A#np}{}>q=`X٭?nᶾ;|..6rVaWSIgzmcxY6ł6v?^l}{}>q=rA\̫;Wcm,H2 |6+m xVkkb` 1p: hsW_/k=q'm4#icJ4* 4Jd;~j_ɯCnou&\Pdv3@/ڮ?缿٣W_/eKin㴾K;xV{BB kc9'BX77NxXbP݀t돽pp/hU5/ڮ?缿٣W_TPjf\yQQ@{}>q=EEK/hU5/ڮ?缿٣W_TPX~1GR_VaEKQB袊fhiZ?",4׼upiHҫ(<8 {U/մf+;rzK4nXIO&1Ѣ,?礿'Gh 4VwhܰuX?MȒ :9`01vy'k( OoaoX41E?*cbn$d@V0Ivm[_WsKZ<2\,39aAO,pmkh< dgh>T2Y<2z7ò[iZmII%dv`g~lƒ˦kVNڳ10V"†g y=*׆u J+CW2~\JXfV 1ᑌm @zսjP3,C1dCoWBVMc Rct-#Z:Nm,V1y\w8?@*j;HD܇`je6+ƩO]{mSU(#DmlZ2<[tB8 k@׼>KF0c7ʌyPG(dd2ؼ7RR$\z (Jl,2E{l'+2"wg< ejiaKcmwٻØDyJǫw9GErKY#M,Gha5cq&,wA7u6<,gEE*[A$yx$1I;*(Դ]jR&k2}w&ݼ`zXjtT`%p!@Q20O-RFުȰ/HomMif#38:`u|3iW4[h= xyx%{劕'o0CtP9xJ+ hݼ^B<(9/м2EcN N㹘рr((EPEPEPEPEPX~1GR_VaEKQMQEQ3q4EER?QTO$PG#!L 1SvY⨺:%o=ecm*M4VI<$PƥG`I$P_?6Y⪻OI"#J# l8R~S/.ల.6W;UFI;P_?6Y*(]_?6Y*(mW͗ȌMn=s=jaEKQ[QE(((((((((((((((((((0Y7 q#_zn/UGsOTTm*_?m*_?3V?K+gY6g8qʮm笿Tm߱UF/bgı< sӂjjm*_?tZ31XԹ1U}&qZ]1 YnWv;fԿy\A4 Ƚ=?Uh߱Qz~QQ@߱Qz~QS^X: 2Oa@߱Qz~PR FyG vYo=ecw)r#UD[,vUNx!gei H7NclU(@aEKQ[u/EnS)QEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQE-\AKq4EEL(xcWl=tS@lY<cnm$x  b%RBk]b{.mCȣ+8a2O˩ 2G5NpS\[&/^ "k!]nf`|s?of`6V_:ƾTizー[#s֦ (u/^M7¿/iONԿy\A4 Ƚ=?U0.QE*+=ŤlHC/U"aAbȯ;,Tޘ{()>eh@*x'sNc'ك9iv-F ! 7=!r3Ɣj,Fn%` gޞ䗑)E8=JvW$b*ZD=0<Wy n>1 3^MEo i,if+}ga! F$=88sT,$``\ Jf!9B:j.wlcRP0)7{ÿ %t7_7toMQHVQҴ}CO-夶 %o#y{S(1l@7JS[eojUjd6IĀC@qSf7:2y)e@!wrIfͭNK#k)wlh&1H+mЂFF7){EytGkgڐ[nEOK'uOy gn~\(?ȣ/+rc"\X~1GR_VaEKQB袊Zxm^iR5(`vZa?ƣ-Z\Vgp*8Fv}B\<Ir1:Ԗf64K ^ضմ\ 2fΥCn+fX/mMͮi=DeudHr71בLZA֡.젆I̖βT Hր2sۧ"3ۧ"ܿNIWeiH^al-S6:uc$gQ&]܎J;K?QK?Vc^dv6FЩ5vRODgROD㕲Ƙױ.hnݲ:l.rpQ*} >-J{i !Y)* QЂ:u/tO9Fu/tO9Zɯhj/elnƍ~aZ|ƙF>ۏGcg?7^=hlf]C*`-}|??:4hSyEo1memFBj|uMFr?^ #VmWhҰoF}j+7PYC <ժa?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5ja?Ə+5jy"> G#T LO:TԿm?tQp9_6r_6rj(u/tO9Fu/tO9]5\g: ': '.3K?QK?WME/ŚyGR>ʼm+J +MZg`YimlZQp9_6r_6rj(u/tO9Fu/tO9]5\g: ': '.3K?QK?WMEΥCn(ΥCn+Eey};+geEc$vZ&I 8cןM(u/tO9Fu/tO9]5\g: ': '.3K?QK?WMEΥCn(ΥCn+RODgRODQEхן\spgw]Qn?P64K ^ضմ\ 2f*(u/tO9Fu/tO9]5\g: ': '.3K?QK?WMEΥCn(ΥCn+ROD^Qv Pq+WJme_FВ A $mE"IT2G*<ǐq۸ K9Q#E,@&^Df-}s2% .O#gs:-#k>ʟgiT2 D@ɼHpXd6=Km*_?WΏcgr5+-6Jf۳);\8T¿1NāǾclEYiQI%#f~gryn4x#J+ry@@ugGɖEfD%3mٔ .XWFs7+KxSPc/V_keae EnҢ4JFىjm_?ŭ6>%ThU/RH 2#Bc*[ҮRc-m! ( yrL9cm߱Qz~@]hq0,װc˞MjP@|ʼnNͭ-jMuX@9 8H0!yP뗛Xx<AIVE'mA HpЁy"P6 dP{Qh +.[z!z08c7F)e9#D wN@ S&ki&hI9wBxW @PA栄FV%g;椂xsA4* ,g Z覜uhe> ꨤJiljQ"% &\ 90"r H000 uj{=8B4 $zTy 60XP ulC9:T%:j7T +Kֺ F̢@@1xm% pnYаlLjzXƻq@,PcVn*,BSD.C5C!@$_lpNmtb{rFIj, !@&KDTPyj@FvF}zOeg ,dp% !H>:yJr8 nxin' J\* )z`'~&".< = Nο4e~~~ܓ>IW_?~e MpO|W(LĂ#H(H)Lă|`$HBw0ЁK_ 3R+D !XP4MCZD,t!cD5"S (py[x r* '-4a >PFp ME8@:bf&RG4 `p +Qa$'RrcD)p U($ gGhDNl#,eGKRD $p ICr.yDYVϘt%0iZc$qB@%hojC#ZXrDPǛ#: XU0C;jU+-nsV:^x  _7PlHpD4hHO+FΞHr NC d["G`[.B/ sxx*`3 p,'X/ nm`[ jP \# ]gp3lH7-dtr-qKP2 PR΁f @ nwL}p"D8Aln  Uo!{<"zv; txP:`{o>SoBVY)q3gLsJ? d-p. [Hon8̖?)4ǫUh,i `= >"@\'԰3FBDԀ* {Mfzsw?zF[H4{7F*KaD2BT]n ``'u|6Wel]-bb* er|;8bFepp 7W`V5/,8o`CT(rdTc4`bx[$rWNp PpU>iChogHgIN@]qU L?[@YFH^G\N$tgH^*`|DBkEmhfp6i=KfBUp]tRhQ\BȌN QM RMc9Tև!L u7w]@vY26T]rtBWf-Mu`@>mCR'r> +fPc@v+0 Do@ksFm؈=Kp/H:ܴ-=X=lpD8ިO}U =ݸ]}C[צ9\+6m_R .]s- ~P:E4^DZYK`qT* aK%3.5<.%ti#*E0E.DAV2Rq)NL&PR@bSR_U^X?f'ld.%o^#r>ksQ|>*=~'$-(>1T .j0]<.>O..j11wN^/%N]~s4OhQI^=^_!1K؞Nni>.uIʾў>^ / o!/^6q(* on2_:0U;=??A/CO?.IK+//R?TO Z\xb?wQ}3ik){8rl_QXp_@y#R}ϲwIlW%O-3aNNDNU_/^/> ^\/oQb!DABڏcϏe_# >Q?TH` @ .dCFH1"cʼnEl8I)U IKYӦA5 hӧž? @BUM'6tfT6hYե&BAэ^}NU;iVl][5&K񨌁6i_rG7<`N3P b֌‚͛6vWo|7l,ڃg!6P A.H Z`Ћ ’0[veceffs#"dNFYJTgFfvZfCgz+:z 0Z%z⭢ftꢫIS!Z#5;l3m_bQ[n7ﻩq<H_qr*V݇{{|'x7W)wQ՟Gy詇 +yApPSO;nmUbXPP D@4R7j`]Y R% [ X R`aAf G$ ÔD\P ia)0GƔL_0t R &A_x@P@ bXǻ~cXhFIGBx/(yA4> oЁ,c Na KZ, cxpEm,H:Df"VБæ W%+_P!/ xPa [7 bXep@0*S?ЂASxAZN]S"n \9Ё',ΓWЂ򒸂MИ^e&]Pυ '@%F>/,_GYMs h*~v e)U\ ii6ydBlQt lpN@fk>լU6` Rd Wx+`PъX"p  r,?Qf ثATZA p8?0Y <9wycdXHA9`%ӘDz7.=g;Y< ^D!#, + 2̫Mذ?˒%[ༀ⥞Aʛyw0 `jwu H8t0k,Ѡ1h`*n&I (S} Qj+Px )^1Q2 dApf K175 yV!`< ];),!0+xD7X Q frá);Ҭf!n+q3 `3-F%@׊5la&`Iټ2kb Qff[{+s nP:@'JpjYȣsm@pך7C0ka${oӈus4\ۅ*|AROü01 :bBdY,yQ'|F0PPRbq@(\ t5P @y1DaSw 4=0 *رhY±k;I%A msc%%82o0ːklq șxA Np0+`;N0.gtoj&jPAa[}I(,W2! C, 2&I ,4a v4Ȁ_ =#\x;{J@,D +l{;Uҳ>_&8d~? `5W " "2'$ls"(sC8 $48=S0 ^2)A$+<  nï7[[ /(%ڂ|$c*(99-% -`?ہ"* h1+-“x(A88,CP2Ѐ"@̷ ;öĔ()<10ܸDGrD S00d-J6iすAllCHIפnɄID?gI ʙJγ;t7`XZ+ Rdw2* X!=|)9$EygưTĚhȚL:.2H 6<(@%-/) ,I FsOg PMقGd-OlDDñ31) vMPPqDn:ҔDMuY4mGNKjQ&QPEWlR!m9(;:%&. AҀ}Z:)+ຏڣ6(HCR3SRe&Q̊Rҝܶ&'50-Ђ $ iSj%ZJrZ%KR9/#=F:PmR'PSL1Ls@(+\XLT^,/ǬF<*Q%M>.2PՂ*2]d-@??9'Sm3ǠR/쫼sd{ L6g-tb`  p $áߢ--|>,.,YX\}0$B ؘ:.( h? A@H pc![5~6X7X HJNO2(  P`!PC^%a|V}? Ha&f Ȁ<I H]nc\d8Ɓ>Cg(`p(@`0  Pp~^f]YoeLNp bN6>N ( !$ip"ǽ<3뉒kj*vw}tX[ٮ|j.n 𷈵-tuze!kkGp! >cO=g߄Ae O !=IlfKbU~˘7 ~8撏O? >c??лgi(̿14??S7 ~8撏O? >c?_geKqJ?3%1r%M~_:g._YJ_$ ϘOgeHc?2SYd*Q~GX?\Oa ̻qSc2J?t G˺N~9?dgGב'{MB?g2 ϘOg._YJ?s/  ?'T0{MB?g2 ϘO0wR?v\nS_g/ ~9撗eA3???S(wR#̔q/mߑ"1=&.\i(̿.RD@/s?3Q21SGO(Yd*Q? >c|1?лgi(̿14??S7 ~8撏O? >c??лgi(̿14??SoFhZ?͞< 'ץ >ڏuN i$*'-`7VSH];2e"P%L&1 `gJB[b9.t7S@Sn_nHuˮtjH<?>[}{JK n?hKR]6knK'NΟ-l%mJ~_G-lkZADݷt=1QsZx19m/+ےir7FqOd^Ғ鰂o9mM=x=)r[s өnI[@Qn=;ַQ7mLz~rt.y? b[t:K&Z\@椃Sl [t%oA|^?J\~By1xۦRVrۦNtMwC:KogN o} q֗.m9 qD`_mE).&i9,?iIt_;c׏җ%=.ߐn8 }:?:|:)sӽk]vǧqG/AΒk(@Sn_nHuˮtjH<?>[}{JK n?hKR]6knK'NΟ-l%mJ~_G-lkZADݷt=1QsZx19m/+ےir7FqOd^Ғ鰂o9mM=x=)r[s өnI[@Qn=;ַQ7mLz~rt.y? b[t:K&Z\@椃Sl [t%oA|^?J\~By1xۦRVrۦNtMwC:KogN o} q֗.m9 qD`_mE).&i9,?iIt_;c׏җ%=.ߐn8 }:?:|:)sӽk]vǧqG/AΒk(@Sn_nHuˮtjH<?>[}{JK n?hKR]6knK'NΟ-l%mJ~_G-lkZADݷt=1QsZx19m/+ےir7FqOd^Ғ鰂o9mM=x=)r[s өnI[@Qn=;ַQ7mLz~rt.y? b[t:K&Z\@椃Sl [t%oA|^?J\~By1xۦRVrۦNtMwC:KogN o} q֗.m9 qD`_mE).%7=8][}> z/ïp 5"Ş+ Dچw^HZO̻IG Zq>oz~Mmk[}o?pga<1ɽZܴJt?{Qn:UC?s?yx7 'G'wU]<|hS[5Co G̺6vox8CؘA䓍!h[qnNtaQIx߯N4V1 cj[^t!\-?x?pZSN G;}rM-'o*u.1Jx |څ%oJ)wI/(l: O:(o;xc},FN>[bNOQ:e.Tm_?̹.rͨzfKRK-4rkmX# aI#,Pu 4Io쒺e9]'tkM$򷑇-Ng'm[dݟZ[=M\}H0Y}}v]Ca8QKYxnï5/֛Y/o—+)_wM4m^2zhzZxo|2>~%ִ[k:NbZKk붔=D۴73hud]jm7'}x q῅/;}GTV_]SӚuZ6B;[DJd7gzIt<Ŀ5|6_~ӿu#oCx6Fq|@߁ 7P//gD{))cЋOvKzwpj94?_)4 TQ@oT}^w__?~+_+}j:$,:_"dX k=Ft>/~:>ڎ|uf.MBu7T֢k]%ޙ=U`k]M? g⍯BLx.>/׼]| s _! #.b?&KhX^Mi|1ռ7›+kᵯi7DzWUC>jQۭi7mhאNֱο>3|Jt3_ |?ӵ[*T?ٳ'4Et']o-ޠI~=DޏM%ekwv~-ѭK[\kcqu{fAE|8xL|Q?蟴4_<1k .5/ts:>kڝn!ѿпk%kbO䬺'>1g5koxWocj~8#ᴞš!3[xCt_N Ft}v>!5+J(]JQZ_h6⯆Z{%s@x$JO)xΕ`uρn+P<ɩMlu}jOعs/x//<mQVӬ-uCI?J:  xZՕ+^Ƭ$u[^elo[ioXknnh/ai*Ao }[0c]Kt߻k1pvR/y_Wյ{=xVCiᠾ%@oyCjI6vM^&ࣷ.ϻ~~ǯ-m_|O:i,K>ľ"m >XneiHqqSƬ9 y/Hw]5T`^g} utEukkW~-ͱ#wH¬ڔ$dn._^i&ӋoA9&9W ,n?B;4l:b7+I7dyNؗq=қV\v#_Oƅ)GC6qϧ_J9Wjaʿv :Q=IlÕ/k vE亽=C/kycbimwrl!n 9 CsʗG19Iun.1^qХ(rۦ#G4J[9W?#J9-rma7}(Wrmo p0=Q-_-v;1B9maw$qҎy_9Rq=҇)=r۸+܎=}?[t]w>}(I]y*W7D~8G4%Tͬ&ۯTͭ19ޞn|cߦ(R-l.=:Q+*]n>Ǿ:P'ջUw{ǯB!n ӯI+o0_?(ڤaʗٵ;u]^ʗٵ1t1G49| { R]M@J9@KMJoʿnr8hRz9mawuI%v^`GT9R6n^K9R68n(zz/~JK鰻8G<tz CV9WG RW-l.;NsI$ռÕ+"?t{jه*_fwXҋuzz*_f OP7c1)IwC6qr@(.A7\c(rݿ*]bJQMGsҎi$ڷrz]qG㎔sOmR[0KnkQy.OPK@a{[][w=b).rۦ Hr&{R{][_˷p Wz~4)J=_鰻}:Q$VUOn#qҎiKf}XMc_J/%}[ 7O|csKk=C`ݎǿLP%[t]ztWT=}q|tOkvv#_Oƅ)GC6qϧ_J9Wjaʿv :Q=IlÕ/k vE亽=C/kycbimwrl!n 9 CsʗG19Iun.1^qХ(rۦ#G4J[9W?#J9-rma7}(Wrmo p0=Q-_-v;1B9maw$qҎy_9Rq=҇)=r۸+܎=}?[t]w>}(I]y*W7D~8G4%Tͬ&ۯTͭ19ޞn|cߦ(R-l.=:Q+*]n>Ǿ:P'ջUw{ǯB!n ӯI+o0_?{[.ʖѵ ?J/%Kӗo p0zQ%jݛ_59/"]@8GkWqrE}n>Ǿ:P7y%m` W9֒>[tPHノ(睒m]b_?{TQr%mo받n^k9a#Kfջ6>_|~=1MNkao q#⧚](/w0籼ot i,V5Eyw#3mm1w0CZRWj1vSFU”#t]>nncerVGimnlG j3:,JS{ɹ?Y;￙QRItVt-O?~\pv%摮M=_ χu+E]. "EIQo1!7twUyoS—>Ԛ%kD;_Vі)Y[Ȍ\n+V^ y' xVkkXk}:Ckqk Fv>Ux[ĚOD|!]OƟck}ͶBDNK },7e1'-.GD|ا"R м#BQex&T|Qǿ? >k,C^*֧{sM'.>^'g -u3_PF>,4}flxjR]x;Zj^[-v{Yn#5]By}#X⣖֗>"|5Y 71 2x?NnuOZ IB3Z^ßk^#е~ߤIh7u:MkKK=Jx(% 'O;%֣C_uMj<-k^_|c(Ѿ&xk_|U7?k_ |)Z=JWmo4WnXj6I[n?PkV]~>.<9kTKYxnOZ+kZKUԮ ZѲmiv/ˍ[R~,G@־ _ũY~_MoVZ^ 7 4O"x>5(|=k$շ |.^[ٯAo>#xY:]h?Q:u}P^E6)2_ej0źޫ5`Qkh\?xN_~4e|ӵ \|B'ܻwŖx/GMVNe&}3O-[Q{!}x}[Fg1d(r'ٵ V V {~rP7J9ZgRQma&a51=5^9Q]r:*|A-,Pa{,Cl<w:FHW$ ?=UZihڦBj}yg[]Oje(-pi7u붗PIYj3/.mcfgZ^ۤ18'qRq*5* |[eV|mjsV9G1oijݢִnl/}Mīh6O&t &X YYɴ![5!9oږ+ߑ5WdoDjbEX-R4^EDYr+Wە6|]&RsIe=Ao)͉e'I*+5pF]:v<\dKv#⺥b@DZN [1~? Wu@;t)?̾1x?J/_x ӱ1R{ c wI|o ;(i?1~ҕ⺥a~chc:~Av#S}᷐c~^?̾b1cxZ Av#1Qx2c?+uJǏ;Ǻ-o ;t)^=փGx2o DZ(}/NcJ&=)+% F? ce@;JWꕅ=? wuZAv#R{1؏ӧNeAcQx2^ҕh/L{SWKy1،~EO1v#+ {?L~@GxZ cO Ǐe;+Ǻ_?x6 c1‹cG)^+V<~)=ցky1؏ӧJb?N;y=Ezv#?JWuA1N_i/mb1c-<1؏RWT/1x1S{ cO•h?1~? w/6 {?OҋF? ~A~c_01c? /Zy cxX_ccxZb?N+Ǻc:~_xm<~/،~)^=ւDZ?;}ac~^?̴b?OJ]R@DZN [1~? Wu@;t)?̾1x?J/_x ӱ1R{ c wI|o ;(i?1~ҕ⺥a~chc:~Av#S}᷐c~^?̾b1cxZ Av#1Qx2c?+uJǏ;Ǻ-o ;t)^=փGx2o DZ(}/NcJ&=)+% F? ce@;JWꕅ=? wuZAv#R{1؏ӧNeAcQx2^ҕh/L{SWKy1،~EO1v#+ {?L~@GxZ cO Ǐe;+Ǻ_b;wҋZVǰwcqqM|o#ꏂދ٬xoKԯui--(?46"n2!UyJ9qm!6Nn_+j۱x pz2}2r|Ou{'c>f9#^_=éE+-=E[^0oqjy.tBg6qC:]4@.ӦO)k]-`Ro^{?J\j@8S0#GG5XN|ړ>t)Dj@>]x4]_+m,(N>VsPݾC^!Q_2bQ *#cAi$' (\$%=մE_u8& /5cs 65[ -Zき[?tS:_ ܷCF04 w+㥖u59=4)MzX_a=dO pktsyn9@5̌υ;˰ rcZ]}獴r|=4 lϢ,!-K=kb\mr˞1P l=>kt rրusP#Z҅/Q.n: up(mk6Ǯ1 -{ qMh[ Eo⯇ iGxީeѴ{5`Čbܼytᱸ~ƫ}-Z_MֿyZN5&5c#gzWc ]_y#?`mƧ#g  |U8Q_a u^ /=zq+=+S+Ko1[b! S2w8_[opcGաgYo0?iS!`̝ⱏU=?kB? i%/>_iS!_?+TtyM~Q  O:ȟq_h̿.?0+\Zu?OR??M_V6{O.6a C +QhRP}Cu??'xcOOP?pX Zy1H<;WoYeAoO0J߸j1BwkWvy8$~ǃVF9^1qZu[VX_b0:x;|zOi,2NZ_݇a y1'xcM*7+bX~J//qoO|1Bw?+W{ PгV݇Pύ08;|[ocwO%.u??H|'xximf+b?>?')C|Wj{Vf+b?  O:/)du_Rcoc%}??H|'xxʭyۑb!!`cYF h̿/oC _Χ)CW\x<2MZ܂1}CS> Нⱃ |U!Zo;zb\"/, ˋ|ZGOYeA-_#0+\~5??H|x5o,2#w`˄S2w~*_-v0_iS|WE*աgYD>_+y1HDzGimKE"%.?l?H;cQjScİ8+iAOCx;y1/zv?`ŭS#`̝xʭ'J{r Χ G;c?+Oo1K^ܰ" X y0#c!*QmKbo0ˍ> T~^o1۴> |> Л?-W o _A b2?N_ ~*cm+b%E0+\%o9B+~*:.?X, ˅#cQ_oO`pO\-<‘1xAoVd`ŭS~ cDQo+M癌Vܴ>s1G'xcO>գo1˖_/qOV1Q_hYd+o`WΧ C;|Wܭ ;oNX/`Χ>&{Ro1N[B!  a GqZi0+\%o92ž*cw$`_|I:qZ_y=?VQ%WO|UimKbڷa`I/: !c7c:VdEB ˍO|1Q+-WWMgy݇/0+QԬ1M~  O:2wN?*1B31_-?EA S#cQ_c[_?X/>0#d̛q_o1ݭPqoO|/~*cⴞw[ ˍ0?^=*҇I?ҘW^i_?'x UE}mq0X #c:cU=i1KU~_}?'xI ;mkvİ8%o:/)c+u_lj迻O Χ)'ڿS31[v".6b‘1BwkWu`0˄u??H|Q_+GchR0+\~5?QC ,Uhyd*_8R_L[,Pϋ[Χ!`W_imf_< V~ V1Qoo2_kr?>_iSqw&yաg܇a y0~ |VgU?+B O/‘0?NXE>*?`WZy1H|'w^!_y-_-?`WJu?OR?MXE~*ڵK;?.?`V?')dWqZ?/%C/~5?'xcN?G`V?HY~|Zu?SGTVط5}??H|'xUZY޶X}G@[Χ!`?V1AozZu[VX}C>u?OR+^&,3]!On%.u?_RN`#GO/:t~@/`>|*zVoC_}CŭS#cQ_ݽhYco[>_o:> НⱏU?+Ok1zv/0+\mS|3Bw?+T :̿-I~QυO+~*%f_X}CƧ)'xxo / XO [x;dcUgYo.?/)WǮTV$%~P/qO|/Bw:oyb/% X 1G'xcqǰ ;̗m~A Χ)CWE+v:gyn>_iS2wmWVb/o`0?b‘087WhYb/o`b.?Oa>U)m6{zv?>[aS2wmW/V//b1qέT`֏?Q_P ‘08;|uέW?+C$U/ 0K\~5?QC +8^Ǧ//`Χ>|(?-Tt%f_ݿ?}CSqwVp?kB+9W{A`WJu??H'xW!ZYi}C>u?QCOJݵHYp $' YcAvaA [$2_ ucu |pё ';9eAnPr&?EOCk2`pV҂ʟ#&:>%o2_Y_Pϋ[Χ?gLj`㑌{ wiշ!~b >u?V1>ish]i"0]ծ@]cmȁǖ`mG?ik{M-h>^%g_|,m ~ <lcuyc4U~N!Du{I֑is_jZien0$PHn[uד͞i{w~6Z-;Ė6&xJL-7_}t% ` ou}XÕm^GaQʗ]vyr.~CYO"#ƛ Ew&G> 9RDY_[ujHd\9WI !drr?fš'8~qj9P]v<ŏ|ؙ㲰{_ZԤI`t~s$přyn%$gS/r(;-l~Tk S< 5u_@0hRk $:2::qIm4oo+F .;mV5O? jLĚw^GrizeΛe1y7qF]Y^C}t` 28c88aOď챭¼ ;?< nZ,'td,wxWISot鮡|鮝kڧ+[Ou;Gat8$ͭݖ{oo6Ρ^gk%6߇D7?x)c'lil5Y/{A%֗e MP4qHm5K߇Uz_Ԯ'ӯ48YkI]h[߼~"H[)m& 5wm~ߨk t-:i5 G;\dt@xlo 1c8ю㪁. 1#< .AGP02,T`a/#;EW_?>#R7w%Z<ڸ@k$D",}XB6=`c8N# Al9q8?l} ?>5~"xKIoZiz۾7]SZ]ğ2@Qm;r=x'ouϿ^o 8Fx8 P] ӮqEa8cH:k'APs8wl|.Pp7? 1giH.gN:hZ`{#DC}:kv9eYw6?=`pO|] GR+Ok5Qֿ?YkgOi+m.'gĊ]1GprpW8'<|1pqܜEps+𯉼L-~wFqm}:P/R<~o%B{y=txž7MmwFarlFZCtbpc3 ZjktQooo/;U^E9~T$m(otg' H 拵 P< ;xFbhA,ۖ.Ǡtg{u!`?yAEp Qw=J.=~]ҋߧJ/o :{~(y(]ma? bV:{~ǷtQ{y o (]ly(t:QwAEp:{~]`]t?.[X61E~{c}(6Qw6 .Omy}:{~(y(Ep1Epb .t:Qw~o :{~_?6 ~u@]A?J.t(Qw]ko _:{~my}:{~(yOmyOoӥ}o (~?LQm=bE`~]OҋtQw]1](o Ooҋ?@yA1EOoӥL}@q 9ǥ MDG2j>n־'uzoqxRZwkqy-l/)NiE{6{9EGKW~ARX|rsn+[oq/]o.loӺt9fUA'J\\K#YXiiV4Q܃WBE#5K{9/ң.3 j7K%pK4NY8go6>Kuj? j)qG>_|/=6]~inz\_7}J[oo$\$V؞N `AyڝW"!}QkSc+1| O t+&kZ_.FT[TiY ?>T<7š&^KuB[QrѸڴeV) r(Q%2۵vc~ <uV~ W\麴S|j4Wmq狑qr 4﷙Iy?pO|k-aGceq-m?-$!"]ĹIl:Gcyrk( X9=(WuasKOok[Yv0e1 Cv91,H"iC$IgFhIYNGǘ2 O%zЖP>ʜϡ88sךI%)%kl|MKi7U֋/.kWͧ$YxF4{ pQ, TtߴV^7^$qg~ _\+]9f}r{{Gy.R'L#^D^Vѭ=ׇtox'cl,lz N}P}KmגaǴM-{Z%kƗ+ՎI @MD;$լy>)hu/m& hkO 1L7~ _$RyG#+yZy)k!e &&g]:[˻mjBp\r& B,[QG$T[#<x*Gu:/}_Ui VuG<9a@Vom[MѼKsu|M,a GۼA)?a_!լ kIF/>\|_awvzXAmrzz[Ի[~?>>o@9{gčO4;xscŞ vn~%f屿E:ʶ4=k Z-4i֟׊,,|1"x'&l~xGtxT~m/ɨj0ߵĬkemYZNL.5 _[^Ɔ.kCLqi~&5)ѵ ]Koخ#WJ[w۟|!!VAj_|5ě=WĿ5Otj-ΥXf֋A$./59K ZIG_@}~'->=4dvρ:_u4 xk\z_izcVɝaG?ſ x[߶O ց c4m;A|+ eiֶ=ܦ(T-ҰiB}|uhWz|:]oOnϱ+!P HZM6Dž[+I>V_cx/RӮ?gQ}&]џzk>{};vK;I>ނK7plOS%|/,:u?gI"ӌ OR-,v7SȰd~K]Wc][W㏄gMZOA7Bػ4iǨ8,h~? >"䁑$sǦ;6ߏG{8$\KIt5A2ܵ΢?C֭P|= x{KLLj$zCiu+x l.4e_[۬@n6 }J?ex@@ZkͫCԴGt+wg'̞)l[&icolz9<{|sWO7H@W7/Gdzψ-<}٥P$?@JG[]y1"k/.p?CIiKa/|Ym]|ZYyecɤy6?^ _KF,]:֣}si(Ro쫕]: Y6 G <9ΑsRk->[MW=5-pֱY0pFl|%[_=x㮓j1h|w.Znx-D~7Q,2bFh`ֿdX|iK2Ws+OkXr^Cΐ'7VT^YT ! 2VZ´g̕ӏt־IoDbgx&ͥӳVGc濂o{O}'u/x\pco~$xvFkMp:U\[Fs[kdĺpt9rF"d䢭.IKp]/v)wboٟᮉu'sx:~-iĺmSXvUG@Z6~$7>^[E$`ۚ|姕ϥg]cH@]J64F,=ח~r:yD^ž2<O ~o!Im->c\d" #,N/_BԛֱO|n}mi[c-u|!HUoJ;{&!*(̏WdeF?FӃ뎿]/+4WM}GJ9sG+Egyc&*J:PR2F2BGzW*=(c8UUE ri`Wm 4;-_h^fwKӵ|IS=2&I"`YA vWcż%-F#e]Y>j22dxuW[tt#+jtEC?>1x'$c< [j.]ẻ vpj%"I "Y1b ivMl*>ֱQ ? t{I@ pװ̱Z\_:hffvOÚvu_Z_WW:\:Hé6iZu3<ڣ[w;X5,}Rմ/]ҵ-\,uYҵ}Vt[KmԴOOXgh)cՕ%xsi~&м#ᗅtOiIxWCY-f4if-5]RG<~^tvHAz |~ ſxDVt#u/ZY쎙?SҦA͞A{mc۶0a.Ÿd5/?bqK?uڿ<#~onSi4}?]iEZ4֋.TJO,PHEt W k{˺(k ~x/O^#W~}'-"E[O$mI{˺3'j+k WPx Y t/zA-O6F&k3شBj:CžյUփPwik6uiXE0ƱM(,א]ВOXjT%5BM[[eѴװ5Yf]OTks=ŭq:I#=LXjAfX,Kxߵ}t^φ4ok)}>/?77_g&olkti𶑪Wt /\Ծ6e//>ۨ[[ןws],Kc<|< Ί5S_hygϻRϰY44 ninQ׵#S(OѭM7B|؄h7S'.eg,@] e",Bݾmi6_rڬVsJe ]lkϱ-1oek/;Mnn Q/N}MD,@] bYV4NNmkV{yt{'ݕJg,_ K] H滷m.>K]Xi m5ѹ+Cˈ*qE`],yĿzL;V0W̖Vky4R]_:VPyox[ - fX,KG<%cТx&kVņaK2^$kW5Ҷwo+A,mY 5%J]_\֗ޥ-#C𭷅.K ˨ u\__]]r"-V]t;7 dG4tˋ:]7@l$KH,5;'Ciqwckkm<i9 $j],k/x. ^b~^@c֭uIDڝpMJeOH0 b3E`],Xм'o  o-M>_8[}>/?EǗnɷ%`],xTh?1#hGgO]߿Qg,K? xxg@<nn?-?mǻ; >gqKv|3fl<;/|O{?E_LnԴˋ++-GQӞMbZ~ªKt϶a?Z5iiᱬ]H]:ޤ,;Y}h&t >ōSþ䶗Yд}^[K}JMSKWknזXĥDБkfl{@HB&Ņރg.e%wFbwǦYJ[n*qQ} 5}<jWڟ1^ZjwuIc5qqj_[6:G3:i̠#(Y 5#~%^ƿ} ZEyNurM4v_Z"[# 40v$Y4ɭx+OeIk%ۘwj-GteUWA,Kbτ-ymwG4" M+J,J4إet[9 !uhZ,bľ',fд|?qaw˦YIk]iQt"1q[m Y,K~]s_x?wށwV{i7ZK;gOhIMG7mܩY$;GXCi6YCjbV$,kc@%Fg?}ϰ~,`4Y >h}?g,OϰYE`?Ƌ>g?}ϰ~,`4Y >h}?g,OϰYE`?Ƌ>g?}ϰ~,`4Y >h}?g,OϰYE`?Ƌ>g?}ϰ~,`4Y >h}?g,OϰYE`?Ƌ>g?}ϰ~,`4Y >h}?g,OϰYE`?Ƌ>g?}ϰ~,`4Y >h}?g,OϰYE`?Ƌ>g?}ϰ~,`4Y >h}?g,OϰYE`?Ƌ>g?}ϰ~,`4Y >h}?g,OϰYE`?Ƌ>g?}ϰ~,`4Y >h}?g,OϰYE`!v;ӻ8օZ]=<C ?9=s˧\%imxi,ۮ\o:Ri'w&ә5VfF }z>,Y#е]BuNTK˿^}z׷X82onoaiR\k*b% N*. **GKE*=i]=[#>E2=Υ$^d܉#EX(W8- h$c=z`|OE]RZ/}G@m--VQab$nG$i$l"]/^>ZWw~ I\bmZMfݵv:qk :8!{.Ҵ%D^VOM,G,B8z jw_qI8k>G 隌?xI۬? Mc+_?!?uuaG(ů}{wn61M4QQ_a/o(ϵa۷?[U`V}!յfq(ŭ,v}[ nAx_8A,ҚVhP_o%ie o?Сv?PQA~yG^;}۷}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Q_*CBxAF?AxhP_n/?_?2rqV?nP_SQqnP_S(ϸCB exhP}OAx_8Q_+?(?_ie_?rw?(/o>ݧ/ƒo(ϸCBvi(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_SQ_+?(?>ݧРNQ_*CBvv?PQp /> >ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP_S(ϸCB exhP}OAWOL/> i(/ )ɔe~!C?_?2w?(?>ݧР+&Qq?۴~??СvBŸG^;}?nP^?N1oeР_ a~_(ϸGA|*F #e<,t*z :Q[ӊK/ ^mVM#ȴ}?[ڷE+e}E?uam4QA?Ⱦ!:vjE("װRSmⰻdLe}XFR}Eӧu=;YtcG,]#UԴSMMԴR{i")tYYir}%ttfiv&Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]Qe.]︝=b%(E` ,q+7 N~?LҢqA7-z4}Nz /61 ( ( ( ( ( ( ( ( ( ( ( ( (q1c$:{??~+߉O|gQ-/C׾JķZԠ&'t ;xCLzlPxax{4k(:p7M]SkEg.=|Q75Jԥ~Gǿo~:|:hx+\?~8|RnZG k=Xrךg/|)h>!/?,u߳FU˩5Ԍe $]N^K>Tv{.>c8=9JqsjZ+mhz5+RGŸ i6]ľuqskb {|C{=oxkVVI:}0M'k^7kn{uJ8z2JiK[{J{4&߇` .nGᏀ|iu;KVfO~_Oj:֥cnڭiDn/=S]Zե]U˪8(R$mB/{鴣&]J*twi=սkmxþ!?/hkxVst_ ׃LԬz~0DvV?h]^"xxД\%+eFi???#q{hJh(QNZj;߳ojWC>OGM;Y"dٳ1_pWi,k({Ni]]}E+e3cX˥?gȭ{'t6eG> x?I6>"iw_ ?ЯeOX7[gO>i[[^kCjJy]T|qWz85?yDrS\v]ƢvKOп'goet%83\J>cۧ;6Okoq$_%o=/owށUj G4 _gMþ4z}-Y:fk#\f[w"*RhK=l6Z&O)aCa:&vnvc|zukMSFmޗ{˭?R[Jn<_i|Q% -,o.{p2?Ftڄ.oyZɍ.#L$c~{[_e}~$~?coLcL i ,m Ĩ=`-\74eΡi2 Z/z c}S+lݎ`Tn\ok&wk^wWV=w G-g?;|Q[/U/xᶧ- ?eo=;Ɇ$3O\ź\5[rn/{ڸ_▫U~9zTqU^ZS~ggO{OKyOx+|s |o\x#Ѿ]j~u{?k&5/hV-rM*ys/ $׸U`T*ך4J*T-ed{CKN4g)rJ\_ëVJ(χ"Me5KXSsGi':nn<.53kR֢hn|9ŭ'ZNEYb-g:nVyWu*O<=T畿ͻiџ]/OA?OX_e{WĞ-nľ gַ{g#4WwWjw}jD=q^?VJWQPj֍TIPSSq ғn,ѺWZjyoۇ㞩{n|/c|I[H*xu`@wE.wY^x[VZVq]Hl/[<%Z1)JT]ozqKK KF<;mx^R匮*VR:{I. ǫ/ڋ OW.þ NlhWDk_6Wݐ :X,M*KUT6kM|7ҦUS<-jJM%/ifw]hh?ndfҳ}0*%R_.U)6\;]o ~2M{[o`4[,V.M+!/|=^x: B[{k֯$ݦj؇V4iݵmr3zFQiʜ'hۏj]-z'P@P@P@P@P@P@P@P@P@P@P@P@P@yY?"gv_iQ8  ʽ[OԊ=G_ZP@P@P@P@P@P@P@P@P@P@P@P@P@ݼ`#^1xBcZyXOY6ٷn;$SC^xsVƱ4"(nu[),oТ+ˈ5`Q:Qwql=N=li{,M'u%nkEkm.x㿊;/]S><Ծc-?Jĺ BI|4_6tgCYyvx*N.ՔU-T-ELmt'G JQRM6읯30>c@3 i:oG37 ,%EX[k#JMކZ?UN]s?{~?k wW:g }C_|1ri-sB5;;&;ԭkIsh~ʑ(Sj3jɶ׼N)龞%)AURǙ$M4ԚVOI_XoxI4 5-.¶:puOŞ 5K֎7SzldeG(U8t6=ٳS#_c/~cG炼K|?_eh"мC[[lG}kMt5K7Ug>DPG5WKFSV2QNROk+[.jj:..MF? i~fZz/fS᧍Kk_?᷉5P0xѵ CKNyV7y#eq˥yR9ARc;'gMyURQQq~$7^k_˟|{/^͟fOX:ψ^zέȼZ<h^iw\j6{cZ[>m'U̷w0n7WhMFc.yիzվ7˿@%BRP8Ϧ~xNt/?ⵋ5F_~ Ć Os%jV;MGĬ?ʥί)'e +]%-_*ۖ2jdjY'_dhៃP-+g 57EҼ 1BYOmbz4叆cmgӮ _+mLj18a8Ex*֋VVjY${IWu+~W'|9V C<;C׉4TԼ'gins 5ht!jc51h)\eEr=OcaHԋ唣$޶_~' .>? ~G&x-khkοP%^gdqV;Vahʵ*:կGtY=ѿSV(FR;oDA>"G ?S#X8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$ˁG_OD>k$ˁt_5/sGkMAzz_rM@%˴AcՈ>:KC_D@?g^F3^=hkܩȋX? O\(YmUr"RϵD@?tJēGֿʞRϵG/߃z_4ox4-obԭ^oͬ7X q 7^B ;0YX4KRwM׳m/بaqM׍HԖ_i<\ɩi^Ҽ9:R kYu tt[׶d2!BbPq jj iM+{(-|X\d䜚V巑'.cULUh?~_a/~'.cU??Sb6j˧֟߇$|cLP"ګ_OD>k$|X~=ih'ȂbVҷy"~ F`sx?'zmTCX.~%.?c?8K5]뒧#R^ֿ<6lcgV#0wwz{Ա:~idlK蝐'.Ո^;bO޽6*z(}K_=? s?ia??%˴A?OV"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}'?ch?*!,O_? O*(TCX8A?U$QȇԱ?p?ρHG ?Sb"~ V?G~?"RϵDA>"'OD>k$|XE8O }K>H? ?p?'}VC4{o"&q m°U G:+̱t+Є)MQ\Z(o$Tu`ryW+Kދ{&C%$^v?\n2ڽ_.GIF) ]vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݅] ,vYv (}Qe.˰]aE`‹.w݇OAdy8tEwӦ+o4i]oM мSTe W>-f{SO7RփI:|I=mk]k]ZNڥ&ne/e}֍ `A<$$TOFtM?viم޺QӲl؎r01/nO#2$˪{vEgi/+4DU$_K_Q?N3\g#0O ܫKhjW/wI)9{eukݵ\^ӷLGp7AE6Zj}[^|7Ikw喫ū[O dqpp ONG4ҳ}-}vݴ{_tw]_Y=.Uӵls:tj2l%%pG߃'dOhe2Y66i[ܢѢM?El,U\ɍJ*_i^*.NXyJɴ$U 3XsǗEhԩM;18?.NFVM,u{'wou-#XKT۵N>80H$vrFqpNҲKJ־]٧wsi=6~Ddk7cď xO-ď>u7MA1 MBK6ƾKaz?hiSe%%^Te*)sEMƤpF\z*Quk QR;続^2䛋$^ҭ5p=s~%m{4sZ[ |E Hbw3E/?!\P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@tq<2An+unM^GGtk]||_㯅oq&E/5MS6jz֛_Dʍ^,x?g__Uφ<߅>Ѽ%OMann4^O̶o+էG"8xl l3RCzՕlx,Jڔi)n+ Q8.Z\B֭VSS)Bt0-:nM]wߋ(/$?OxKĺnx E9U]uk+iz,f-u+K%:96bhRUN K |14"2QRF5+MR{(H*Qb%:rq?J^󌽓P抽G7_Zn>~6||)s]|:4`i#w\5hCIcڵ먋GFL.&&s+k,u./_b=hVYteGS %J+^<Ы*,-hS,LoZ|ܮ44N$A|-5MRƥ&~kz.Ιug/Z5]_4$5FqǤyz mu WDkⰾRƝjO1V8N*aJ qRҧJ*N>Quiҩ{hb'̪NZJNQVRIZ|?ğ 7"`῎5~%o' $xF'^&?h7s?t!L:Ɖ=ӧ--fX||^Nj>p8^q/rYcgYy-hKҭOg[3KЩ9T!8Cҗ?y\bܗ$Z]{.KaxB{Y|=麞_? <1e3u{,!{}AxFO{%|[X\?r|vmIpViF5(SW̩a>Ĭߊm5iR mKNM=^$_jq|_oq<2$>>"8_C-sN0ΨecóʄsYP8PS*1N ԗBtj*Sé(F0rq(ɭxv⏲y0ԇş3i ojڟ|/gς{~/UTsDb*BnU4jԴ|Oⴒu!S$>,|wHoQ|b.ZxZVZ=֟i<5sqȴ^4{QK# .N-yN#Y`U2Fy,ʶ:&Q5SZ.]Ry,){]aY8yʯ"?eG],/OR\[c;qK.Mr_ }sMD<{KOftF}Cu[gi.򊓥ar<[^ӽ9)T&,(bi/a+7/e)FopT7g)FNIsjRrUݪ\FxVIBS\i~ x Ols4=CźmOHգ#4 }vimuqSSU0p5ԚX)JP//N*N^"TZR$FrQ%(Thض_<}a|4_)77⏃~&=luK;֠MI;m: y=cxj03UPhЭ_,f%{7ԕL,e^ug:*+*rӧu+J'UT_[jIʚe;^4^[ɼeOwz>ǿkc*ik{?5 SO.fI5oW) vc<]j8 u\0xWy).|m8:ؘ` ќbFeBjѤUUbo7Zz,uOeԡuO㿄<xC'Hu|?MJzGh1ǠxKmx{Z?eYoXn.:^SWژ|&+2{9s'*nobbp0%_Vȣ ExSYx*W/P{(O]gRX6ON'9RkݜJq杨X흶WFԴ{imҞMXU/lմNͥVNp >K1NXpӕ91ST7ʜӌߗ^q,c42wjO^ZMkRwQkm_h^lj&-o~3`cX5<7'k$ Em!,z­>ִ~:x=)bh1-I4C 9b>4ů3(rdKBpib*/e9?cR4RiN17eP<gټI9OO|+^kɨZ|mGo Z/F|=ix{ZLbԞG 5qF40xzLKr('NP.5TѯI UtR&6oIҔ9NRHs~ qX 8g <w{}^ A sQ4ZͭyPC5,t)[ HN73Sns?g FiSGa"y XlR*iJ:)SҥyKJ}PSmK1|Qq[j1/@1‘ӐGB?xf[hh+{YWoCjE_l߮fN>|9h~ŌUӌ|ڟrbчz}s ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (Oo+yN#~_ڷ:^]j:<Ύ]Os$4#;bIig tqhЊ:p! pIEFEE($I+#8\" Uh*)EUIGHVKn?gg14\``\C[1^vm|;kT]C>2PzXxcRtOkRw0;hgю?x(c? i9`n'}<v>w? '"~<9{}]^v/Sl6LVcc>c=7'V'=}̣+[^FU+׭okZ^])[kۙl5?jα;׷z jz]_ֈ^^,O3I+3~ _ &$`p4gz`ңF_ E('Ɵ_ xkUz KoCŴ7jֳ[]ij*T|^Tup8* 7fNQn--V#j$FVNPi;Zn&?Gu6q'1ɥ=`vӯmcd-[;-!wem=$B?ܕ+kͨF~F˚N;Og$#fVnIzݮڼw]_v]wσ8 9 wu?>i>ҷ/Vlݬߵ3?|?x(t0NԿOI/y$dm٣x2w˜ ;ӼP $Wմ>JXcͽXeTW])q/RJsk)FQU1UBi*SR7J ۖIaOu[VoK5k9ysÿ w~|%ßf/d[ϳ}3nɶmb,~?{{nEV7/5VRq|QUWֽiֳn*sj9GG_Pյo_ uMWSԵ-OQyCPog{۝滽I&y]Iر&ig pG9QJ* ^""c5crZ%bB ]h (URJVIYY-?c/ m0N5Ks,0J Xz][_Tx;?F| \~ؔ. N۹}cmjZݭͷxAVڄCkIl˦irÏ^'U uޟ~moA_|yO_lA k+x.fWKU&*k܏a8'3~Xm? F8< ;xާ2hK4ˍCK/4C@.=Z].MWE9uIMGPe,vi4Bդ ), n1/$d 'YgEu}Bſ zlV&#Ǿu\Iղ> o]"iDp~gc؁pGQ#=kxBGG?ek:_e:v6WŎk=;LM46`nj@uc;[Z@3d=0.Luא,  @`ktq3qHq wl"yqizeޕhWw6ww:&6A:ijR;NmI3VV6 .dϸ>gw_uNmh^<)P7@m [KZ1 G4LJ o#8h!u 1hkQx#$￲5 XG!mu~4 uƫ}NkW2y6K+x)$u 'YFqk] m#gy;R;{kmu驧n6 c9{d䵆gOL6^G!Y SpO `0A\Pkszt1,m `k(м?SIկɨx_y?io{6Hxk>"Ow$3S%ۡO6O׵oOY7GM$o+īAb/FO\Wx4w=1 >AxiOr>g+\{mⷻ/i/5{:Z[^(sܛ\;~ )x89g==qzG=G\эYmA T`Ov@߁<-KRV_{߅D|CU״/q|_Mu5[lҺb]-VYB==~G/ "~!~~ hWRh^ |aYGee">";CK+ѵ.YY[Ͱt>eGm?;'cGf'mBķ3j?쭴O&} ^/ xGkkȯ=;࿈. kƧxMM_΋HO|qh[PGmº[?k6X7Fu=N7y7"Jz$B~)?8^Bҥ.~!7cӴ+8JxVäVݟݬ)Ɵ^(cM3S[x _KԑX6+H.%,nhLD]>Ujw @g+Akq?([J֫VOًÐW~ךL5o4Dukrkg5񷕊Nfgmk)tCօ~5*^&k.{PDž$t/EO[j]Amg᷐z'k7e>6 O]zoģU~ k 3^bx{E ct  5d?B^NV-tj|_&guk t -hsxz9|_-u׆5Kˍ& $!GG|KoK?'Ct{LJ|5?>,|.m<#!፿hSٗ_tEρhWZx,+yqhf4[oKC׾W_}}s⧌ᴓH׈|ieu⯋>>-oU]#CK{+b[ɩt-4x'P/YOᦏ ?;|5 /Z7+)%kQ^/:.]-CU,NVO.eQh}ϲ~k߇^'dEg?o+Kk <붶=5\mmݝrê]G`./ KI_9*ARltk;WIq|Kxω,.REiP[zp>̚?h:; ah>ߴSZ֔ů~?|3} ~kOxTԤ}f-C m-n/UÿZtqƏV^&4hx~?~9|H i ?h-%Oe=CLEkowBNcti-=;şg]{ gO|A 4|;O؏>ѯ]iZ>{;bw4+Z[ S?rGn<5uwGľ}uo|=i/u ~M\s!Ƈg{M]kE>6)1_/١:UAcIvO"_NAŗh5n.KkZu+c4|9fo:-W᧋<[¶Qc_g/G~9'T{ xK +E%ivwzED}KE͇|]5$?$~$Ϸӿo~ѠM%uA,_ںT~k?ghm3{봄P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Po[B/& I>xCD3c?t4~G0OO}1ҹOҍ10_-wk NO ,gY&\7 7 -)֛ho w~$;=z6=c=hA1ҍҏd}3{yP v3gdրgcƖ&?1_jh^|Wxv;Zl_>b_0on=om;cmct9'P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@q6U"k_ct זG GssܗҭeW(gxW̘9h28OZrb(I= O PNz0iC;]P w5NwF4A w @?/ޡ5Nz0iC;]P w @?/OwciC'x\c܍@mNz0iC'x\doN(Bwr5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5Nz0iC;]P w @?/ޡ5~GCFxϦ:c4NUt='Jo`h$|C>I]dBWd-AN7|ÞYE]8W}&(| ?#a@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@AVڄCko#O?(hCt?1_G0OaP@P@P@P@P@t#Pge}jj¯ ]SBEkzUƛ{qTZ^}PwiIo"2b24DKa~c|Y@r>?t"_Pi717Ŕ#wM%Le~c|Y@r>?t"_Pi717Ŕ#wM%Le~c|Y@r>?t"_Pi717Ŕ#wM%Le~c|Y@r>?t"_Pi717Ŕ#wM%Le~c|Y@r>?t"_Pi717Ŕ#wM%Le~c|Y@r>?t"_Pi717Ŕ#wM%Le~c|Y@r>?t"_Pi717Ŕ#wM%Le~c|Y@r>?t"_Pi717Ŕ#sM%L|Y;\OJ[|c|+S3Zޅ κOcuIdéFӯ1浕wHcXM8O>EOV^[*kmFXnH${;IRݕed.em۠%\?lٗ' zħgdrY ]?5UR > 2]k,6?Ew 򪀳'&ʪϰc̟W|?* > 2]k,ZG]>xQ:ޭw M/HѼK7>DmlY&Cm Ӿ;cG8H/o Iq <:( yc9]Rwo|yXOo7Czz:;v !lv. w׿oԬ6זN6ys6ZjuN[rX( ( ( ( ( ( ( ( ( ( ( ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( =:{ hx"xQwVWƞ)MkiY[m}៍fֵwo%Ri-m-^ܦ*+O~*Yß |!'a-״|Z+o>%{k+=uRҼ=uo麌ivEa%d|pgV!;r:ÍZŸ|[~i hbBa6\o+ -ZP}UuOi|_h,ᖙ~!^Үm(ooh4<1Rq:u$xo_ ZAᦶ74.X f@-E/ouc@p3D~xOZ:a:ַ㟊!~&h:v ³.'?28a4tZjv^ P_xo^kUۍoKWm)umw6!ֱͮɢEk$:K,ו?.O_wxMY/b]h<;[Q@־ɩdV6t.oו1Q|)|O/9{'zFM{Gǃt6 5}:k2Ih5+״2m=7_ػ5=3ow(oTxgCH5GI-4mK>[lmar c/ xgm| xCTPm*KxC1Ɖ?:kom-zLZ56[I|t-|&?G<_])u!%>W[Ru %shW:=uۉn< E?go {P^}<_뺆h>+=?Sj(?ǿ h7/ KOJ|=6qQgu CI-=Zy/ >c$ ?zNƺ{giGtoZ,z [OZ_ oL<@ +KϱzxPs6GykygK,|Hdn#.擄[幰+:+09_c̠fZxW>'υ~ /tR$|J,o5ۋ?᎗w *[MO IXkZ]nm[y g3|U5C>x?[/Z)ivz|+`x{^ - ?Smf[6v/W'MOO|EO?=2caƯ| hƭixf {=RZÃ^MG]-md쓷nI>5ӵ?7@k)} ?%\I隷?"K}|D|#-o¾y_3ρt3&3öo/υחoGwq q>ZVMhy-g| ||?qE2Q G >?_ !KY?+'~[OW|)/Et kcm ɩj:zG 3GKCRm%J]]Ɨ ]š|Mx4|%iz^?O:_ijmc"O}{]Mae}}t/oɥxwg?Y>-oCԬ->6!aO\u/Xo|4 QI.7_ÍwSyO#-O]gSx+>Um>9 )sSܒ_֞$[\j?4(BxH,8|Q|⏄W#Vw_|U/>Y x#zOŻI 7^![[j_5^jibWﴟ)Լ%#NgkMF ]ŸĻ(jjySnakZu}ao} ^ho~п 4 V׼e _|+ۯxc-awcvXGu5no]߃_k %'<%Uu]Xh:γa^<<4~~kg/_:kK}h[Ŷ^ ֵy'_ҵ6}7= %_w?on"FMRԵY1\_DW 3W5__Njc^!֧ _xR&Ҽ]xsKk#HѠUТ`ZDz%Z < akKj-Dž[eoO[_(|)i)o4mjSo x.mbojbY Qv<bgÛ_?GgcGi[ƶ~֥|kfkKB>.uzF&t]NԾ1|=cu cPZ7Nw}YwYˠ_9|+Oɪ"@u FU嶶 )uar=4~G1ss|P@֞F^ŜktcKIRAD{YuKO!K#zvj}tOSjP~ֵY۹c4/5hM GM;BS){Zw=<Y>HP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@AVIO?(hCt?1_G0OaP@{q郜l1h|>6|5<}sKKtz|ٮ4MSڭbDg;}[O֭bС=7XI,zjOu ޡ]j'!OFo+w۟ \Z }-s:~u t-DCXmY.TM/ĺ?1+~ /KFԍS̖ M/^ ? <[jG㏋p&{~|PJl>-xQmдEas[u`RIh% >uCu{k(о#w^Ziw7Z"xk?MksxU6=GÚġ-kXW}2~$mOx_Owg?_ >6m_Fɦx8k4>2H[O1[8vo k~ &V!TM/V{OO1𶈺tkx3ߊ{OŪCcq|ok߻ަ4o _efyDTxſ>3j%:|qu??Zl.4SzSW]'  }?Vކ=<[. 蚧/BJZɩ>!kgCҮY"'t=c99<ό 4ekk:|-CZljlO hoa{sT7/ D5h^y>mas/xs7X;~xZGkx/W^"ipjh goOX]Zj^h p*cu"Z;6=7>(O~&;O񷋼EŽi'>|A:Aپo2 0]6%K9;`GwLx_?GuK4O7vƚΗooǦkxjtd/s~[Momw򵃛=;=S ޟhg? |S4OMƷ{YAwa}[hԣM>)M,Rܫ VKɞ|Q4P^eGKO &ȴoAAqo Ik|5cou{qA8̀sF?K(g]¾|ZUU wG- 4m~kp"Iub#dgR4f6 a~xm4 kE?=_jiK}WqF,}̕Ӡ|A>9xN:8"|& [qڧFg|=&cf[X|=Sᣧm;!??>n"4  ZcQNٴg_^{a<^TP+h4so<~ xVNϧ1hIZiv^Iyoյ_/O$t^%t- 67lj?|'Gon|)+ƭ?|:n3I􏆾)оxHխ$_6Zfa:Zԭ5 |5;/߁;gF`y}{Q4/ kM_ 5 jxY /.OKӧt;i|_hj:qGs׃S^LZ>-_P)gKIq]~ _ξ!~S6kEܶ!ׅI +мZox?<+/ ž?oI-Kជw>>S^xtxi-_ XG&z~VIkY/~ f>>.W4|.~tmԿ>$YyhmG|Y~&|( </h}.u{ŲQP_K?]-`>f-C㦟3> xP:KJ~x~)CRmգ]ǁ4xݾ .xf9Jm 3OS>>)\K%4=?Cĭ!Mkh:EPh\_6Ϩzjo֖{X/|4 K?bo_ěj"> x7V:t [GT>7/-|0eDKO9%=ğ ]|b$<3kֵ%h$:>_ 5߀>-tK ^H4]+[>c-*}U. K9Z߅>-Ekl5?h-|XIyNs&^6?yi桦|Po|C&mtkCŲtZ|!(Ϗ> h:ށ7,ْ/xU~?Y/_7/sMuKO iO&6? 4} =,.m뺯j~w?]jQxJ~\|0ėKPï׵!kZֵGNKdCrF~sǓFu2o|5k׈4KO_[Þ%?o练ÚhF?ƲKX\߆? #ǯJ8Rӯ-x#v?#o\|?zT6Mamj 0O=vk=1Gin-[UFJޅ^ӡ| 9 o x ^4ҏ-+|s>ßoh5}TUF\gx[_Bk l$|M|iEg-Yv?߯Z;axJ{,t˩ⷶe;x#]YUKL}V ^=/x_ >t]@-ŭͿ4x5XH|COf\#Q\^iMXGOG'|x/þi>a"GY^[˛溹k.${eaI{#c7 v'~-xr;둁ۥ>HP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@AVIO?(hCt?1_G0OaP@6cӏOm=Ӷ:6 Py?=Nt( ?p( 1~&?.PyOoӧJo ׃( ;v29Nz{|o+o x[H|7Ϣ 8ӴWWn%^2ygi/,[%7i=!dtʿ_{1=:tktKZF?G*i}蕼ǵ?#G*i}蕼ƵT1ʿ_{0%kQ-uA#?#r/n[?G#ʿ_{0%{Q-uA#Q4a{tKֿTFUZꃷuӁ1r/n|]Pc#Q4a{tKZ}I4a{tKZF?G*i}?G81G*_jK9֣ZF3b4r斞o?箩Gu?G*i}?Go#Oʿ_{0%kQǛ:dcѣ4a{tKZr?#G*i}K{_zwYq9WKDAk=uN׎#ׁQʿ_{0%o ƴy1?h_/^5?{ ۢ_ ǵu?4a{tKT9WKDT9WK_pc]S,h_/_}k=uO?4a%=?{ƎUT9WK_pc]S,h_/_}k=uO?4a%5F=5hCz4IXkk +oeZW4#M;34If'4zV}47mŬW[Owgoq5olp,ѳCgf"X] A0},` Az=xb_$kz >1PAt=)|_ D>ء Uӌ|ڟrb=>P@P ;L4?Li/GBƻ/|^ii ^&mSQ>wޱ?7 '5IҼIOhGj../"m⇎uOK|}/? Bd/|QSX[|q/>:w {]㟄~??_4xL8^/A>,@] '׉~|>}?V޵;W÷߆w˭:~) Jxdmgo붟E໏_<ig֟>ծxx?gψZ2b+gIa /!$ռ[ 0|3ğ?F{gC5:i{Yt_ H>ڎ_\j>#xr [|Z*)x7}njt.a?Z:o㻏#'~c7n4k?OM-|_ S|GP5x Tk{Kuҡ炼->) u{>k5^!V'k饏"wo|Nt.I$~(Ul|Yx7`5cO5&{=1O٫W~ 7g.}.A6:[Z?I$9ec]o.>-x ğ 5Snkwz&c#؎e k~44/k]_ ><7jK=;֋{#'/xSUu[xv{ N+ Rk//!X?$?jpsRkSWF/1w .;>O xv?߆/38~O k6VkӼ{9< G<5M'gwO Ns ;w>!xK-ox>Ժm߽Uj qKuǟb/h^:i<; g#D𗏵/+Xm[_k;~_4-Sp]S;IiVryp)o⏂> |x? ~/_u x?goPw5>kO{^F^amy=մ&iZˡ?uG>a|'44H? wmG3fb#| _ ?$w_IڌJ"y#Qo_zP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@oa0> $sb1WN3_jɊP@P@Np<Ɵ5jڿ~-|Mevi> z[iM|O*KD=kCs:i:Jioӱ/n {N_ hj7Z{I|jτ%|E^oe[5/N;II?5 k=2Aa~3k&GO-ƚW 2KYխ{ڋ1Co4 ־U{i%X%6ߒV|<|M[j7E0._^ogswu6?^/ӴY/w} _Ť)|;&_>& (GR<]q^#V=>GPռu]cfhVr^[[i>f+#CXjמG?_| xNM -sU|6^8Oaq%ė3nBח #f{~!׬iQOľ=/5[-K=ޛc}a?mveѬt0]G% V:{<.|Hu{M+6t=V~1AUWY-ͤEΫáɥXk͞m2d45_~x/k_# xow Ŏi]j;k:ƿ1^0ں4_x# W|Exš;sxsS ;Lu9=);t] 7|qψeũ&V)Eƿk-'ƟI>5VzōӶ0 CO|?:7߄si8ռ76Nj/~o>)Ovzu_\:O 5Um` t T!wԿ5< Ɠ+Oc)yhiW^UQkzmn(|#x'}_.i(xz iZߋ4j:Uyhp3zi-冕oήb-7DE8ЎzdsH ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (!]O$P?*bO91Ca\ ( (]( y1hPrxyX?@PyX?Sy( 6ӑ@88Lqt?*6m8n6t 6tLw6961?A@LvNǥt_h oNʀ8ǧOʍyX:{c6 ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (? _Mw@?|9CD??z}s ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (8*5`|I Hbg+ӓ>0 (="/{&xM&^.Oo({+Y6_ i3,s\,f3)`֝6*)_о?fKGy„#LUvx]xL$ p š.C3?iCׄL 9 l׷-WRF0:q`/oԿi.) %3~G%LuO`h /?/9/g @_} ߩ<9:g(!댂r-<|HuK=_ڏZ4KVy>"15K/Fm5mJNm潠sqt{CA'TT?huFRմO Tk&WԬ Ӭn ѣMf%ekCEfKKŸ_p7_r_^Rr_O^v|zuLc6 ۢ_fH:/IEC6|Fdz\i6e?ΝcuhP\Fk6)++^BlZy!7%L0a\`9`-<oԇ܏38L07`uqN@EB_gQ0z4MYiAqqX7ՒU! ֝6 -<01 υB{۞hmvg H<N܁"<3 NhxsLFs1#rA ۢС'TT?huFRմO Tk&WԬ Ӭn ѣMf%ekCKO"3~G%LuO`h /?/9/g @_} ߩ~?]xSKfKKŸ_p7_r_^R043ׅ?/oԿi.) %3~G%LuO`h /?/9/g @_} ߩ~?]xSKfKKŸ_p7_r_^R043ׅ?/oԿi.) %3~G%LuO`h /?/9/g @_} ߩ~?]xSKfKKŸ_p7_r_^R043ׅ?/oԿi.) %3~G%LuO`h /?/9/g @_} ߩ~?]xSKfKKŸ_p7_r_^R043ׅ?/oԿi.) %3~G%LuO`h /?/9/g @_} ߩ~?]xSKfKKŸ_p7_r_^R043ׅ?/oԿi.) %3~G%LuO`h /?/9/g @_} ߩ~?]xSKfKKŸ_p7_r_^R043ׅ?/oԿi.) %3~G%LuO`h /?/9/g @_} ߩ~?]xSKfKKŸ_p7_r_^R044:tυN9AkKv;o|5MuM&~&mtcy&i\"y׶IVVdgMmk{P@P@P@P@Po[B/& I>ء Uӌ|ڟrb=>P@Pt7Ů?h;x~6pqzMiTtK/>bx__ǿ 9 OR+?2_|1u㟋:_~xD灴VG| )Ej~MžWwZv/f[?Yx0> x3͇Jh+.~3MWď0 B>`g5O/>ľ#s=>[=O^u?d-?iyXd.b|;I_Òxdw:ͷ"tKM@^VtK/;اc~8? |]isTY,VoB,|Q #x?|:BM. 4;wm5p“*٥~ˣ<5 |K P@xgUq$3huoM=`GўC|bW#±wΟ`V,?,?PQTl{?$~?!w cpyJ-ʭFw|q@Kj6|xw]ϟ<}=|SCz&2ԏWb1x[9>"SՔzh{"B :~ӓ(<ځ M{< dQO KsG!|9s+?=O 9q4-tuH| r{=EY+[>U~,xS|+|)9Uӟ-=<) uoRCEW6Ή-p?5x%8/P|5?2 W>i)/xrN5Nm{ WV1y G#Wu?xoW>-xxi_ /:־5CO^=|w{폎_??oeg*oχ?KEKG',_KGY[pt}6o!5l ')o :/g ]Bo?~g!!K4x;ß o.o=R@-s'X<{[{e񞙻 Ʊ^qj%#> ]5|ҵcOk~aoĖ>>.Ä ~k?a?,m7>!o׃O`޽׵k%4oh5x/Ӡ?Gg_~uiղb+E'mt7e/J{n_|dӬfOB}{᷅&x o RMrJzffukyʙ缾77^&>|uE'OCn}c𥗆ԭK7ObHmbC.OiW(x>9A>_?} 7W'Z4/"~ݤq6cʐוb|CPxO_=y W9 :7#4 x+wY?$zu+ E$h+k?n_|dӭ~j_h׾x_|@ׄ5.ojpե׆*g{yXiSGOx_&__>iq\4?>$jt½=|c]wr6-<7qi>֠m-g|o/'9<#o/[o? +Y<sHm$cꯅ<[N < 5uOi{ͼCa֢մK4x;ß /e._jK\Ǒ|07Qu? aj?|s~Ӛ|%(ԮG4Xhp$NK2զ~o]\>$|7W':4F+ OMĺ`)\`&yXᇌ~iߵ/ w⎁>K {R7uN;sP;Zށ=tx,48oRS٥^oui ?G~7~I|`o N-+0/kCO[Ƿ7V6*I+t>c `=Py0P@P@P@P@P@P@P@P@P@P@P@P@P@(=0xo ɗoO+X.:O_"5iP@P@P@P@P@oa0> $sb1WN3_jɊP@P@xdobc6k*:tl}2;G*I#n;`G @[MA8>` 6g8Ў WO ǷNxzq ܝZBq:gۦ0x#^9'k#vݱݟ\g4~à`㌂sFvPV,9+COum?G=RJn4][Qӝƛcwhp\\GvK1 yp6 i:^6iZNi6vzuVvnYĐYְ0[đEJUE{|鞘 cSTО#18 / :F8=ABM[JTԬ!5[O=RJg,mJNb..twA8 }f%ekC.Rnvn09 kv#09#gp6|Uܡ&YrVz~ꗚVqY..t {B88_YI5*ۡ3۠1-ǦFr= V:^ZmhzNi6vvZeVvXZG:}Q$QFZۧF? ?.ו}:sױ-1A^8+g:Nzf~hZVivnimV^٤V}0o GqƉ^Y4A=01`w9ɷ~@ ( (׏@[C#]j^PiZp?)\kc/$Pgm7#KơZ?@o1cٳ<o~@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f'v?O1Y=pqӨu)TtK/>bx__ǿ 9 OR+?2_<1w㟋_~xD灴VG| )Ej~MžWwZU˯MSle ~%χ6_)ƟƏ5_?0xkOxo~,|>xkúG<[еxH(_Ь/KEkMMo\6 Nܫf.s(A//CD_C_2>3 Uq$3huoM= Tz!o h>1x+TcXgO«h3:?ٟ/"3s.Wo|,Y! ˙_U*xi㉥7k˪C[i*Zcž^O oگEo^OxN/|n}Om̚A/ 3OkMFUƟm57]ԋ~4XR#׼#>%~8io֣xov_|e/46|QΛzhZ-0ؗtIhI#O-<J>/_Z|Vψ^ Ҽ!MO4 KV>ӵ kSCDUՌFCi+imQ]KǞ{>#g߅:pW{xWíkT>7M> wxŠRm?ǺUp뷐kI(-c过.CS&_¦O s$^Կ{2DuG'O3i6ZFp=~ &}an-_DO9HROz6ߕ+X>OSߎ?ur=[./h3Ɛp%">I ]iF z4kXW?~7j?<T6>"/^c_|z@uM/dݫk76ZZ']~" |^e^xM B43ནnM~ͺcqg=շsȯg|,Ko!47_|U:@xÿc׾跿Cu#L1bt_߳n{\wk\,PBm $C|O7|sᮃ}ƿ ?F:njtNy Ec"i]wwsr!e Zݏ<vg? Oo.Z~Eq1o~=Eh摦 XioٻV׬nl纵yDpAvѤYHWz7cookŶmlkkkVԼA?;o|Gm5GS/tyF CknDǏ_^}{BOUMӏ˫$&˃)6]b|EPxO_|._)WMׁ_^&ӯ^@m-#C[~)_Mu\u &kRF_x'&sxsS^ľ .6S;M-d>2x*4hB__m?MÍB|#UE&%~ zþ;Viỏ}sO𿈮om+=-d}U+x}_wuN=(|CWZƋ~wυ7v!/1=~/ߊ:_x4/G4O;sP|;x:ށ}tx,m48/VO/#!lj t>)|[i9h_-Y#W`WE{Gë[|Neu_>*n]F-r j抖6S| t>)|]i9~7П$[Gğ A\_Û{-Gx] UoEP]º5N>i\ -swCXoߴ*'v qϏKococ}Q.|AM޷˨ZEC]W]լ uD~}o~|t: o5=(H,?E}Eh"w]>o+ >E \ȟ?h_G§ã8Ο#BdcG%؂֔moF+hw߳?gOR~ x:mε]Yx7:Ş㟈C4~ҚǏuM.?YxSgg_׉!4엙 ?G~8\I|?{jjxK<"i߶/oUGu|;-,x.$sWV^Nm|A>4~Қ׏duM-<;e;J,[ jy'yZ.g_⿊ZxL|+/#X|w ö X>3z~>U݇ϊm-<]fy/ K~־ēJkOxW4_@~x_w|Aׅe5 *o ͥ׆.gĦVKkX-Fo*x;_~&mwVj^2Y |g>#xQ5CUmbSk.K]-cu&w^m?]3X|!~>+E Q>wx_Lu5 |Gmd F=,p?hOWz5zw3=?Gb]O7uKX\k_+#W/ xT?A5>R$|#>%8iOţޣsv_e3Cᶛh,=u:֥eWZ-mZ,`6KM9UgKAO |eWLJ~)x~*(uO>!|[>+h&\xvo0R? ѭ *z#bR^G|/Rug_^/T^7/?;/ixNP .. ϋt|?Z̶KC _n o|6E ?_w]wV7m o5EDޝS޽NC^Hb\xKWx[X\k#qҼ=uT^}cX2\/O5{j!կ) Y$^GumR~~c[o7f@uͮ؛yKe/~#W׿~%4mKT2ዏJ|gHo~G(XRI.>?_k'V׾&|ƶ^?&V-5#i? u< .hO>/CM[Wv>6>c `=Py0( ( ( ( ( ( ( ( ( ( (GQ 7/Xҽȯ ZdP@P@P@P@Po[B/& I>ء Uӌ|ڟrb=>P@Pt7Ş?h;?gM䊎6[Lq뀠ʒHێ¯5Vm{00A$#U:{qd'8WT/OϞ󃃜r09w'u=c:Ipwlpwgj=xt%~>,m;z}܂N2qxɷ&ӡǩS=y; N&A :{c(1dvPyX?1ӡ' vFVpQ}o:t?֍6GោW\ZM/xƞ9}wwZn` Pt7zӿrm[kOlzq$ # -v|`hf&>~}&7>Onͬ+47VMy|`hg&>~  xXhE/'89de[ SN}`?C;clt~~T@N>tR?@t@~ӎvVA>c~ |çN9 1y R?@o :/o:o0@xVյ[Wį4]i/-u7,|?: ƃƣڼ6=Cv:;o `H )+&1<;j?cöm[qoiB>2q ˀ8${qs WB1N`G'C4U#p@8:7|f1cm<6F88|Uӿ [¿ؚDH^9decTӡo/0+ 7r$px 6Aݷ9 ѷ.SM[_>?4m|/7;J?,4M"[,FoTӡo/P+vAӧat @c>A' %z3O5v#'#H794yXo~1zyǶ8oFAw?}29xO.?N1OLcQ@8P;yXFO xowzm@.ig??a4[zRwwq?LRt:PyP@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f'v?O1Y=pqӨu)TtK/>bx__ǿ 9 OR+?2_<1w㟋_~xD灴VG| )Ej~MžWwZU˯MSle ~%χ6_)ƟƏ5_?0xkOxo~,|>xkúG<[еxH(_Ь/KEkMMo\6 Nܫf.s(A//CD_C_2>3 Uq$3huoM= Tz!o h>1x+TcXgO«h3:?ٟ/"3s.W>Y~ G'mKT jci N|G KkMSBƍKg=߈M)hh|oɦ_O~%o"ρUi' x}kqFΣk5Wn|E.bm`J/KX>9~ !*@Osj'+l(qc>te(ؖDOr%p2{P52x_)caWLj>io~/.e}Vx?G?&%.nSzuȫ%kwGʿŏ xOzO>22?jsxz_g<=8a?i2j/ 4=5_?W}TtwR-ҳc?gKx^G?h|@=᥿Z |U࿈> |#MG>Ĵ)~j;e[Z>!xJÍ7>5/XrN5NmtkWV1y G#Su/xoW>,~xi_ I__PS●t6Zۏ7ބuIV îA&V $cO[ -?+Rx>R c<ͤMi+ |[N -xoPOz_aHo}M?^›!K=~T`P??G~8]~I|?{ 9=,?!'"մKӼE|)ލ@iZֱ~ n~,'V3mց ޣ}߃k tQ:vnm5IkQ%>&l$i$JA< ~g_ _Wz>Gq|2ѯ~<&kF^7a fZ^Wľ%Imm?_/> UKww|m9A>_}7W:'Z<" o4.9˹2؄nRZωZψFÿ7д sH~ Y4ݫ^k76sZz<[8|M٠ihKM,_ mh+=y]1S7|5M{l6εj^ WRWZtu>#6棩ꗺ<#^_F}UO~!5uϷxG"c/nk Ώ>c{a&ieߒeR #_C__'>?/t^g?NxGºo ō]~ 7^~om%n?ăJkOxW4_G5x3-|595FKÚ:%nuiuʙĆVDSGOx_&__>q\4?~$jt½"o5m{_=aO?'Y_]x][S双?)wּOh{%[v8_G ?CPt|h3ďXG:@Ka/$X4OJKN tW*u"؇Y1:H tD x%~ V<)1:|H 4q 7oBm/ۏpo-nnGW(w :&:s gb֢tKӼE *il|+wڿdOA_U)>$?'GTǂGM-umo-]aok&G~@~-]pÞ85OúGAZR%iCƐ.o῅7BZ{m譳"xcÿ "/fx 5Wំ6(ts_4- \ėsmX Z%C ?no|6E 0:o2~g[~ݶ&[Hdu?&>"ZIhύ |gh|ToWP?|6dj*մO cϞño:fsxCT}bR.KKXYS-39lğg)w XZ$+ xVڅ5/uMKVBNֲL/⿄ |fd~#~/ |- wqxo jtj. 4?욅)[Nꚞj&VJZ>smgGGOx(Ծ&xo >!|Z߈1m5x!AtM'B:-C]6(|(*O«W7|ky?i6>Ն;_t[ch>6}Z<2bM ioI{5*GY//~Wρa xC^gۏxn~ ~bE_Uψg:l'mcbxc_hu=Oƿ˺>*^#Uxkrl|+5!GNsq'oEz.’Ivx]:3|f55\ұi;ISIwBx!x]:ZmR~>𔑷Lw9}:-A@P@P@P@P@P@P@P@P@P@P@P@L^Ik?W2B ( ( ( ( (!]O$P?*bO91Ca\ ( (?:]|&pqMuEGN-G`8PBI$mpWi=  F =`2 vN+K^y9;[V}1Ltv $tz{dr8^Ns8;5]grF:`䒿Am`Q11ӌ`dqpA]:wg Kg {nIϸM}~mP  ێ߁@m`N01ۧpEyXw}G}= tgZ6ß~S^xsqki7⋨EOx޶ߦk7WQKB7SN ɵmw=0r07c<'4ko">}@_ov XhD>6TӠo,[6 o">}@cov--aiXW}oh5M: 펝qS8qӏKo ӥN:ۦ?PyX;};g qS58L@,ǃ` 5䃧Ko ӾNHӯo G<xO[Vl_$xu|Q/Ɵz\7kkjX_LXE44%_o~o x[HH^ю6-ӡ`j _aAiPyNjxF#$|qN3H.p? 6Gr3O Wt8cA䏔#dg$6`ixoO}_F<gigt<;l=6- 5q2؞a6?{^x { uc;IҴJt- MtMI4}&O+N=;KӴ8 ;hbhcH$D]-flhwgӌ9y95 pݎc``'6o /ObOs~v=cFB:{cLci=Gn=:[y`W_It{AmiP)>7> ^ut//W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f'v?O1Y=pqӨu)TtK/>bx__ǿ 9 OR+?2_<1w㟋_~xD灴VG| )Ej~MžWwZU˯MSle ~%χ6_)ƟƏ5_?0xkOxo~,|>xkúG<[еxH(_Ь/KEkMMo\6 Nܫf.s(A//CD_C_2>3 Uq$3huoM= Tz!o h>1x+TcXgO«h3:?ٟ/"n5npGq-t/ # h&|>4O?nA_[O[%SI-#S? |^&GG⟃M VŸ KŲC᫏xW5?,O^ mdxz5#xPҵjx]zm*t4OG'Ş1]_o|{[@y>@N h^^X87~to|h i$l|r) x-8)H!}e w?پyICVQ{I!@Or%p2{P52x_)caWLj>io~/.e}Vx?G?&%.nSzuȫ%kwGʿŏ xOzO>22?jsxz_g<=8a?i2j/ 4=5_?W}TtwR-ҳc?gKx^G?h|@=᥿Z |U࿈> |#MG>Ĵ)~j;e[Z>!xJÍ7>5/XrN5NmtkWV1y G#Su/xoW>,~xi_ I__PS●t6Zۏ7ބuIV îA&V $cO[ -?+Rx>R c<ͤMi+ |[N -xoPOz_aHo}M?^›!K=~T`P??G~8]~I|?{ 9=,?!'"մKӼE|)ލ@iZֱ~ n~,'V3mց ޣ}߃k tQ:vnm5IkQ%>&l$i$JA< ~g_ _Wz>Gq|2ѯ~<&kF^7a fZ^Wľ%Imm?_/> UKww|m9A>_}7W:'Z<" o4.9˹2؄nRZωZψFÿ7д sH~ Y4ݫ^k76sZz<[8|M٠ihKM,_ mh+=y]1S7|5M{l6εj^ WRWZtu>#6棩ꗺ<#^_F}UO~!5uϷxG"c/nk Ώ>c{a&ieߒeR #_C__'>?/t^g?NxGºo ō]~ 7^~om%n?ăJkOxW4_G5x3-|595FKÚ:%nuiuʙĆVDSGOx_&__>q\4?~$jt½"o5m{_=aO?'Y_]x][S双?)wּOh{%[v8_G ?CPt|h3ďXG:@Ka/$X4OJKN tP^.dOBG> Y#ďX?ЃH_~O%Eg삚VeZݒφ19/ϊZox4G59sP;xŚ΄<;auw-+;MՔiqܳJz #ek=wGW(o:Ǚ? 9+QF/wMsɴ+%>W#U±lCGL?𑮃oCǂ7X Pϴ/N^tZ [_s炵?c?!ZvnB +Y c[:mOs7ǮU-MW{E#?mHW]c'~$~;O^JxgŚ惠\Ex Ú^Eku %-@up???&'`v>&>8~}Lj:t'Ɵ<)w XY~m> G6~Ϩ\C_jMcZԵKȝlG~ϟ'WWxS7's~x[o}xOþ~ x[U'ӦVu׈ad,<|=OuT{Q6WAw/k:<?h~ N3|kWq ~~_u}'M>5|A4 xjRǏ:|JЭ5m4XOY}|B^X 1kyXvK$_|#w○‡TU g0W!xw^y3mnj<7a??1|Aگu *z36ֱ_m|9}"aI$Xc3xdiXԎ)$<RX_6o]_JHlqGl;>Ap  ( ( ( ( ( ( ( ( ( ( ( ( Q}h&_ ?C5+CV!@P@P@P@P@AVIO?(hCt?1_G0OaP@?|gN.>k8?8"M#mz(!r6p UDÌI#ns}@t{z'i% <@E+>A}c;s~f=p=9/'i9sGn3߃1rI_ڟ?:cޙ9zǢ ӧqt<~oo FA!=c~d2;q{( Г;to+᎝: F@_#s_ t+x.-~ &_QuiO>;|w-fj70CtI}__:Fia6=8WFg»]mcM[>?4h ?MҾ H'A7bjt 囫u&vM[_>?4 x?M҅,4M"jo-B>Ӷ:??*N:qmt:t?@@;t o+osxn1ۿJ{yaӧq{ft)mQ~wzuTm7tI<_x_ڭWďxڮj>EOKAtmtcQ^ i!;[k~}#@6?MҾo biö1<;feںt,P#!k(=`m*8# PBO`d1ߩi=1pC4 :Fpqlw">}@NҾo HhH^?<;beUӡo/P+ c?7`@=@uqc8309`@=3sTAm.U" OV/-oLu;==\i7z &d:BS[/:q8?Sѷ]s4mpcFW?yG&Awc hdK Tk 3kN%#?d  'MgCaĴ~cUڼ `0#p& t֞AwcE7;A6xRGӴ[Ÿ?xjEh?ƭ =qogwxL bҀ( ( ( ( ( ( ( ( ( ( ( ( (uZɗCz+???զHP@P@P@P@P@q6U"k)$']8W}&(| ?#a@P@@3|Z'M8:*:_1~_/ ߄x')z}㯈ō/¿ j <5#Ş cCZ及l]/3יŞ.S |WR௅ZωzψBÿ׾SAеM1bݬ^VS:<[8|Mai[_#|~/x :x‡Ğ#|EHZxM[[𥗆m[:LațZ+c~c O[>ŧ)H{mF>.3L4lv-h,{"BJd|_\Ŧ~uSNM/57K]]RܧOoVJgߋ$ |ed~#~/ |*t x{qxw ~nd_u ~i{Zj6⯆4o [)gǶ~"F~>-<{K>.s~-O|AGᶛxU ntG[Ԭ(EmǗ&ĿK@>.ÄҾ k_/l]7ŷ!oý-“i=ҬÇ]]MIEkE=w/Ÿ237ZWß"}#ٓ/%#:?>yI6V_3[.ޡ73 Ɛkqj%υ7Bz{ѷZp?~>xsQOzXC,OE{ëhϧxS},)u|ҵc_AݨYO­SgۻgAFÿ׾twG~ ?4۫k76ZZ']~" |^e^xM B43ནnM~ͺcqg=շsȯg|,Ko!47_|U:@xÿc׾跿Cu#L1bt_߳n{\wk\,PBm $C|O7|sᮃ}ƿ ?F:njtNy Ec"i]wwsr!e Zݏ<vg? Oo.Z~Eq1o~=Eh摦 XioٻV׬nl纵yDpAvѤYHWz7cookŶmlkkkVԼA?;o|Gm5GS/tyF CknDǏ_^}{BOUMӏ˫$&˃)6Gȿ~1xľ!(ƿO}'/π~_ូυ+wtx?oO:"J4e_Ο7Z𯋾2iֿ 5/k /? g ZkrkΗ758u Kk3 'Y.6OfM2#PWĿO@|#Ӽ!P=i_~"HվI k߅zx|=w˫xP>" E4zZGW(w i:f$&񞕿 Hu4]Ax7 e;=~=G+ OV^Sig3ϭx\|Kȶo&p?W @؇Wяg_t 7qì^Hhm#Y66/UD#DڳScu  x^l_3jl iE$}|  @4JxSct  ho.l4_ퟲZߠZݭQ?u7?-? 5LuOiA?}ExU{,1X6o+ VW*ȃ$}?GƜG,#AdMea/kҚ}G¿υ || Oz;ƚM3W>0Ϋ5/l}6mK/)$ݽtG(-s\I/% 7J5?iK#]C,?C{EhxSzGw8 oO;>%@~6{?~|_g3w<]/|!=?^>x_@[օ$ٚׯ, 6lSI[G#_/> ULyw|l:/;~+{᎑{4 \߃/?M>aue=ݯs%H-=w~ 7Cx 'R_xQK/x_:oxs–~R)t>LK-|>Ua_^hO⮥گ?)^'Uxkrc^xF 7-|: 6Kr[ZAv[ρ$t=sJŦt'N' ’vx|uiKH|dvCw !m ( ( ( ( ( ( ( ( ( ( ( ( ( (GQ 7/Xҽȯ ZdP@P@P@P@Po[B/& I>ء Uӌ|ڟrb=>P@Pt7Ş?h;?gM䊎6[Lq뀠ʒHێ¯5Vm{00A$#U:{qd'8WT/OϞ󃃜r09w'u=c:IpwlpwgjGA`@ uS|12 ICvd@1ztlttҁdt(P1@O@#GA <`cN6.{~Ӡom 8? %BooQxƞ4#mO_MW~"o-֣s7IڅnjX{cӌ`y$`nyPNil+04E7}#@61+4<=l!|=_v.maX-@YRlk4E5}#@>1ǃ([[D.?<=ly>Ư&,jt-P+1;c:qʖAOoӧJtL~6v w8k:qn=pY7kOoҖAy| |ק_FA{|1~x'L:/%|H=Iyo{?/f_?Qn47ML5.ి3xhh>?4mK+&9<;j?cöm[qoiB>2qƒۦҠ 2=@$FHf]3`$s:H *$`:dd`ikO}aiNҼ M}"!sxvɅ٬4Xn`Y7֞AwOl~o 'I/M=6Ft]>I}&N+NO4>8m"m#8UT/:dsSo G $sb1WN3_jɊP@P@?dx >k':u=eG_ x?>_^ JEq_x'.scKh6cx Ho<[!wZ~ bB ur-WxŸ4X?G|ex3ŧ|/ Ks_r|9 M@o"셧6o5+~̗ߌV߳~>Ӿ+r[~՟\PlZ߀?}TZnXp_Z麟 W%?>k# i cśO xwHg+~cx^IվԚiph؍i۾i!Iەl?en|R>%(hx|P+Ga`?+j>l&j7O >;1<;oX,MZ Wd}Who T?+GKBQ?]h_Kehe)'KVQ{"B :~ӓ(<ځ M{< dQO KsG!|9s+?=O 9q4-tuH| r{=EY+[>U~,xS|+|)9Uӟ-=<) uoRCEW6Ή-p?5x%8OP|5/2< W>i)x ×Zvjv^h|sXm%m-9kx~agp|l_JooO*u|jŸ{Itx# MJ_v wQ6I%}h Ti_–ڗfOXdm&BkHX>OSߎ_u7_Aoz~3ϴC,?C}ūi"w>I ]F {k <[Nx ]F_?a /sM֞mJֵ|vg? On^c_|zAuM/dt_nyXOuk]h-!7e$SI'VZ _/> ULww|m;/9z;{ᖍ{4- \4ςO 7Eo6׺ŜVߵΏ">%t-KkiV|Bߩ|YW­W_뻽oQ^cx]A43ŋnM~~ͺasc=ݯsHC_M ,N|=?)?(K麽?:׼+waoyu~]ȆO%kv>ڗş*>&|M>6|D_j7=ƿIkFZ5ah fZ^oOCFZidw"coA^7zᏊƐxMѼck au[R6꺔:+ T^6N?.. RVi"?>> z>x!s>|uoF>,hWI?W|i+tH$WS]~:k¾.ɧZ7Կ>ѯ|636k ɮj7:\׼1/t;K T$6򰞚%d>2x>*4hB__m?NÍB|#VE&%~^߇|w.mB×G4 P6Ik#ꯅ_.}o~y>xkPWzV@#XuӼE *JY4xtW g࿈k>Տ|[k:,ji=Zڞ,|ey M_>}sE#/"۵ \.dOB?b_"ãF?~$?$]ǃ y#}dRZtVˠ| \.dOB?H6CWcu ~cH_I~;5'ۏpZޝkKOSC֝Ax2sN-bo'?F?#_Gw _}|.5 6-ַ6v]4>Q<]i:/Ȗ>xoPGƚZna'!bմKӼI |)xcӞJv>1x_|']__^8+c?%&;ODMCP#SUdi+i>/xS%{׈w&ľUZ)xK>ci/~hM<>t/@i%8_$<#{⟋xM+oxW;/D6𽗃|eڟ}cVCoVV],g:©/i W:/Ə? ]x>:7gWZ~!k_>*÷Zn6^h|s趐Cxj3k++ ~(x~?ƵA/u? O?hiJZk?J:6>hZ[YwB(>b}svw"KލOÖx]^_ͼ+kkY )ƿ>/xC?mC:>x᷋$w^Nj^4Z|3ksFso!+t_?_u}{\J|VԵO.qxčV؛[ 4}s7-پog:}"$c폃 E-um{gkej:bR:v\ CIb~<>t:ڥ}uwc)#o+_#>su[| ( ( ( ( ( ( ( ( ( ( (uZɗCz+???զHP@P@P@P@P@q6U"k)$']8W}&(| ?#a@P@@3|YCq)HeA \$=*Um6$HہB8P5]<ެA9zn#ZIuB889#0GrwQky ϳA8nax䟠Y@lG Nwvv}q݇^9#0휒W-Oӟ-:{cy(? bGn:~>:xnmaU]`1ӧAShkp?~KNyůߊ.Wk`ixoGmcW/ai`xzC{H:\°Z SNun.אixkG}cP𷅆\`xzb|?_ÖMX4[gڠV7c:vAOtN?-ߧNw8=GC=X`(Iw]ݽ:gft)m?(?@;`䎽:*6 y:g$/ռ@/mV|U+G^YnC6F(_~=ziZ[ˡ9e/:~(^4񧎯_]jZ;}-ơs 71-zo5?A][KTЦԬ" ?Rլ4i/m/I=9\\7zFՒC^B~пp##9.m=0Aѷt:Q o )=Ni o cc00{3`o!8OdN'$v1z:c9G?O?*[y c:6 cҴx[Tt=MiZ>>7Eiifi 7x?7~6-CgIDX_Ug:bҀ1~H݀yqѰlPVҡ,9u->oPԵm?F=RJѮ4M_RәΝcuhp\FIؤyp?BP@O3`qr@sрr=>>qF$|o8d0N9' P89@+daI6ӌqA#{rg@s6 (syxǦ8#~Gh׏@[C#]j^PiZp?)\kc/$Pgm7#KơZ?@o1cٳ<o~@P@P@P@P@P@P@2oO^z'_g ( ( ( ( (? _Mw@?|9CD??z}s ( (Qo\~w`|Pσ< b> /{>OSWa& zuGBffK~+oÿ 9-jI.|Qx6KoZ[>s\xr-7O,t/tOeXb Z׆1wͧSg;xZh} \G/$uM 44_FMoʶijw>) „}4Ez(#0[u_O(J_c6VЄ/ GƃF>7t*`?i 7+1#>0!ocj *)x8 2˂cCO_'> ߅=y W; xW//5ke_hhVCB?0_/??.uO9]CIF<_;O -𕾟?uFNeyto(m9|k?^Rj+.|{2o x7TzxO4ž.5K[3a&i uX $X |b(OڿƳxG  c?'.KQ/S%3N uH4[oQ[ѼG?G |L_|pd?wkݗo~ 7? xK,;u:ΥeV5i:,a6KEc>c O[ -?+RBQ'bS(k$yc_ o"mh:~ӓ(<ڐ|x⽍ _2}ᧆ%#[M⧏8_ksxo얺 >O =֞"*Ͽ<)I>>Fiτ^U~?qɫx04m|?_ imSQuHSJϥlj:7eWZ Mm7麎YxQeCMgD8x]Ě9ѬtKkxu_ Xm6O5Լy ^w8>6|}]㯇 |7'~|:־5CO^=|jon{폎_??oeg*oχ?KEKG',_KGY[pt}6o!5l ')o :/g ]Bo?~g!!K4x;ß o.o=ROW(w i:/f$w5 g> i _>/xsMw|-4ZX?C__'>?/t^g?NxGºo ō]~ 7^~oM%n?Ću5sa;Z𯋾2iֿ 5/q|77~5'&}uJx3Ce^_xۥ&ܩuxW+Y?#]ʭǮZS%5a]$9G?k &&]> C5P ɪxP1#RQ'ڏ1{G+>'WiN=EmV(u{7k/Ac|O7|sᶁ}Ɵ >ѿ5auOxPhXc&im˹JȔ"W>V!7Yγi?Uֹ`kQj3ҋxu;tx; o%.Зk O6{~j o)ğ Y,~?}Sµ~ϾX.5:4b[Y Itt}gHOOx)Ծ%o > ?~"ckٿB#c;t kkW[CI.~~_}'M>5|A4 xkOIwk~Ο f5o%l<_X 1kyXW{9Ѭt[[x5o X~ܩuy ? ?Q1x#SU%k1sG)^'W iN M@꺴n^$C#7YƳKgO|sX/>41W:<&||m }B*R}Oݦm`i+YZE>X^O _oگEo?R/|mu]Nk]xonm5/=OuSԴ{Q6ud}|p>{ԥ񞗼\,?C}Ehi#w^]ѷȉYh`X"E[KO߄?uGH׼;/imE#4{o;ִ^Ykeò|9|jO^ooxO2xQU%ڗ1wI<'^'ߴ>%ih"iu?|V<o<caq|NԮZtwi ⧇ ZoY^ֵX⯄u|S~xB^:¯k>񇊼-[ρn|a 3?{hKI&^*ktx_Zw/ex[o_쵹kj߳5Z/yP4 [N4̄{ |3 >1 g]}B~:Ÿk>$э\.u(4|O|a㏌~"xP [þ|❍;4Oᙼxm| $tݵ5V!|_<#{⟋^x|)>Z%ω>0݇,/~> |Uw#K[.T[os+{ڟ|7+ߎIߏ u4$7¿ZƘ|'Go4_MuLjB:O`:徹yX,fKx^Gh|@>Q<5/3 _NO|?×:nV^huX薰CxbK$0~b'{ xL_ɣZom5j7It |b(OڿƳxG  c?'.KQ/S%3N uH4[M(R#<#>%8i/R׼5/3 _!NWW6~5 gREE _0 4V ?G8\HՄ> w N=/TPWz|w߅7CH6{$k:h?57g^ZeƳq$G<YSm߄..+>[Y$Mid[O? |^~g!5umC¿ IŒ'<_GԼe>g?߄mukqFޡk7:5F |f'& Bǟ}OuG𞧮O4/5;{/|a. uX- CI.-|HoWS]~6 &iR|: Mw<_g’Zdԡ׼3?4K[ T- @_7퍿>|"<2KVx]ipa?rQ_5!2M:`!c}Fo|q@Kj6|xw^~Ϟ?N hn~>w\Z/M?S՟z~YOts_~=#:>%>"X6Y[C _no|6C u-WevЭĖa?fQ_2yݗ \![i;Bi%>&7/S>,<1ȵC`|S M[? Qh:v?[_z#WkۏO=*JEk~Y~x? `_??nu'+7HF<[+O |-𕾟?ޥuXkV$j ED^j|#iJ~#XiR|&#>+j>t~&G>;7+z?_5k-T6eeM|s*|sXhux/MjMk/t|o%sFMK>ݞ[//~"F>-|{KSޣov|ex7Cᾛ,|;u:ޣeF:-i:#?@v⇁SkWp4S@0AƟ t|hֻ;Jk뚖5յg{$-C/x? {|tBO+w#𥇎 |1[GejŴm9_E;ĭ77u'm6; *⾣O}->[#K֍OÖxa^_t{l+t<^O?\IU?.QxčVͭH> յ1&Iz敋MHOrO /a%j;ݏ|dvCw !m ( ( ( ( ( (Gր2eޟ`cJ:O_"5iP@P@P@P@P@oa0> $sb1WN3_jɊP@P@xdobc6k*:tl}2;G*I#n;`G @[MA8>` 6g8Ў WO ǷNxzq ܝZBq:gۦ0x#^9'k#vݱݟ\g4~~^8$c8J(t( TmpOO#o qJ6:{c(~d2;q{( Г;to+᎝: F@_#s_ t+x.-~ &_QuiO>;|w-fj70CtI}__:Fia6=8WFg»]mcM[>?4h ?MҾ H'A7bjt 囫u&vM[>?4 x?MҾo HHl {Aiжϵ@oѷ~9OSo  <0ztp18#へ g#~ttyt]qo tʀ]l=y$`nqԌsF ]v04E7#@6?MҾo H8<;demƩB_ڠV7{c?/)Ƕ0~1y=Ӷ){y۷{` _QxVAw`; tF } .] ;o `H (+&8<=j|=lvžf5m:囫u`/|Ohڦ5L:>%|Hiz]ť棦P~o j7zv>mIgpU{|ޖǦ᎟?*?A~)pto+AQ o {`O#>pC:q3`crPmӦv9ɂӱ'&Vv#s? 4+ .-z &^ּOu|iO>;Ե?4/7;Jt<=ly><;de[jiз SbOlzq9^`nqNhk-i&>~}.p4ZFXHV ?]:"c76 o \;m}~|;1Mgv.?6ӌtg#tzsyXw}\ ( ~wc C]j^PiZp?)\gs/$cix60 (G-q@~݀|&O\tz {y/d/|𿂼={LO ]Ɨ_2<9m2DxJCZqoք{ro?f[?Yx0> x3͇Jh+.~3MWď0 B>`g5O/^ľ#s=>[=O^|E Om.jV>"/g_+|/|W>?$F..jo<8Ωsmȴ?Cྵu?+Kc~|..ŀ,gg(6KȧG<0@. ->|b|EQOƿO}'/{> |/_C<%a3cG𷀿M_V4ii?g/_gv9MOXj>7O?|=sQ~i?<_xov7_> M#I|Ci[U ^i$ $O3)>*? jν>1^81/Vğ-GLDχ?;% үn!47fKx]?h|@>P|7/3o~i*OYvMuJ­ku Xl9$Ǯ|=/Ÿ33/ZWÿ!NźPH/ؿw&DGR$:~ӓ(<ځ M{< dQO KsG!|9s+?=O 9q4-tuH| r{=EY+[>U~,xS|+|)9Uӟ-=<) uoRCEW6Ή-p?5x%8OP|5/2< W>i)x ×Zvjv^h|sXm%m-9kx~agp|l_JooO*u|jŸ{Itx# MJ_v wQ6I%}h Ti_–ڗfOXdm&BkHX>OSߎ_u7_Aoz~3ϴC,?C}ūi"w>I ]F {kŸQ?u3_IjiK}an-[D}?^›!Oz6 Zh~1xľ!(ƿO}'/π~_ូυ+wtx?oO:"J4e Jkv_|dӭ~xj_"^oo/ok xNMrKÚ:&fukw˕伾!KXM%Ry/_ ^Kc|;O W|WO?:x_=+D:>ŧ|ek-PҮ|GYz[]#7YƵKjUֻb0G:N%/3T]qs|cFZOS׵ c=W|Oikg4/6zjڬPn^iEmo'm\> }fkAF^ѼSMӏۛ#or")lE|CngK~:~rxO!k2k?|හ<-}T[j?Lju)5 _kvZyM,<Usu~gkw?W7'K_5Qc'->~)kٺݥm/ %cQLUW\W,|/Ow7?/A:F@(T_ $^i-ֱQߎ?uo7?Kh>kQj3ҋxu;tx; o%.Зk O6{~j o)ğ Y,~?}Sµ~ϾX.5:4b[Y Itt}gHOOx)Ծ%o > ?~"ckٿB#c;t kkW[CI.~~_}'M>5|A4 xkOIwk~Ο f5o%l<_X 1kyXW{9Ѭt[[x5o X~ܩuy ? ?Q1x#SU%k1sG)^'W iN M@꺴n^$C#;YƳKgO|sX/>4QW:<&?l<% E?𮫠ONC7 7O6m:OjZ~yXvmO |[Nx [./(ar-cDN7HNDJDcMj-"_k'~&!:F߅ֿ!|#Oh/%D恣`V~ ֵd7\\ܛ.*|9|jO^ooxO2xQU%ڗ1wI<'^'ߴ>%ih"iu?|V<o<caq|NԮZtwi ⧇ ZoY^ֵX⯄u|S~xB^:¯k>񇊼-[ρn|a 3?{hKI&^*ktx_Zw/ex[o_쵹kj߳5Z/yP4 [N4̄{ |3 >1 g]}B~:Ÿk>$э\.u(4|O|a㏌~"xP [þ|❍;4Oᙼxm| $tݵ5V!|_$<#○^x)W|?+D:>ŗ/xR%{׈6|~]㯇 |6'|:>4/>=|Xon~=M ._D?x?|=uo ?^.u+ E4oA]Ow\u &kR| 7o |O<oxgrkΗ758u MHtZKtؗlP?>G~7IĞ>2A jiKa+r=_Em;~‹!m ZSs{~3|}SV>?O?|=wQ~i?߄_xk7g MJ|Ci[V ^ѽK_S|UG~5{ |b?pc_?t Z ™-.fwK?A_Z-BiElo"F>-<}I.OzY|UψZw'Ιk:/h-jmi\Qſjg̰AZS$71E Ƒ_kqj-" w{Qȕ[f|O~2>o|,Xf =Y)xg}4OG Ii=KAH%Sd: 㦳~3h ~%~!|>OWn!:xV[w+}> 5JN֭I4k_KYGwӞG9|.MGs?|V| o|eaqM/4:|woWjZm*˥YU%5a\%9G? _Ԛ_#- TKmB'$:|-5=GUA_l^R+@wOvWGFק5-kWwwSk7H[Eߌ^"'~ xyoង |WG|!Kcⶏo}_g>'~ ;ui/r?hOwz9Zoo/; "OlwU}G-Z}Ʒ7Q>XG}⟇-Vk$y ğ~k/{D~5]|/1W?ś[c^xE 6J:wÛ>x(v $kl|/l|k>c[/LQ+Ӵ:'^ Kզ-w Iy[M10sӨ<BP@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f'{xl S]|QӦc6=pRIq{8UmaH$9>pjyOn=X sݓG @מppsF `gq>1@9?@]89 Wtb#'I@:om0 NM-CttAH䁐I v}Viv_@:{1I*N'LLtX㷱lmH8#UT<-R}oF.5t|IGT4[CMkn&mBI{8 4-&Kp]{|}-Gǧ#z(I4m&N/NMtHNa4(XB4?qO?@=c:6](iZNZn6zNiVvvzuv~qki 0oiQĈ6t9At8 PcFCA26Hx^H8z16`5KGlO:~dz AwCWFOPOp84 cE𧅼5xoٴ vFy>j_n@Ʊ ?OPڗf1s!;8Ar2:c {K߶nq8~8zr=tP  ( ( ( ( ( ( (Gր2eޟ`cJ:O_"5iP@P@P@P@P@oa0> $sb1WN3_jɊP@P@?dx >k':u=eG_ x?>_^ JEq_x'.scKh6cx Ho<[!wZ~ bB ur-WxŸ4X?G|ex3ŧ|/ Ks_r|9 M@o"셧6o5+~̗ߌV߳~>Ӿ+r[~՟\PlZ߀?}TZnXp_Z麟 W%?>k# i cśO xwHg+~cx^IվԚiph؍i۾i!Iەl?en|R>%(hx|P+Ga`?)gD< aNGB$Ht,-R 7oK58.c6>WM ?@w_q_4vR}^o;xsf?⷇%U.S}uyxS+w.u+oæhvz\^o!OADZQ\up;:gt~\ B //چC7|㟅|)ĺJ|@gh͡;У^CdV4z'#!i߳MJ_%{߅zGl>+xv[\Pt] xm;N{_ChvZUxhM#A/?8//Y? c[fN0jKǮxU-ދ*IAFr>e.ku~d_>k$ i xcśO wHw+~&cxaݼUj |w[)c[Dk=M:[Ev__O$c^QuXg~͟o?hj sg+L_9-+M>X7,IKྦྷ|2wk3x9_@:U¿͢~?'֡hq>$+%/~c O[¥#? $^q S)o5d qg&ΟJio~/.e}Vx?G?&%.nSzuȫ%kwGʿŏ xOzO>22?jsxz_g<=8a?i2j/ 4=5_?W}TtwR-ҳc?gKx^G?h|@=᥿Z |U࿈> |#MG>ĵ%~j9g[Z>!xJÍ7>/]iچxM֏a[[êj#o!~ ^ J㽇ů|8M+%7¿Zƨ|)KǺo=[NuLj|B:Ato uo+QzZ|]-L M)oH}d p?nOfm&@>-u'W|{ 7M=/D07Z}&/xsMޞm*V|).s'W<Đ[./'!!մKE|)oKw|Kk>2.]x)WMׁ[^&ӯ^@m)#G[l>'k^wM:᷆.5/ƿ ׄ/9CbohwV~O^KtZ[.2 >1 gÿ]}xJ~S>|- Kρ_ Ǟ!i67[? [ukSo+*cȿg/_gpx?95Oj>7?j_|?={Q>h߅|_gw_ M#ICgfe6Vl}O~"|65Ϸ@'7fl4Ni u1}8͹6w")YZ>&>9~{Lj:,>4~QxOz|.||omB/5-ST&FKx#^Q@ho(W~?u}+U>5<A rS:KmΑ0< -c>d/k#I$Kne}+x[fhax+m5}]7^›bK:@%-|S? |^韴~*x?ڇ5[ |3'K<_Gg~xk_`Ko/ /X5Vn|C.dd_9>)xY%G|d u/~_»?:߈1oеx!A]'BoKm_ISyfo5u"?]w߳ĭٴqŠx{;5Op?/LZVcgKx]?h@>IQ<5/2 _>)/YxzMuJtk [0-t*]e9|k/>? Bǿ&OV𞧯j>{# K_/Ii:o-5CYYz]g—tW:枾#8_ d~F<x௄<+mSڅU<-uMKUJi+imQ?ŏ >|Fi߄^[~߈t Wu׈fag~5&֏m{Y ğ/wD6?y)~!#U{kRl|*"m9ᅷ!ev>:ltkH!ռ1$YYX|Ywe@^ i5ܿ |#qk;|9t={4Z943X:_!<wëݛ@txg⶯*վ3׺ceX~#g_zFh$wڠjIҧ&[|]-c5|^A㟌SVl|@f5G~;~~|_ZZY_^+t}'NCfmJݬu~1xĞ!(ƿO}>_'3יŞ.|woF>+|ykFsh"l*[?% 㦳~џJEL|:|r>"I#N->JN ]kQn%~8OR<5/3< _MWO^ukQCE _61|Q˹|0YxBs.?;ÏfEcEпaWV2I _>[ix_|rLJ>(lj|Qпu[>8iKj=焴XllHoQY-iD E-fyUD#h!V :>4?g_᠎u,G_0W_$W*u"4@ƑG,hA|/gѢl {4[ ۧ~~:׉(x?^ƳxC4KO~/Axƿ t<h3KӴ˳kk5˻mb[de-cMFo#_% 4ݵ烵èkM0YCA5;ÏIj%пaW׾V%PG86?B}Yt0>x_|r>(ljUm?g_|Q־ x:mε]x;:ͦ㟈>ƏSV/<{6ixrW;?xPgOhy %Ƈ8A^7_~ᗊ5^]K/[5?xfƖw{-B\uMFG6KO^ON?]:3rybu-]k7ISK|t>/kCO[Ƿ7V[yV}48#@zNa( ( ( ( ( ( ( Q}h&_ ?C5+CV!@P@P@P@P@AVIO?(hCt?1_G0OaP@?|kNO{ŧ8>ƚ䆶~Kzq1~S,/rA^AF8R$Yb}:H<O0ۉ8CW{c'GH<|݁g`Ӏ8@ی8~^.0NrAYAz{ ۜpGH#+@NHGRm Aw=:$FGwlFvRm.}WO$pztƏwv4koӷ:NGCik j$:ƝދiB&-4b`oEn2K%£+ ֱ=x6p:u<W}çKo+A.mtP Vn1Qv?@cN=ѷ&~3 Þ _I]E/x7=~5_Y7Z$_jΆQwXnMmc펝-v 'I /Mд-6E4[ =+G;};Kt:+=?Mt8`8U^A:^h&agh>goizVEga}qcami0CoiQˆ mh?@zr9t@ AwLvNLQ~ttyt]qo tʀ]l=c֍}(i:Nizfi.zNi6vvzuvz~iqki 0CoiQĈ?BOl~ ^AQ]0so :v?/o ӧ.=lcOo žד ;]NҾo h/~mh4ŵ2?@7{cG :{clSo+OlcRw:{c8o Tmt((:{c4 o S~ :{ma]9xOz@/7H։!|eRm_:wpOy}swpO+-mV+kxMAlPw?]ء Uӌ|ڟrb=>P@Pt7Ş?h;!Өu28MyG/ïhwx,<@oE!!i߳Lo+/~zw υzOtOX|Is}C5޻O߇CԮ-|;v:Uk^" ?w'_|- k fď3)S$e~4^,tQIz^iq 4[so!;K5[^Ԭ|C2 i=-jK_x:KoZ[R:uiei!6 ](Mi_u gM4_~ںˡgZ7i|_/i|2xufC ~6:&M׃ݯ$r &iߵ~xs;M[o/~ѴEׅS࿈J{Ʒ]ܽƩgx}5/-5xj?3_cInB ;yXRyk/Nj~3Z<3~1qE֡k:G|C.dZ"K{GvgDߴ͆v|Iq~׼Cu"x?K'o_ڰ.>!5'Լ-Iz(6k]Z-zy:Fp}G#[||k_^_=w>0+-E~&?O^1=RpEJ}'F4 iIGv|YRߴ͆r|Iq~CI-jMk/t|mmF"xRR<垥]RQr$~F{|? cPyt==cҀ({::{c8P@P@Ol~J:{c:PyX( ( ( ( ( ( ( ( :{~:P18zPے}98?!GP@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f'{xNH#6-uEGKMzgx9#2Il_Ot`gdem.tU0p㎼t9k0668A4ZC|Wvǩ@'9lG$8PאN q?Lր:{cydSѷ]@]P tbOl~o+pA@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@L^Ik?W2B ( ( ( ( (!]O$P?*bO91Ca\ ( (?:*Cmd nGKQ'xZ s:f"_tsCl5//؋SӥP]\=="G!~1olI*|?; XxNV?#Wqzvm*N2-•ccɨCzGAf>/^jv>!/mgo|+Ҽ/U>ķ^*<5/}QÑi~ke[iԽ2x ~)O,hOAO_ PLmds'mk# Ǒ zGk؜z9=AmO<`zg ɣoj:{clg (~{z,?G\9c9+gKۧ Tx$s@Y}3Ӡ9- :~;88=LPw sA<M6񑓂GqH F o圂 ぞ@],(ßߌdo 3q$m4m [;tdH;8^䜁@tL$3Fzq;g 6=cda0:`28hKO1OQpAޜ' \9~@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>i?2A(xu1Ķ4 ]nmFᦻ h.o4_wɴm|G}Z#ooVQ?<c$9;yXRyk/Nj~3Z<3~1qE֡k:G|C.dZ"K{GvgDߴ͆v|Iq~׼Cu"x?K'o_ڰ.>!5'Լ-Iz(6k]Z-z~Nr9Qs<m%ne_֟7Z2iֿ 5/k -; g ZorkΗ75Hu k3y}GkyX$>Y~ G'mKT jci N|G KkMSBƍKg=߈M)hh|oɦ_O~%o"ρUi' x}kqFΣk5Wn|E.bm`J/KX>9~##Y]qCdD~n6%(>U)ƿ>/x??<kjeO,? xzb ǎgA=_siZLJ!6su'blloYMP޿ 53mblu=7&YQw77߶`5oB\|j~#-gCxm6Դ߆-Xhf|m76^Rl'u-cnм;oM2!:>_ I?xt yxþG5杧iܤ6W^@Wv</>.%_k>7*>ρ_ xG^gۏn^ /M [UNxG-e>U|oo!~6[P?6dxf)}GƟ c~՗ѭεZW $a> |hM)? jMW? ,Qv/bKOj.tb[InE6mo!IYh~9@)cK_ 4|;Yk.7őD4 |"q+(Wˬ?Uහ<+T[j?<-5=GUߠ4};|}ߴ|~."ih.n.}Yx]a'|EJZQȖDP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f'{xY8tNQk/>d `8g=EHmnq`N#,i:\íjnEgmh:ޓ]+sI3Go&$jw1f= Obd}t;';8?G9j-o y$&{.22@A=072@9$pt1# 9.[Lq㐠 t'nh WtAuN3@k`=Ǧ;*6 I:{cj6]Ol~o ? vT@]lk@;Q_ 6?Z~,OGtNOʖV:qN1?ytL~mk?tmp˥8ǧo!;c#捼CMtK=Mm&5mZ[]6]S^/5sR XPuCP. ...&ggv?@ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (GQ 7/Xҽȯ ZdP@P@P@P@Po[B/& I>ء Uӌ|ڟrb=>P@Pt7š'~1zMp2;Ȩ/>Ѵ ߆2?L[i_xl/|jg[h: >? /L! o,bv___2)5 |Kz4OW+Gh|gM=:7$"Ykc$vYo7|Ӿ1|Λ?W mx;si*֭//kcmiW/cM{5OSǮY-ihXoDYlˣ(OY??^#.` '1jAtd/#?d;_|?Gi-E/6:< ν?[P z[U.=p$pOx_|rLJ~(lj|SIG4O<9sQ;ށ54h,mt8oO%--lkh3Ⱦ®_'!DڰxS:H tK!gZ>.Udـ"$/?/!?#CPxxi~#FrH' @ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( Q}h&_ ?C5+CV!@P@P@P@P@AVIO?(hCt?1_G0OaP@?|gNa>k4_$TtMzcHWk=Nh.=bAv=Oo0:{cGH[yLvNS~ ?A .P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(>/W~L ( ( ( ( m EtSI'ß4O!qSNLP#~G0€ ( ΀>f''U6Mu)u>iP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@(`^JtIZ\CҮVX$nlga[b->hR{5+CHP@P@P@P@P@q6U"k)$']8W}&(| ?#a@P@L{g Lh~!/:>1ҬFu|E|BMݬ^_q}* O{_sEv| 1\m+#Pt^>9g8G~suɊ^h^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?suuOoмt_\Q`?ˊ9LB:qG7_ɀh^T/.(?0 E&y@S?/?su^ UݔV^ fgЧs{XO{=#oinn;jG6jޟdX^ 3hjz.ťVkSfm[/ZTկd}c_IKKZ?VKNUP@P@P@P@P@oa0> $sb1WN3_jɊP@P@(c1򧷐m=c] .(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .Qw]}(Ep .t:QvO8p ( ( ( ( (? _Mw@?|9CD??z}s ( (9WZ ݤvI%v̷ #GR@XЯ9#{_qQWvy,Motlnz 1XiKu.X&pIۉy ⟶t嵼} [=a}>2}>{iI[9R,Mlˮ08~1dK"[],} </9X@d8[NdqGD&>^135?m5W܃/C8 }s^K_`Ku'Q}ȃ.>_=/9|<ٽ~z-*$?xO8xs UҾ c|WW95-Z-+Z}g@ {i{Jnoݿiق+!WEažڥዽ^-Tii[_:˭>7fMGPM6Nk46bogӿo!>F֛O⟋5oK Iյ?>%4mDŽ i$^5 /K HڄTMI7{lkn[{G[7쵽'JVO| _PF Y53Agkaڨ v:LE*V]eyi#~_|EOox?Ú;_ [z7Ε^0Ơn%Xu>esqqmi6Qtڕ^۠K~xSk^%4[ꚤڏR|4%ͩ'I/5 B}:Ku_2oC.<gsZϷdž7Qc t kWS[| nmmu7$QG_ލNN*N$[iݐ/W@,hwö>S?6}c{Ɩ8iR]+Zhw'.R(U:ok_^ofꍣM}-0%+]~;o=1/?‡̙l}VfP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Po[B/& I>ء Uӌ|ڟrb=>P@Px^=X@?>:+Z?Ϋi7HWaZZFdx-͠XѢUZ?3:rQWʆs-ֺ̔j<7qI]~#O?u1>2`<'Ňg(jf( ʖIm{ohex~H񟇭|/xA/}ƚƅͩ\޼kֳkH/Iu Ź濼{Ӭ՚iiJmn?d;^@~4?c)'ޝxoIt ?z1K+_xh.b{TZkݷq7ݫy>l_m[7Nb׉4GE,9aZwIhC4Yl%M6^4g {:r+/xQW>%KǦ>ψ~jm=͕}Zѧ=qؾ=ڕظjIem;z"dCУ=?:%txOXӵ[p[Šh[YGf+^t鳳|3<Ӽ7PiΏ=7Og7XokM>98a[ibM/iZl5xL|V[Y#Wv/<j,Vc J('tgWѼ71)s|5EV:,.{JkZ:$o6Ï|;m{Vw,Ήa>f>1bTkY#exw?h=º\h_aBm` аdVHFR)\ &Z%}L( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (? _Mw@?|9CD??z}s ( (? P@ݽ= Ni`'w?J[y}? bAw`펝qFAw`~]?*6 NOhuOӷ򥷐])k.O㏠y}ARP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@oa0> $sb1WN3_jɊP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@q6U"kPKehD#PK.A OEBPS/img/spgateway_browse_3.jpgJFIFLEAD Technologies Inc. V1.01  }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyzw!1AQaq"2B #3Rbr $4%&'()*56789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz ?bm's~]CC\]_ez嶙٨%u R,"%A+VX+WƊ]U*QR4y94蛷]>;r{8_^ߋQ|<Z+XuwE}dUgPʏpQx/gqhɮmOSQ!|5ӵ8rQ"k 9E*T8R2x1_f /vϡ_8o'qh<}hf9WrG$2 d&LD>`wFWŘsN$XUߏ"I $SX. C{"G |-̅eڏ?H?pȇ3z !GřvO/8! 3Ď?#n(QfK%nЗ$8U~?7w12<-ȇ3_ܡGǘsM*m?D?B2G$8_qCgGݿPb̗JJݡ/H?po_cod)y?[1eaf#Q vB0ΛXU~?7x#[,tC¯CgGݿPo_ͧȇ3z !Ob̻Q'' ?7{12Yj? {"G |-̅eڏ?H?pȀ;Cāq q+mI۲_//y/~?q|0tukwzek}A jf$_iZ$O[Fc#Idԕ WnyaSܶYGP!l⾙FvcJޥ0WNjzQ.jkZ_ }İW I"F ӅVˏp88tЛAy/Is#+ '늮[/g>c.#rekk Kuv.ycӵt]Wt]DH!n;Qʻa<=SK½-51^=xǿZ\MuO8֎T_42zQmլ'G/O֎U?tޡoNrKwLG_Ҏ_+X]v#G*K}XqO. Äxirõ5a<;Z9Ra|8ʞ:1G-VZVqĽ?Z9W`kz=;QʗA/eQ1eJ9|auO4v~-aCہ>Tl+[U׌{ʻ]hK~뽅H*xxǥZjkZy]h]KqG*]qDq](oQc.#rekk Kuv.ycӵt]Wt]DH!n;Qʻa<=SK½-51^=xǿZ\MuO8֎T_42zQmլ'G/O֎U?tޡoNrKwLG_Ҏ_+X]v#G*K}XqO. Äxirõ5a<;Z9Ra|8ʞ:1G-VZVqĽ?Z9W`kz=;QʗA/eQ1eJ9|auO4v~-aCہ>Tl+[U׌{ʻ]hK~뽅H*xxǥZjkZy]h]KqG*]qDq](oQc.#rekk Kuv.ycӵt]Wt]DH!n;Qʻa<=SK½-51^=xǿZ\MuO8֎T_42zQmլ'G/O֎U?tޡoNrKwLG_Ҏ_+X]v#G*K}XqO. Äxirõ5a<;Z9Ra|8ʞ:1G-VZVqĽ?Z9W`kz=;QʗA/eQ1eJ9|auO4v~-aCہ>Tl+[U׌{ʻ]hK~뽅H*xxǥZjkZy]h]Kq3{ >T=:-W+naewssvH1x6c'$ xTcҠ2ܰnߏ"O?\cqsJ=5u 4s+Ex&ZVs}kRj)'s>| <;BQUԤ{>}R%kf!Y!JiY[oJۯ6/|ͦ^/ƐMk _4sGE1Ku 64P uKlG3"[i-KZ'/8t |A x:"zw kl;/fֺ.0 x'tu}?7>f?c_< Z|U-OoikOa^^y/)Oᧈ/O_>9j>-Gß@Nmh>"iHC#km4J̶H{-ߑ~+> xƞ'τ4 c!Ծy{xNյkW~Fiq/kI7vUeKMy.?im\xk>'G S?~}>4mo:ls[_aqqm{hh_jN) &Z_^v|L|??3VZ獬u6KwUxoWu-5l]W-n[$yc>/]Ğ%5{x?z<9?cix{N_O[Kzo!t>!jω;n5jM-_;K7%ѼEϤxþ3_P#3xX~[i)ѼQ|Jxc~tW'^c\T/Cj:%kter9ľ%}3W)u=#Űiþ0χkjZOhχ>9ej:țF|G-̖^ԭ-[뷙Sƚ_n<_lˬO6iZVyFhK'Zԯ31iO_KHm&u:c蕼] [|u1O}|܂@]!p̠]v4|܆z֬z8մx%u@k4jNrlhª8 Ua[E?vR Qb|R\Jp͉7Lė_iAJ_/jou?y|֝jUԢRIywVG{<'X[I{' d(PH.JXK\6R>^^VM{iZ~j)i\Ȑ]hrkpVh\^" p,9#HuS,|f> Ԕ$Sq'esZQW"jn-{BMZ|3Ow^}iڂC⾗td[k.FfЇA~>8qt)VJjKTiKeYmM5 >R<3<={Glac{{6JuUx)(ؗ{zvKE|;~uHj +Z]x;Q$Ӷ䏧~KIi(E>]mo$az}8)I/+*t ;T뷠kXI-wapGOU7QM,.Ha};’ A%kY Oj=v\!Ӟ){I-7޸tK䃒=3O?i%`[[nr1c*]v"kv<~qC%r7;i;qRSֻy!$k!wGN >$ vsӥ/i%surGޠy$km7F? bRKރ]`ާ(u$U݆?TGm4"^@s? Jro$5d.<~WarGÿNzt$"{zAҎy.H=>~mmPIu|ʻ޸#$Wx>INWZ䆠ާӵJ;l.H|wNPoo\c:Q%AzvӏŸr@1I.zv;z?@8Ԓ9WvtSuyvӸ))]VvI_mO;9җpP }J9H9#oPExRK`]oo\MvK +w<N+vCPIZBoSڏi%~$}>H@;J^KMB(71(ky m@; OIyX9V۠oq܌~ ]ȻZOPIl :iarE|;~uHj +Z]x;Q$Ӷ䏧~KIi(E>]mo$az}8)I/+*t ;T뷠kXI-wapGOU7QM,.Ha};’ A%kY Oj=v\!Ӟ){I-7޸tK䃒=3O?i%`[[nr1c*]v"kv<~qC%r7;i;qRSֻy!$k!wGN >$ vsӥ/i%surGޠy$km7F? bRKރ]`ާ(u$U݆?TGm4"^@s? Jro$5d.<~WarGÿJ=Poc1`~TH9bX=>S_/O|Èw9#JI+mcAʻ5oP/zҏi˲Iz/bu+u,8`yv-yc@1ؓڦ:SR(6ek䎩$}>n_|:<[{uAĆ(E)/7 428Ĭf:Hr6W[[a-5nV>V:/ /V㿅e៉^y/f⇂-ulu ;&w} &Md:_<xW5xgyeg^}QB:;Zl$]+[Ǜ_'g-;S"?>#|G4'siz=/ïGgD5躦iFWe,??_oxCK?6/>'xGK^k ſ/{MW/s&ΛoKQwi7?|=w~W ?ZW~Wt? |G`kޏ?MjT%mFS]t?>>xKqxSß/ |#eᚼnm,n^s^ k6ψ|GyZzNjufYյ69o=VQݹuflCX(֥# ;}.أF>k>C\&YC)]w e]pQx`AE>W$s^4㤛+Co6 kqԿkF[8}#,P(yGD_*Smc1룑=IuݩQ.^>$oIh5+ L͹Jj2=~[}v=B? 8Y Lqz+J KkYoۣ>[2,>)ڞ CM"D5u'Q`+&24r-żZ2tbp.jBҏ<\SKm,b(b;kgg}WTIgKQ-kX 떦QPme$duM45gC҄%.D7*]ձS>g5-ߡk{%͏BIf^4wjJi^[d,7Cr55aa(6Kkrpvz-q?FvKqdl&?鵂˶BVY.}i,KX##ӽ+/Kmqx;[K Go,l&~4Y ],p?ϭ %`G0hM] 7?t3NdZ  nYzX,l#i ]6H??+~e`G7`@}hY.??Em`dy1ЕKsv %HHtJe`G?NGFai[ .?H,K7Et7L1Z-km'Ƅ`]6 ӰY.GBG[zV^ .??v,M:3ďJ߀Yv7MhAdX71Z,KanX,l?n?L~4%o+gƝt:?;ҲYv7GӵAdlј~$ZV˶n?E %`]6  ~֋te`qc+yX,M>4KБޕ˶<OƝ %`#ҷ] q?,Y. n8֋n#c[ .O [dl?`]-`N,] ~4oY.$tf֕n#dtn cq>X,Mq6Yv2~hJV %`Ϲ?;k$t$we`n#k|t#0HmqOƋ$Kqdl&?鵂˶BVY.}i,KX##ӽ+/Kmqx;[K Go,l&~4Y ],p?ϭ %`G0hM] 7?t3NdZ  nYzX,l#i ]6H??+~e`G7`@}hY.??Em`dy1ЕKsv %HHtJe`G?NGFai[ .?H,K7Et7L1Z-km'Ƅ`]6 ӰY.GBG[zV^ .??v,M:3ďJ˶ec4~"jZ{/ x=QoYrϫ\!nt]^6;iLuAʞ Oo[[^ga)TcQG^z5*7(D#TXF*+㹣i@vVF~rއhc3oa*\Z̓(gtL̾SZ-vD8ttunO;~A0O}-n/"N0:g϶xyX|.u@DCtq]q:g@AF Z`n-z5))En+-8_!]o\[;&o;Mִ]bT,W͍eg 70+]5ׄ0u]L<6xݒ[&N~g>"L9jjkG‡p?ӷzG~׭ -<k)<տDGƣn9Wsr=}ϭ`ԣ˶_zaB†p_X=1N?JK`Uڞߏ"1r%Ę壧5or|lpqLj<1 S\Ird۟D?C9TmR\T85ȀdǮtQ~s+Rǯ<:ܩGɏ_f?p??Gmۧ<Zt.#-d۟DOQx 'ObǯGOdۚȇ(#Z^6 r G/@4G8^:ܩKcǧ?he`۞ߏ"?ß*^5F)Aeս~?GÕT|k? _2.$G=:s?Þmo*P9h/0??R?ԪX|4t$ 4տD?D|9 Kƣ$ -JĘhJyqC?Ñ^5dQG1XhA_njߏ"'(#j?qa/MNȊ>9Q TdV?kONC?Þ|l1Q*Th]f‡r/q}SآhS5nD>%G PGO'X5Px`gC EJ> oVȋ  Z?&=}Zy?X'?//qSܩGɏM/gV҃'ȉ #],c-:??H?pkOQGq/ܧB,_ =?"xPK$0.|sEKƣ]TY1G=?D?TmG̥&=}*_k*4o|s*^6*S|KKᣧO|s]/<?Tq> %oVGQGSܩKdǯGO'TN }! n?`_2t -??H?kNi! #q#=Tq&=e?kOOGr8_ 8RLz4t@=?D?C9TmL NLz4W/H?pk[D>R0x+ܩOcǫ{U-dښȇ(#j}S=}_ ??(=ucJ?K˖dz~?%G31=}Zv_sD>RǮJk|4UrX+MCGÑ5K)ս? =?D?Tl?`X?T=/ݗ$@???Õ]/~RɏI{~C/O"^6+Rɏ_fO%`枟ߏ"'( sRtc٢KN }qE<|Q P 1hkMCGÐ?x#|PP{C=?D<>!Ojإe =?QG/SܩB<{4UrYp=?DOQx 'OcǤ+ݟpkO#/.??*QJ:ӺKd۞ߏ")c.e*Lz4U<O"?TdǯGO/X5ȁc.Lj<*S\IےJ?pkOQA- ౌܧGɏi/ve`jߏ">Rct>#ǫ{A_jz~?x`FvN-??H?pkOQG|kǧ<JؤA_nz~?>^6ƒXe8?皷Nx >|9 G}Sq> (d85G?Ð3 KƼv_o)ҎOʰqSvc.G1).$-<-y~??Ð8c9???J_==C+R0x,Kd- /M ?>^1 q&?nZ?wuN }Ӟ?n;o(tRĘGOчV }qEqWǏKᣧg%V! #OZ^5 W?jT$ǯGO'T"+k1(3'ȇ(#j??ౌܧObǯGOsV?D|9T|l 䤿h?p=?DQ?v_8"&?_v_jzt"a,tRdǯGO//5o?Ð?xێ r?o'F)t" *>6_PRĸ:y?XP??‡sEK?Wn*SXhv?=?D+5G?W2?&?h?p? ?<6y8|1֗*S䶇[F?rKz|u9«Y]Oԯ4x/6Zh_UKCS# ƻ9prۨ*v^Qn ٵ[u=]vҤh"fnUFIb; 4rۮbXPfYGtp )GS@ FhQ>b;=0 bmu؊kmaHyEa3$I,*ĊʼnPmc{kt[G>3 u?^$<6'TVK^ے.-F&#rۮɦ O+>&.5;"tPNԡ&˯y,,D"%b%G/*~{&mQq}Sc+&9Kv'*~_uα^Zh&~d̬BVFM;PC;]m3M-~w.J. xO=C]Ѽm~O> zG-wLԥZ\zܚ,ni϶֝=|S;W$ig<1? x2γ菬x/ZXXjZTm=-oV WBY\=*#93J/~'O7k>%k+ٟZn}cO4:mϛ6'M#dhm-渕,no;hi#ู11A,l z?Ck"Oik>浩i^<#3*FH[?iwA3YBJMѿ 9.|[{|ok| VF߲iX=3YҼ? Zhk:|w ܡ cTܑv#5^ݿ$> <+b]?Oťӷ8|T _D?ݥV6V]^]m5B#$M+%i.]Mjb4éizvll];v}S!!2fC1F 41q``unOp1,Mu:{`$c1w]>ˁ0Cy ]v }܃)^c |v/o+yc|kh^-_ns L[Icj|1ŪߍFIn"Cq c۞8$n= PC/tӌz rO@8U]V|E><%ck؆fڎ5w !Z,Du:wF8qק>(kw=3Ѳq3r;}@y8ׯ(]13}9aq 6`rx#܏$ /nr |P?D.|QƝk5w<ڄ- ŝmyyG+m{I!/jۢVAvzGOo=# vHR煷.]~  ,E ]v c c3׎ҞH.Ol~tJH6>5t/x/pjj^9"+fcMT:G-Io*>T!F_ ؿ> oMkK(6ח~C^^i[K>\/>w>(t{Þ>ìY}VKXk,ohKv]u:ezrO%+y ]v8|Gw. "Mѵz| 滩uV4srmodk/4Y$bmV^`Z+6;5hIcGX_ -J8ϨZt_r ﱟ?xwriV_w7V6zjXڵt] [XK1@F$6蕼]x_Ǻ2k tk Rkk+ءEn5i_Kl=r-5F:7RZA]1Z:?T>l'LvU8uEH;mrZL2Cukwi#-՜駒 cOt[57o5K-R ӛVe50k1X}MgYc0 V] |X]ޭec'Yb@D[t<9ﴟ \[4НM "XIG]蕼}xdZcQ-+k<7{Rv%ֵ#NѮ/|9F,O6M~u c]Ey4H{ytFfMMZ)XGX_ -|J3:Awc?~2ҭ˔n4}/Mҵ}{Y屵kah6F\N(C2щ cf%Bhvͥig&.O-4Ny (,//sмM_ nOY&,>An&ۢV@9cϷ<ܐ;<]pc7$\m~/pzd2xgI%+y ]vzs򓌎率#1ۥk?u_ 爯>^Eioy-eq}qsOmp[TH.e/ u?xBwX>Exugo}l1,rȪwd%o%Avcqp:Bm`_-vsn' tNO$}0zqY.[]Q'[MG ޥ+z͞GnEֹ{CHKH47~7uw; ?T>l'Lvƕs:E8uEH;ܴ5LZHug"Wt]bdӣ<[6̉CslsF %iu:{c16{1u%蕼{zzqHӧ;$,WԷx'Go/kSޛqO6HT`. e6蕼]{g8A 2:Kk?'?"mkl,1eo(Vo\aMTs`#ddu-$dK{K Kk; y/n6{Gp$#3p^V ѱu;/e{-KE|?yj<Zޗm 2m"ISa$JΦx ]mcҵK j SJz}ךwAys ]4*.35aXn-x܂ǟfk{h=1qGQҼ;{qkZ/C5 ng}4r<BIsY.|W|ci^ou/ĽBA$y7Vx!7!Ѳ 0tJH.z;qF0@#:Oo]DZ8cOA:o xJQmuXi6Qmyj:{3KҴ[SPL]M.*'hvpt"fEtO-S{xetcE:t߅'cS+<$o,@X%ֱkcMm#ʤ ocҌ?5g+:T#3ᗋwS5nd/Ⱦx4*zsEִu(?,~%-dyX-.mMU'NDR2~^:,Vz N㸴lNH >"le7atב|v%#ே?j~ѺoaOo>.W* _ x_%VZFa|WZ͝Ҫ]K#G_sֿg;ĺ /pku_:siƺ7t7xoΫ2AK5qgԬ"OeV>o~O^~Lƞ'l>*]/_?oZ*u,Kk ҵ.HT޳Ms$ItG#DW Lk_ }0y3z7xp~9Ǫ[g~EA_e xia5햣2$,2./1 cɂp^̓ G%L˴9nTu1؜ :P앮fikH~l5A{ @GӍ7p7`>"ZW@Vm[^V7SGOx'A-3[2]_Z9-(}> f?ƙ>xKSyk]E|)Mn'MLK} ݴV֩qus6wGĈ<:Gr+OKCN֌~A[l]I-ΟrmЯ߉: 2Ѧ\Ω^Ҽa.-=,OS'ӅZFnMA?uk?g5oOӠ@xZLny+[=WCkڽ&oȖk m UWYE>h|=h.>5&ίWӵk8mHeW% /:g_k |1Okiy^r^h&=] W΍H$EA偷Hj_gqq|G_'..|cJ^kھDW+Km6pZV61!xWUw M&xoW 85׉k[x ~:nYo=>#|giojj4K!6 |K~0ƿ,>3t_@tK{M"ż VzQOlWSiAaɾ"xZxECa_>k\t@]FۡK(?Cu:YM)ƥq\Z޼ov GEUŅ_0MRm>ikjv Kkiu}/־?ڽΗ#oLtlx&᷅<}_\{5喁 D<)6o?C1t0[5NM{V|AS>6a!iOȆ.fD goY_ [L: zi n1cCX=Z,K*h ?k66>!XCF CV?v𞟦¿5x~Kx_Լ?񏊥mfk[aQ5v𭯈G BԼR]i :r$C7F`q : oM5=TּQc_v۵9lenl<[GKo]IfZ|waC\(@,~%#״]BEօ$Ki;}G?u_6_Rqmx%|?iAƓ,@tR]wMK֥gj= 줷/.!Kco~7O^+д߉Ɠ}R+x/Y-O[Դ_ )AM-3Idٞ`XO&۷х"𠿰x~ ]k"s `fP?+ kTOo X\u7^/pf$ ]a77Rk"YyqׄA}'P/nTMcq4WW}GT.oM>ytB{Ky;xU;6i.T.e;)~witr&|hL%% o 9ĥIoP0.WMF{=>kqgitڥg> K]/N|.?g@3Rմ]k[%/]JʰF~VgZo? w>x^)tF_ 3 0.|iܗ~ kKG #om=᷌ ?%Ѽ= bt^%֕&kt}Đ5kD5зo|/tS?į|P͟(MOmmXϠ4kqmAj|qLȍ.m zGUoI>"7úZt ? xUӵnm5[|ep?CӾx c~= w763:uh:>csu&,hKrM;MDž _IJKO6o⳼յy W]fi|cЗ_#:ƋXtZ {#u,7p*R6:A;+6 `|_~ b>"5m~宼3tXEHmFY0 |/ KĚ=߅+~1jso.z$.y!0놕G|?Bյ 7\}ON^6v Ud<=cZbr^kksixm GxG= K=.&$xZE-..JtLs1k-~o|q%5i~:nj>9RiWZχ9} w5,{9-F8⸺!%I|lFsxWD~>to A|2 Y]6[aww1\RogyVSom>w& |Kkᆇ ZpLZ -Qf.j--d4ZA-'ߍcKxv~(>M彾|?Go[o >%{YfK]d|i{[GC`|Yb{5~.x- [tKG3$ak'Zx_7oucNKM{qek hd]Fq77ws(g>'VYELJdk%귚icO a ]dBMc<=kwмGbxƙEx=zWanch{vA]I6V<-K_x^(5>(n+OwKOvRkZ~g[\iQGth40\R:džY ˓|ZO[im5KXӭ",h]-M-77(7-b[Wh.԰JΒ{XZ[6rƳ|go Mx~յ}c!`Ed`bg>+x_“xaxli af u O=s}{e[Imo%AViZƪ<%J_w Ztzu_:kP(-?B_k"=Gz4?>+6v5|)4oz^)KԯඉV)'"$O/,P}ƫ]sZkzx{ >^$&X>tܲ\\ΐDm;|i?^/}c$4xxUpM}.چ# )olwmqyeOVLl/]_~jcx^_3xA|RxoYҼe2ϯj!gI͖=ÁxgVw6i^owM7"xJ Y{xEޕ=KIf6QZG1]wǍ< a#W K joK|7cM.t; x "hv4ZRO.T:ygYj>>!W6>O1h%uZzw>׋#C0^M›:=^3|/tYֽqJF{k#35^|@~>awSl<7}CNLJMw?,>Һ%得{6,uuK{+ hs+m2A< xWĞӾ%[I~Zxsž3o ]6^`fo%ZM6ͥwv@mc|'xD񿅗JѾ$C૭Leo4'þ ڭݎy+a  x^x+?j>7:OuO zq?^nĺ~eK]?QKˈdna,&AP'_]xBEZx&N[?;mfծ-o.!ӑD|9c⋻W+u_..,hspfP, [VKd6#-"hi~{4I65u@bҴQxA姒 V6I}m?}FдYtkN sVd_2Imh/mhQ _p<'a7$"T6_EzϨC"r"2XBJF|C$uoZg> g87?6>g'ƇqlQZՀS%[zl|!kO_K? 5;.;='MZ;K[Maf F~G]C%h^#4mVW~ˮjeoo W{#ygYo.h/|C1['G-y2{."wu o+a$E3oMyX^籠bYV $0wFem;MҢ1ZK|+4$!h v9$Ih %4#3׌VV_➣c}[fi=av5ݝm ͼ6C;gN[mw>G>K=ŦO?o%>YIȞDؕ7P^<۾` ['|1o+[hٜayeVy.nfIi,1;+%a#VETvuX9b$PV$DO@& }Şaݳ^[ZNO]WC싛=SI*| :`GI)c7vOwƚm`mwKE4(҄mN{Df-%By( 6v=|+Xhl&xC5ijO+nILL٧KO'Ú5'z6 u<|D?:kI^ubttۋ>G]_ l<]v֭qm_Yko,oYk>m:c×Tp%ռ1W3S>džu WKWuK>8tu7xNӴ*XlmhfM*Q;ymyVᏋ>5<5g^xTo|G|5=[^#nYjZ\AjvrisYtOps5c>كd~I|_woǞ(6ix{W4O+[VѼV/kޗKy^Vf~zi>/'Xw/{xnOƁ6k]sLvZyur9w$m§PIixS%x"}?۶::}o 6M<\I* ]]~{ W&a#-/oe2G>fϖ/Bm;&PկdQuڞW3qwq4#08Ql* {?#hm5wkԿaSl;qm PM7$ڱj^~894E?@XWï xMw.qgyiv2j~ì@sBUD#@bsC?Wii|R[VIJSkzk7MpM5ᔼ xVԵ;P}IC Xnm_k0k{-.{0m`l5kKxCź/3:VkzOnt)a5,k|9vw2JѴs6λo-BG-9Vx-XygsSu} 0iw>={K%:IץE⷟MKQ4Pl']VZ4>kͣ/=?CEAaӯm- ϥyhlЕJG)Ao |7L~_djMjz9R/m6&eyR D`qgJT4Ƌ i~Bv'PKm*OwZP,4vnQ"wCs]vmk>"JejhBW&K%maR-."6: ᙼI'RPo4}Z:VTy!_+{EnV0m`'D4BMRB$D{5d"-nb Atw~% Ӥ^_aYaMpMC]`s|.$sB#Kh-e/mVNWSҵj wtqo{.$76 |8;wx\  h5k ~ZEa%ޥuw[mFNfKyV xa}߇ -} h WmL4Ʉ:iV UO.Yb,bDs xOM޳;M(S֧ KywWjWS4my=*$H*C`.|++Z%ƒ_L&vWkM:;ŖɲK9dnho/W}KRuCxÐx+k{}~úō{pNC,cRsJK:]Ε{mwV[n<'iMC-/$0s qo+.;zQ¾ V} %w1kWkx.5}B[뛙u nJV y|C?;-]OL cYo^x H6:~,-JY{ eihØu2!~|ӵX<5S?6vW^%մX_Os}h[C X홤 mr~xo𜚭ƋepŵֳZkZVcb޽}yqyp@("d.l gx+G°D{V[MVQ֚-&ȥf1Algׅ<1.&o2mZ*g4:lm!m4ז'Ra9 b:$>jjxR 5dk"-nb A9c Q𾃪6bn5 k?znoc޳VF-Gnbfc#ῂ59/JƚqKuhE"Zx]J Pg 6|oݼw6rCwҽ݅$R=\bRXn/ kpi&]ᫍS KIԼ5yawaɧK8fH$HayygyXΓoJ΍g8&im5k.kf)75ߝ2 F/ iJL~glm/u84 2;oj^߃wGu8}BeMu4yޅ^:?."РխtۈoX[[%nԭm;OG{ -X6,\ivwvsoikm5f{EZo?~gn!H.6*oGH?,6ZW|L|/|@` h^/ կ,cF6j ՓDCuY]CPH=3y FơMo#|;]FÖw6575}fYF[xaLbnw,lݴw6s5Euo5iwwawsH3CscpeŴKau`c<-n$+|7qj~`r:,.44Ig  !o"/,o+|$%.t| oxQ{Ğ"Yuk;Ԅffw&i@ŋ?>to^!;{#ղ}?Q% I+qO'۵6DZҭK ]Gvk;ˮ".'[McGоylcC&i^\?Wڄo~&KOI$+dx6,[x/Kπ|H[t_>hjẖGQ1H66 /x[zrz5/Goe&-/is[JfX!RY"$6=Kӵ;>H[i$46sB-C=w#q 2GtV{?/A accu}dGh |']ޭyc4cP':ȲGy#T5P5 my|6^V/FX/4jݥurfOy" .Mi$52k:6׺{6qtl8"X̓oygRl뺆jW~A*/,584R+܉j1^ Ӳ7b64wJ6>i#yKY]?z]jDXoY_ʦk_&fi l0U݄vtU5=SĞ(Ii-ƍ>bmkK 7[ jWB /Tڭӵ ajcOi/"}k–:.ooRdgo:MAf;]:xz3]˨jz-o~&c3Xkw7myy-goGc> /tK,KGt[yaFζLA Q* AȽ/4mC!ӭMT6*9,#+Y dhgXf9t5uo#Ms2S]ƫ{hjW(5\hKKQinydv*fG1`ޘ;(64.~:F] LMfi{u"OSŒMmommi Yd6\9uqwjYgkyXkku&WyZA:"46FMQ5m[\c,\9u˻۲$$(ES`Xmll-`,mH-mE`#HUTE /m`ȯa`/-\WRZNtMWFtx?H!mظI yYͼo"xᮓBa㸞=A`!ޗ];({RvYH!Q)6 %|4BwPT4xN޷}qτ5t]T_[wV|7fdm|ßw\bֵ/O'6['KkZ=VdHD@ȱ?<37jZ_4}Z:#_jZi!_+{۷ٛuo+V:-.$yth#Kk7t&sc}{l^2vJAc?^=A# mԁz|>^ >njƜ{Zݥz> {n&^\I{vJn2yF* ,iKY^}oJ? _8o!Ҵ,x?&/eͨY+t K>3x+U~ ,-5=\}NklS?^5i\ș!'R\~ uӵ Nк_xUw"͢ju{i̯ ۤ&?CbxGQ hOsqiAoc \K3\kl s[:JƲ6r_Y b/5Wv }lj}BM2[k7ѡ[3 Ql|sS|j|E +Txsƒ-+i{6}.{FltU(s{:0Wi{ӵToz;j?$X.dCWڿ7Qû`}(%˙ivQAGk|>b̿}b_7a Q? 򾏩b̴(/yA['|(y <$oDIg,J2X?c0;j#p>2V bh=o飶MoO >aA51 /bW`7Q5⏩b̿}bo z''*t{ [?/tQ& H*$*Xⷵ8'#EP;QLWkEiw$vli?Tů}j$v>|1パڱG1 /#H?I꘿}j?#O&bA??~|M>?>i֨C;7h/Z?#H?I꘿}j?#O&bA??~|M>?>i֨C;7h/Z?#H?I꘿}j?#O&bA??~|M>?>i֨C;/>&|h/ZU qϏW꘵0Q_?#n?揪bA+^E[G`?g^|ר꘵0Q_?~|M>?>i֨C;7h/Z?#+7-4Q|GasG1kaTW?#zϏ꘵0Q_?~|M>?>i֨C;7h/Z?#H?I꘿}j?#O&bA??~|M>?>i֨Dω>i֨C;/>&n|~zt2;LZi֨y׉;}sjCҏbI֨yGa?>&MT4T~! |4}S@Q$vo $L_H>G~Gω'G1 Ga?>&MT4T~! |4}S@Q$vo $L_H>G~G'$v^|MAG1kaZ~"G`?|<_'֏bA+^GO?! |4}S@Q$vo $L_H>G~Gω'G1 Ga?>&MT4T~! |4}S@Q$vo $L_H>G~Gω'G1 G`?ϏT4T~! qO>?8Oy꘿}jGa?>&MT4T~! |4}S@Q$vo $L_H>G~Gω'G1 Ga?>&MT4T~! |4}S@Q$vo $L_H>G~Gω'G1 Ga?>&MT4T~! |4}S@Q$vo $L_H>G~Gω'G1 Ga?>&MT4T~"Ea?>&MT4T~"G`?|<_'֏bA+^GO?! |4}S@Q$vo $L_H>G~G'$vo $L_H>Eȫy;71OLZi /#H?I꘿}j?#O&bA???#n?揪bA??ωϏ z"bA+^G/$vo $L_H>G~Gω'G1 Ga?>&MT4T~! |4}S@Q$vo $L_H>G~Gω'G1 Ga?>&MT4T~! |4}S@Q$vo $L_H>G~Gω'G1 Ga?>&MT4T~! |4}S@Q$v^L1sT4T~"G`?÷|p@P?!G1 /#;7h/Z?#H?I꘿}j?#O&bA??~|M>?>i֨C;7h/Z?#H?I꘿}j?#O&bA??~|M>?>i֨C;7h/Z?#H?I꘿}j?#O&bA??~|M>?>i֨C;7h/Z?#H?I꘿}j?#ay?揪bIX>G~Gvs`L\coz ~~}SH>EC;7h/Z?#H?I꘿}j?#O&bA??~|M>?>i֨C;7h/Z?#H?I꘿}j?#O&bA???#O&bA??# tIpN?{gҏbA??q Ϗ2rןT4TWC;7h/Z?#H?I꘿}jO\Zx~>÷J OwTo__qޒ:ozG7Q{ 7Q{ 7Q{ 7Q{ 7QW Bҝk+y}>KAEwQw]{0/}>KAEwQw]{0/}>KAEwQw]{0/}>KAEwQw]{0/}>KAEwQw]{0/}>KAEwQw]{0/}>KAEwQw]{0/}>KAEwQw]{0/}>KAEwQw]{0/}>KAEwQw]{0/}>KA}JWkyY}:z`g c8 ̓M_A1 22s~\[vӾK%'uǦ@ F8*6tͧj4lr8#NG#E4}/EaծUj|?gӷl@qF;Uݯ~X. +{$=>c gqoKn]E{[A;c*O3/+hڶj!mբБЩ2p>S{="m+]--~mgݥ{u@~s8/n)~"]^mo瑴c p0s9QmgntOnxpNO<wdm[W\Nǟ??V'{bM/O[3 ( ( ( ( ( ( ( ( ( ( ( ( (Ƕ? bV>jOX/&HFTuXxLyckWZo}lv~It,bƒ581aM_gĵJ۾~\}c{/r; 5x |`/,6^ ]2>!::+RKmsI[5;U5m :8owRq-Jq zJ.2k\v[թ,\%/rZ1rS嵵Rvv꒻# xųCV@q1ʹXonxw䳾^ڜyN9u[|~ ~hWx:v<Jѓ[?<)5 kmJ/`m>Wԭlŷ5(ԕH˗RKZYE_keUX:J4.#ݽoƷ'|<Y.H1eޯ&FyagHOWZ%Χc%[ϦŨ˪Im+z|Qfq\웂nOtŏ*=4*ݝћJ.MKE+:o3~>!k7V^)Emqm+~=KO/.JӴWIdD / Rz卥{ӓJ-Y+o5b>F3N6݊Z]kr|KW,@XB.5SO |I|9 .IImt?h/!ɢDׄ|5/)^usF,1jKyVҜR"ukK+6b[^#`o<4þ3<xWk^Ojk[IjCiy-坣^KRa(ϛ+W|\usw i;s5+:I.k%g?M?|P_kφwox?KoTSզMΉvHf[DVVSY*wգ&䟴V[[iwk*ifQc$^EF+kSԟn - %k>#toZo ~&Mn[Eӵ;aq~t1Τ=#Z0~կw['kk~r1)OR8^׾ <?O\j4<JCsEuǣjiW-b[0PNcJvrnoh}bh*gR^փWqIE)h];M~g|PvM}*FkvzŨo#p$xE9a*psuUjQW7VZ\ќFvźvjl|gՖ7 xuoj5֝}Aii:(ռ!].wMB_^[eL;TӡUE.k4h.^nYY_K渚 NSE*ݼ[m4x[Kuw.mi?|? h-7ׅ5oT1|es.}L/4Ocrn?xŅ$R(Ѥ[KޕЕԕ)5$mũ9$߻v}]P@P@P@P@P@P@P@P@P@P@P@P@I3״ҡ\RI#ӣ_ թ.n['tҶ۳J^ki1O ^\~&v[i?toϨMz4կZWY rce5YG QTbJϢC,&%V\T3?M<@<kOQ ^6'Fgٍ&.bl>:T6N0^IlklT*G.W?etk-OسH|R_B^(֯gI|5O8-FM`K3-[$>1iK +kOu*/X'6M3Ju K[6=Ɖ":jԯFNV\Ki'֩[GU1ԕ&/VoFl4_zOd_ޙDqa%63 = }ܹm(ns:+T%eϪV?d~ j5>5gE۟s_xzxnt]dO]#kuuk'H=GkoYk:5.ǃO4k4,|92z uQ䕯;Y۝LUxLVmo{{}!+kK$wx|@ŚICXE^-*ECs0UgF.IFRqIWwr\U۾ajRjyee%Eyk_~?mwz c>*x_ƑxǾxFGYmKiiutqV2GZŖ) _XB4K+]Gkp+aєJ\ihdٯ炿cwx7o~'zStWN[|/4OP<'w-֗*sv݃$a(ԇsKjúM'M|N"z*mYtwkWc~{c?߃a3[6xk◃ Ҵ|?0\xY^j׿mB-K{FsE*PQ)~UHA×H4OޞBW{9-T7ng|*0~3kx[u{gu4: l-y}kd#쵂;@F+RrSC 2n ks^-ҵu4+cecA)%c2~?>&x_M Gux;ҵ-k> i|%R/jai.uMNuMUU֮*UB F1uWy6"&}`Mi~񿍒n|;a "8jJQ{JR{B4N7VUm,n)( Fm6[;{e([I/~|dek4y +OgxmNG>.x'-"k{H0}%zTkµHEkfM7t1ԭBJJTi5).YZNJ+YmM%iO]Ÿl|%OkַWOo|C'4Oc-ok[ZCwuGA,Uj8;ݪFri~E}53iGߍJKvwQR}tkfJ?Sxoߋ4_i{#6{kC{=ե%Η5<cxi{F,:ZQ(J ;=?sJʝ$ݩ_Nf]ҷ֚O(( ( ( ( ( ( ( ( ( ( ( (,7gwwg㯲\y5 dfxgf-N9 c,ejTœeF\ݝ]u{6'<<9"ۍJ-u侞DZc ?ZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>_OO.*+r_GM/QZ}fK<_W>'K}\/Pq/SY?a_榟ܗ>?N7]sz7%U/Q}t1%¯I=>CSOW^v3?S[tmtKt=Vt*Yr_Z=_ƿ ψo++}Pӯ 0nHdXBOe{ms 6זsksmʹ\Ae ֤$\_uY.T uc(ϖ]:=5]Ӻ}ShwnqgO4<ԅe_W #K;':U)M8nd5,}hUIF.{;+n]7F?q?tk A4` r_Jrn_[ߏ_%A/Q/9xd4j__x}^V ?~D}^21ǧ8G}baz\TV?5<Ǎ#/aWeӞ6`R,?q0CplV;b)FZq[z0!ijY+[[/cazcty>\qGְb)߿I/ӴUMݽiH>__M]$jۥlZDrO7CI<I?XTZ~?>=!˚ݗO+ZvBdjazw`~CSZmCbƢeAs?%¥ooI%m9Eנ}_EٗWwӹ {'fG_ ol T`'ݥMN_ jtW>UK5m-.LFHsWZJMUGUd^ዣ5R)i8Gfׂ+#>Þݒi6aވ-`Wywa$חRusys,7W73- :q!^cd?>NRTkItDG>@Gn_ְOz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KW^7_na"? c /Saz/}_z>>_|T%A/Q7GְOϊ??u)q0S>@GnZE??W>*  X_O_t}k A8}^KcdwoU_^VW=^Yʟix}}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/Qw>K_]{}AEw/F˨&K}q>ZK{5cF|A7O#7ut#kyYGqEgeBWR)K֝x֕8[Y988q-&T}Ⱥq5)%mﵻgӭc{uKt_5rudkZM݃ w>=qpiyE}ktVj'kEwkݶ]vkMہd'<8֮)&IIJ,<;*)+aԥJRuJuZTmpRi8E9>~EN2ir4ۺ6=vhOldtה_Nkelt++5ݯ )4m[^]#z9ܣ˙i_m=l>Tj> g동&ޛWi^K7l;Y/%vk{gz/ͷ#=zӦ 6d=M;}-Goml?:v#: d0,pq wn_6ٶKJVIkA Zr2 4Փvwz^]봕lѫ/c:q{l}:?ؤWlZ쏘?f_c/.TO ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (z׃wӢm`_zUޱ]'O_?mCT߳= k}gK^3hڎs3[\.qye}[ $*…>ZP7$TSN79A*Hzz%R^rIIRiiiY)2['NZ+-c'?h>Mk4h_'ֿ5]B-BK7ԄxT:Х%,bTyt8U(ԥr΂U)Q<740F}])~p񭈶JOUqh0JT9a({±k^<x{Ua߆j;Qx_''և{TvuZ/uQ=[ 5;{UY*8>iJKNJ_b*ژp5*cRNPTaJ'Fj{:ԥJaoJ4JY9iU$rһ!S&e¾(мU|B|LvB-t^|Nžӿ֭ehk%֣o[[ Ve^Ex)Q\㇅ITV,<N ZRatRjܴ*rۼҩ)rZGe]Sq|_c‘?[:ohߴu,5Z+/;8-~4? >ɞTrc則l xB+o 8aRQJU#JxJX^XETTY:|q7ʜ2o<e7,gdžkj|O[jy2zyq!i-.᎖Qge0,^_hG J>2N=:?ezS#Ni1UzSa^UW'%M's/vҌ̚WH<+Woo]z7|?_?b| ڍh5-ZOsXkڬzN%aJpXYc)iVҫlElSN4*'Mҡ(Ƅ*P:7WІ&)RE,77<)ʅYya+StJI8*T'` Hg 1 ѿW:#gھi{RA XSO:_Yi׏Խ*xgJq~?P~Cߖ8[Uny*2N-S.xY}$-MVԵkJ_~8<;~ WY5h^SMvF; iȳO&/0Gю_N:qPX5:ڣVJy$q8ªq3,$ҧF>Xƾ.èڔb\.Qme%M,o_|5k^#<^׍|MyxT_&_xkƳ;}Ik\w\xNF+feXzN >SK3p=8JQSSTŸSSN%MX¨(K\| 4k'N|hIJ3i9|a4^ !í_:o?[ ojw0OZ=WP;FZݾ]Ϣ=//&˰B\0S-yS3*txyԞ*XiQJQp*%8'Rkׄ KXF['7JPa;|>,,LĞ%FyhKoFhUގ&MI'2島xZac 9:pS^2pPRJVpw<t'hIE)Kҋ|ZI5jϞxL񆥧l(kqn}O:4ho57_ |'. zVW,,/`J?IaҗeQucɅ|"߳+2IS:ru'Vnf:EK,4D*zt&ruZN+sySsrInjx̿ןľ`FR|^pZ{?MW@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@?F  ? =BOV|{isv)*P6`k| *̊r۽ƱDҴV$}\*Tie K&i/v'VqJ7rv,EJ1qi{h{(jBN%hz5 3Cc @gcYzv]?ۘqr땴(>]>߸‹N1M=)_N,]gg^kVŒW0?=tw>rķ*.;tVV(h_m#g@Q ޝ|Ju0ln/7J ܶp"1G1`J*XYBRn얕1z#.(ݹ[RmSN\cRM{(w_Jrz$e+, c2#oO?!]O=ƽehʿ뵿 EٽZŒW1`v;qo5㕭C/\YFSm6V*ۗK\.e 3_Ōc0qY?xĒ_h__E׫Vn˞Q~ 8xqۥxo|rV_I'Wءo}Q 1#9 ?1/oHʴe-E+kWl2KNֳjw ?mJS6-w}?j~'mt.6ԵEEqfm.LRUhJ8l<)sA$J4Z^=%FFNroU{VPlj*-ֱNFx(p5όjhz N^:ѕ[侥e6o場JV^ѽWә[8wlxpF[zi>7n-f]|M Mn飉.p,KX<`$tӹぐ[Lqz ~zq Ptp@{np  Z\j֚~]iZݽƋMi:nu֏~\yR5kramy ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( OW?zǍ~(|]Ӿxn:K=Sv:kxW4#^?~{.:֩gc9$DCcy'⦙BjkmcP < xxf]M<_:˿>%Ag/ЯK _I]WPu˭h].~!/k?uƽh|Mk[⎇K●O3xH r=V/j-7u,fiFn ~)_ |&w0ֹX$&ɦ>#wWY|C>;xg_4kF[^񗇵 /hzN"$|tߋ>6 :G~9[];H} /Iuki`ŒlgPcwo]ψl4nt[KV,gOxSKMZF7-/4&ˢGE,⻑nm`WE)6/~¿,x_u_̍ $Y|S  _RGK\xvyO|-k&m %pOO/흩3xv' 9W|u[] owr_\[cR|QCoG櫪jn#u-CTÿ:N/n5 To𕶧_I-x[z-ڮqw_hךV3jϲ]<#~%|@}ua6ss=N=l96ŷ/H</PgOC{o ¯Z&كu3_ol۟jvEZ|K]R4qgu)]+O|% O~>}u_PGmWz>yM٫h|C¿dcA4O|2/.|_+ZFcvSxJ։}jZ=ᮟkeg_vKwr%'g[?z?j:_h|S >(qD7Eּ#@WNjm+wv:^2JӾ|B:O@q O;7Ѥj+)&m~!/n񦅥۝O7ZMv-<+z证5oxVּyxgwNiZ֗cҴš6{/ -{(#|-&+_CZ=RVKsa-|x.gZqxWu C?xcAuYnoK^wk977?_%cK/e8C|#|Y_ ~|O߆ڇ?7usDN:W.ÏPi]=}z>^ t@ӷ>8AZ{ sx_HF mlby{xB8E6uzߡ=Ҁ@mӌ``c8POnsp8tzqgxş?}X}i >_ gW_y&8iйxX>)>Uk xo~:g|!hn.FmXBɰ͹id#Jv71ۮ1'4 asl=J6!:{c96Ap tϥ|=tn=>?Z6.䓐NH>@s:A⍇vHAN1鍽?.j>\zqt#HϡJ`g#? ''iӎtǧOqN( A?N0>y@ݽ;Oo {s8ӁqlAEtG~=l3ӌg=8Jt:{cqIMn@ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (䯀ƓDϏn-?m[UFBD8f +vjS5oO [oh)lj}ؚCp$0}lnd2-[}œ!loP@P@P@P@P@P@P@P@P@P@P@P@P@P@3Ϳ~'G~Љuk,?$hzPl>*፝ዿz-'>OI(n|-^[Eekf_x,´dk?C+C]6M&Vk?h4&m_[x+>>㨼{ xi_|+Z,X@t}'Xֵ/Yg4Fk?}a P1}hu>HenY%3S^/g!xFDo\WI:_|A}v}3\Ѽ!ZKTtRkxfyJeH [n.6=N2FG :l8'cekC]? ;kujqFoCƿqZ@m|j[ f]Vw_~j?g|H +|;s ڟn>8uGؚǜ5Oy_7P=4_++z埍!w\ҼqxGJf[eFBփ×E%q{K{ML~*.o>Zֿռ]zݵsYeKؕe"k>o`7o?%1il/oO!E;+'~+|.~'ğx0$|akBiV5c4k}<.&Z(o4:?>k@ZZ𷋵?.jIhZ׉um+HG---(صZ.Ǚ3xJI*x J𯉓wOV ~f,&{K 64~CI{ x[Elj$~ w#0׼c6b.{oO/mG4./ywM@O ^-|+xPm'GIhzuE?1ꗐOWҡ5s˿RM].z/R~)im7? 5-n;oj-¿hQM>[zOn乻yuo@[ޟo|@xĚ=όt].dSռ1Zv$ZΐqqmDjD9qn6ywV:Bp.w( @] P}€p.w( @] P}ço]Pj>z>s?z\:{RZM_d1C߃(Po5)G<#?21`>NYo}h_ړ7 d@k k>8fO/CPWEZm=&7 [򕉃1fZKŸ7,FfOVXCFV<!Gk?>?0.-(AyHϋ>xKC!K_~"2#:O }˥;XKN[O6 P#Pw߬|"mRϸ?|% K@s> /6 -3^~zc,b NSX MXM$}&w2MYO,$l+lnINI=DNХw~.|?֡wU?w(|Pֻo:/,K +ž9gM%|L4z FoyizG iAWM|]źݟ%x_ms?Yx(x^'>{? xKMwuwF4=/N{{E[l 꼊<}Pu7ur_> xŞ> ~'x#v^a&}[.tM/O4^ԯti;u9<5#㷏ߋ7Z|6zߋC]?jŞ ?ZM5h۴8caqC%¿.O2o_!ӠOѼ9-/V>/B1Z@ּ,4MZ; N}Z=:DWKށ|;@ċ&j/g=_–%b~ßI-/u_M5hjWi4mX.eǑKOxǟ VM_2u% 无LJ7s/hҼ]x:ݼЏM/rt=KZ +T5c? ƾ.O9 GY_4̍aM_*?'AltCZ?thBKƆ|GulڮާoV"obk~_ xVѮ iVRAkj[iܛyԭtyX957o|0<+l4σ|Oټ?ooj_? xR6[뚅ڋ1d5yb77h^oO-_~u<ǟx{GuGo6 |KxQnp rxew&ɵ}6 eoس{cc+oV?>u+i_CZ_ ꩥͥj^[oxu;MeQ_h{hYj4N^Gy ]MՌ{Y-t4kew^]/x {8|/ 0x@tOYMN]Ilot[_?~CU,}Ï0;$o_-௅# g _i:l:cueեM2]m~r~G+V>+{XYy]5 WW^a^krjwz4]M^\@s$|/h-3T^.cbuk^"΁-:IͷN?߉Wq|Jο )Hga"BvopDZ~7o~4ד? Úl c?W#Cw!?%~6yW_|Ao/8 /-~ź/ī_@ҧm7TXOEd@<%_ _/|=Di'/OOA:S៍m<'k05◂4t O_~1[WJ2%75[c3~-sXgY3|TҮ֏=]?4 j!h/߇6ii}{kZ վϨםנs$ 7o6^zW"h|W_ 6ik_ľӬU&a@P@P@P@P\??^KOc>z>.{xztѯ&'~)jzt,WH5V;]/Ly 6 FtZh-;/EZLEYn-vԤ&r'Gįڟ _l%-|'6|G 6ox]ź-] OĖfcuóK/%5P})>&mRDoҠ}y!^x}O᎗6k4 T;MF;{]vE%/N}|SW½SE0Տ|7o~/#x2^hGՎXŏ48a1yWZVsid,!^"V[^E𝙴!(T\cJV\n_}K>ON3ߦz)N*ܼ~KD}TfP@P@P@P@P@P@P@P@P@P@P@P@P@?w_CĿ1qҪu:>:€ ( (큏N@zݨ ~ =8?4ltNcn:d8a~Tm G aG|]Z};E-o)2&&xR8 Vϗ쎟>â?'R~&}m &kPVo4\hVKSM3ڃƯŹCk[G9Zvo41X c}NQ`J9_]-3_ k^k3k&i:ƍ4WK-4KNФh&x9cՕ'+_nKe tƉ?ϯ^r]5o\:.x| tL i7W3^hjޙۛTKIu=&i&}7Q}Z4h6o&TsoM.Ql4Q`r?鷘Տ :ct~'!yڇt&}BJzOG+^=ϼޭ~4_So??7ߍr? E=xC>zOG+^=ϼޭ~4_So??7ߍr? E=xC>zOG+^=ϼޭ~4_So??7ߍr? E=xC>zOG+^=ϼޭ~4_So??7ߍr? E=xC>:(=K뵾d k<^]]T/y ƳTk|1t׽.Ucm[w6T 9W|ϸc]S,(^{0}z<aG*3Z1q j)y[s>ȿnf&ydynYW2K4IJI-9ZtJ+/91?RT,?N}%A[X~&[z%?a&7G-:Jd>VjĬobiXMB ( ( ( ( ( ( ( ( ( ( ( ( ( mS;E_8ULO͟LWa@P@P@P@P@P@=1FV =1Q@P@P@P@P@P@P@P@P@[S6 @P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@?w_CĿ1qҪu:>:€ ( ( ( ( ( ( ( (}{}T>2xE߅4_@/oMZCQ.eqaDvW#".lR_NiDi1+('NiDi1+(O0~?cWPa 99<sM%L_@ry4DK"i73?EӚo/g ?4"_'NiDi1+(O0~?cWPa 99<sM%L_@ry4DK"i73?EӚo/g ?4"_'NiDi1+(O0~?cWPa 99<sM%L_@ry4DK"i73?EӚo/g ?4"_'NiDi1+(O0~?cWPa 99<sM%L_@ry4DK"i73?EӚo/g ?4"_'NiDi1+(O0~?cWPa 99<sM%L+G9-:v_=ZoVZ$R4~!>yKX8I@5B@hQנZ?,}W7!W|RKoflamt [>'sEc6j[{[Kp /LY($YeuVtb4{oO $אǰ=B( ( ( ( ( ( ( ( ( ( ( ( (ٷOQcpT90u?6}1_>&BP{Kk 2YuMkG%Т]WzLqu*C~EE+E/Ə?c:|Zw>H5iZĿ~4xK5xN֭-|g&vl:Z^xòzo AOĿ`n]x o!h^?JִO5=J]1[mk?.#ZS 㷏*|-3W+:bXhΏ_.յ4n5? |+ IiAmY۰Y>2| Gn}ƚ<-Ž݇ߍ -6+Koǿ9Ρ>m:{IDR]KrLBlKVv?jWA|9|Mi>v׉Aeg4ׇ KDTKՅ)_=k4O*|f~ Xi>G_^?]S~ g#߅-.<1x5!}ueNDyoG:|ww੢Ai+eýr-=ơu뚌;#CIԡ2AnmaZֽG{F|W>|/}?ƿ >xk3\Ž>SoL[*k> 4[=ڳyZ$--E~T<'_ S'~&_|1i_ |)Z|?־"-=3O;WP<+/jcNt5kGK>m}?a]Q=7OӾ<-3V9J7vv]EߑDŗ MYth> WG~x/쵯DO-GDwſ c j|2o[y_^޶Dr& ."9I$|A`OMKxxI|Eox_/^y<5u[l,[֫M.mZyXQJ#kT=ߊɧ)ȏw/(%#>$Nw}GK񆍭c\עm3O,Un![Ek~]ߍnPo h ӭtR|5u={VKMyyt.:} N#?i/oF6NF+pj<7'[x]x\U-Kӯ)HRjay~GQ|m>$+P5>|Gx-{>34k}D5mu^{'L ]š%^߉u Gh#+ӿ7j SC<O׏ ]|[t:6oSѮ~6^|K.gZ?$+7mkwxPJ OSCǁ j;tTN-] %~oIi{=V{m:.G)xS?=}~#y]6^<7ȭ-5#g<&!>|h𷉬MXk;%$?#|F-Y$l,myh%ߗq~V&tZ$uơfUzw^7"(<1on&_iw?hxd|6HJ Ѯ3m}w_kG-z_>4mmw|_+:&J-(^i$^~ό,3_gk~.w DKB ~+ox;J5NJ1|BCcGҮ7cnWPW- B??Oo ᎓t[:%? |3v kz׈>toxZih~ŚX]Q>&'.QiwI? K9Gj+Zxgᗏ>00Gզk}2Śdžt;o\j>-c>V$E/bXɣh|Ԟцut]~(a;Vn@x૱ćWSyO-/?ho X~ = >#Dӣ.⏉:GZiϭ5[7t//Bu["ψ[;Wԕ=C5tlo55C=#T ?U񟉼#-7Pw:6ڽ亼BE^Yrla?[AYfxtkFڅ؅u8ɷIYy%OwGkniK[nr'- Ětz}ڕv酚>[|o㟇񇉼SMISDv|_uD<^/tέ}s [DVZu.Y+ E#>𾩫7o^  }*DӮ~x?*8բ֎3em4Dx^C5mCTK[cC? }xW~.5[ᖷ"–jE,:>.|;$ u| >|Wk:7?@#\ӯ5:î߇5 ~zG_<}?At+ / {JG?yqht]>Zɡ-<ǫvO K{_j?h_+ڮ;I6i nhrhtIxtmkω/T'.]|m'zoEU0x+ڶz5UlYGmv\iL~g\_<|^_tM| xڎxA4]x|WlxZuymcIhۆV[X~ڎx[.?/|Ft? n6ZuԵkMKg5[;]>==[Z=voc:tx:u>1փhwo|AhZa~.ߣxm[E`(45'ӵ xL}@5w&n|n!ICyJ|c|?<]?I?W#K7+ }_^-ޟx^ >"i6 xjZ"I1Ѭ>I,^[_4aq0"GOG&K>0񮛨ho|)hRI%k Bv(^~^Ӭ4{ KlgkgxkZ _:Gծt^el5?[ ֓/-$ׯ{m7[B-o+'kQqi^xe5KOTou cEooOk^? hz-ŽtևmKu~ڎx[.?/|Ft? n6ZuԵkMKg5[;]>==[Z=voc:tx:u>1փhwo|AhZa~.ߣxm[E`' xL}@5u&n|c{\Z?U|. 81rzV5Q}-8*?_K௉^7ᳳԖӯэi^mH煦#ͤÌhJlƗo~^~]6IM~ ѴM/^,#hhL[-C^u/ o,zK mޞ[e}JOƞ.l.|K?wq[Om[uj:fI(t˭:Tm]zytë =_>t4Ɓ? U[}/t=CH}FO<K+6vmy>C&G#/T!_O.?#Ԯw:jZZͧ\Ϫ|RQXIC4nς*ѴTe't K㶱qReV 4KWKP^&zwx׏τ43O~|Kq~"h˛L\i⺴g6,7xн4[uן{]Iƞ8ğO|7>Eƥd|L4)G+S52Iu+v4X |E."@uرI<}"JIj/ny"i:Tv:]j7zѷN4 ~x{P~*ny|MCK4it1sr!m[| x T׵O o Xkk~wό.GiB)״'|)wyMIאj.Iwa]^7&}C Ы+%a5ٗ]_.)|C K_6wýG_Hń .&u 4K{6YE-u尣e~oWO}0FcKa__C_{x¿݆|+X//u]ᎽN񆷫Xea.,FMLKwbk['ox^_ZA>?ThW5xd:'O-cMY~-֍G<+]leo.xl_3!=x×4~uHxGLzFx~/~hW~-՟]vxt ?dǭƁ?' +|SLډe_ρt[&_o|7ı]^Oi57ۖmk3tSOmk [gDB!ETdh/j>hk-#: <7{~$dg!Ж6=9oi>?k .Z gIm}A}[ wgh,5Oj<7sg_x {|Gd~"C7cEZM_zgMut xD3KC//-~8ÿk^-ÿx/w+K}M֡\]FUI$wVizeI$խډnԾ5^ZMCk~/}GĿ <:w|xSB5aO/{&𶭩C6K+Z/; oۡH}Q(C{ڇ,ILTxjW6%|)qRZe Ֆxt悷j+GqoO⇂o+{Ms}3ijaGNj-|Du?"Úƻ}s-ΝxnMD+/>>zIH-/φ>3g'v^w-l{qIJ mړ}WZo_ş:\j=BqE$j/u[? fO^_6auؾ!{/^ZVM7I#5Io+!+//~[ϋh=wO'½+⯀A|Z⾯~ osC,|A#6i$amjn{֩ok?u~# j o>]_i3x;@aI<#=֙evoKI!C- Knh'mڊw|Ïg(G|Բ-D­w⦧{M>/ i׺3/4!o@6kY_xKm,zw~xgZ-V> ~.xA?_ xc㯇;.5%t&6t_MoC6_Vךվ7g!CL!?}_ $ӮSgKwzÏ~?h-\:妼Y]ZZz?jZ'|V_ $6,_X^CcJÚ5chZ@ZhygUmbX.4!io"#Z ֕vڥ"B.tٯ"oycYQu;YBi>ef7GOt]Sߋ,/|>xč+^|E/ v7z}>:Faݦq_kJ]!v|B7+gz}~:&͎ Atx?R Z;[BM;YӮu :]vBlQhO+\|=-#>Ӿ!vƿcMો-/;{ylOiMuJ.o+|λ {>`.-|#l#ψ~%xľ2״W:U=vF8Oγy&9|9]࿆7<<> ߂Q*soj^4? -E'S vXtxvK4 OBc|]V\zo,?&H=G~?m5W5 z Ruc_˪[YkP'<Eͥ`~du_,wxW߉-=BO/G=ݴO wI?< w}_xr{t[yoV,:t_xs?oNjZo_7^!}'V+qu}5 E k-͵֛voE8/QxN_wV_qY&: zCx{^nt ^UڳkzzsCO]g=6S[,w/xS׉m=BG?GV-ݴO iQ]j]9'"|  xNc_?~Þ x* Q>xzVm#[,5{S[|Qsw]mƛaPV:x;WG/-~,l0xǖ6?Kokw<}{OV+{cgxWzKx~YSO.h/jJ?~1O_Wß xR|?y>Im/=ŸkvZsöZ^jzJH|#|]gA7>ZE>Ԭc@o ^OLJp]GhZuN]R_A9ߏj.m-kM'Ż<j+h^~-xRt+/S-!2moMtx#wڥ'Nh7ڊ[|eƺ5k]FLIY&Լ9w7C0DKϵb]+2q>u5__pύO;Q~!( O`` p }L`oPyX?A~36n^}'ۊo+oN>( # t9o+ lc8ǧ=Mt1#p8:vc qӏn3h6~z``r00{v=I>Vyt6׷ja:~}FZ~zc|m=IH?\wUկ =83$c^ac<3Uկ B`5|~_^~捺<_8j:€ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (f?~oGZ%ÏP|P@P@!G?|@.;2kKkأm-RE4r[4l d B[mMx{ 8s#G# E{Kli9:}R+C5G9tB?^7N2z.Vߴ燿P_+oks`t?n C5G90:7T! sO v2sOO:0d #h Au¶<=Cꀺ__Vߴos8NQЂ֟ _g|69?|<1].y+oks`t?nBmMx{ u¶<=Cꀺ_?[~Cnsz|t=At`>~\Nx{Clp@s֍@t?[~_suH.Vߴ燿P_+ohxxc1N)o?[~\Nxxc>j9'ÌN ].y+oksI*@֧`_?[~_su@]/i9.Vߴ燿P_+oks`t?n C4?<=ǧ ='GSK5d燺t<` AG=ht M5G90:7T[~]itgA8FB Z|}A燸3SZtmi9 C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i9.Vߴ燿P_+oks`t?n C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i9.Vߴ燿P_+oks`t?n C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i9.Vߴ燿P_+oks`t?n C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i9.Vߴ燿P_+oks`t?n C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i9.Vߴ燿P_+oks`t?n C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i9.Vߴ燿P_+oks`t?n C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i9.Vߴ燿P_+oks`t?n C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i9.Vߴ燿P_+oks`t?n C5G90:7T! mMx{ u¶<=Cꀺ_?[~_su@]/i79?}zAtmih k>⏏_mo [hIu?A\DvrI$i$qKn$앴 WKXB ( ( (f?~oGZ%ÏP|P@P@3~lGKwHOoGz Gᮝ<tψO Vw (ss)gMWG4p+>a|#?cᯅZ{o w_n'&Uz|I;G!u ÿa_eV{t֖(u.ý[OzR총OT?~&wvH %џZ: K`Kx͗/O/ |_u}?ş/<=kkrsYUn:/3 !ljKx_'u+?<1=FKm:W$zT^IKg0wqqej?uw_t E}OÚg ך]\|PN|-Pn4O|9}>׺- :mKyMGKſw^)*~+hw\NO_k];DN%׌UĶpW?>$=׿mm_-_.▱Oj6W@g?^k'_^Z5y~F^A4{Hxw^w~ռ%xkƯ RwG|Wt&uZ4tBI{<?h%|G~|$>1­DW>3.+Uŗ'^(uwh ?.Rh:`QI#w_7|—6IjvgjXM75m28y g4m#oxYTf:?3ῄ67tO Υa=gӵ YIm_j?>#eZN]>&H.;?#7cմtc?}KCRxWҥOk #WlWԿ _qx\[YOKkcMW^ ~h>~A]j=KAˈE66ھmJ6p7G+>/&>W$ƚJn'ΏߎaՒMYtƖQHj.>ww>Wτw\6_5|,Ŀx_+=5<)AG֛a|1xqV4Ruhg{]W@oxׅi^-Tzo0MpHg?G5갭ğu햝&6I?&r>-ýcOw;_־)ҵ=Vo /߉cOK?&F 'R.2<~+|*>(5Ɨmi|8Mֻ756oz׃Ai2Ս>{J.mn5j:MԼ[fGᗅ x ھχ7\iZ|d)hkOO4;V &]VImvG/įk^x^i~-r%\XwG_?z^fߎ0VMZKдƖQHg9' 9''SIi!?x;Ζ5|I7&.? AYԬ> Gzv+)-^RxV-'C.Kxݕ%ƾ fx?]uGs~5EΟ&|mkSbHV^hʴH]u >_zsx_j?Ï~6O.5϶k6^ׅ,{MWGþ"~Ӿ'|έ-S^5j'B?{⾋5!5宓xG_RѠm0˥K߉~7G+>{!񏆾k'ﭿ9w_,:G%^~/H3G0ieYugAIw"e}V%i-'S$O];CDdng%x.#9Rj^,eA|K|]_OgË4k;/ܷ▋IiMkS4;Wh񾙨xm_O` G$d#` ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (1{|oqk4sƵO}jeĿ4R[#R!@P@P@Po?7-}Jr`?~lbx ( (l6O/ZH hב隗?Zόb|$e@PuX|sx ·1TУDԢ-֔z%܉-D!Mvm]t|5#T\ᗏuM>*X-VK 6 C].gOdI0ȫG,m$@O_K}s'O^<֭,"ҭux¾(mt.nam sJ;$XK(i&֛ O~#? |yZii6_j\wpjs<6N d@bvCC_S^)|s⏆'|55xwXNMؾ 4f/"y'lMC{\xTιxG:ׂx7WmoTn4gP즟A ޓNn,dC.k&6v:N11#?^i cP𧅵msþ'|5j~%"^!4}:\i#ͻhR;kg,?i5mo!o#Ծ xZmux ¾)-t.n m sJﯧKtcY/'prvmhhk ~x:/|Q>$񯆿) }#5Z\$mɷ|k"}/<[ZVuo#ލMƟvu? :tŌHemdݾ2BNG\81Ӄ02;dg!l`j𶭮xw OľK:^kߴK xwW{J8 (9 ;E}9|?7\|u\G%@khnY7=wPk$Yu2.ǣeduOZόc|$c 'U:+}M,8Mu>`"Kl6 yXI|5T>"|$ejOIf`k߳obOK?4 {᷃u_G3'z(tY#srmo*Kƾcdv{OElj ~5p5{WNͬiؾ 4oeȹI,.ҷA|Ew?>x%@<)|EZ~m;Ujږ5h~iQo4:RFLyXqA/~.ԿmV?dkz?Ň ~aDz yf@IP?_K}s'O^=֭4]cƞu[m. t5 sJ;$]9bCG>\c1~TxSڶ5??[/>{x[~4twV{V1mvl1Fͽm ׾.O4_᷀|IO g_wu?ھ:t_ukoOqq,lFb С'?Sjz >K. iĞ𬚭5&X񽞠Qz=γ{<5;5&3oԚE|')7&>|G?3ٛ{GM.Rk_K\O#O {Zii6Ǎ<_ꖺ\7Wi]O wwӥHP9a;y 6N45? U>(mx__wu?ڄھ CNKqq{mi.gx>g|>O -G|Sx+P:uoFO jO:u Ni@:TH$2鶲noCo!'o#p:Fq!lP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@zzCaM OZ})m/ԦHP@P@P@?w_CĿ1qҪu:>:€ ( (oٓlxvӧ$c]: {|ת<(|o#L ~FK;>!d_Ws_?,t&;hM,2ہi+=6bx~E7[5'Om+^xj_zOx ;$]MOXHö](o5 I+v:gB?? ^]௉76~(|uui&hkjtl5G{xֱ "м5-x5Muo|KЭ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (2šR?_*=}L ( ( (ٷOQcpT90u?6}1_k׌t-'ھ/ 꺧t2EtT~[4qIiF0BM6U G/^#k@Ӽ) |- CE/@fӭh<j!ҵX|s ·1TR-RZS1tZn r$[uBM> [>|2iŤ4|Q[ip\^AkUGy}}:ۤ5ykmMӎ1ʖ P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@zzCaM OZ})m/ԦHP@P@P@?w_CĿ1qҪu:>:€ ( (oMom=?{z8QkuA|9𬚖~.6V25v1dczO{^tMgQ׃ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (2šR?_*=}L ( ( (ٷOQcpT90u?6}1_|lwgx6><+ÃZIoxa=7guqo\kWZpߋ6_ĭ>? /g_ ~ F9K5{>>_z,-8}kT;V)&]ڧo x՗/N |u};_o>igi&G]3S)R#?V#X5-QXfOqk bRI"ž'7OUx~(>+|_Z_] xYl+vڎ gIK]o6Y.|Ef6KZД+01­DW[௉W_xMï?Oޗw4kI-= I>/Po7>oXG )!-G@YO>69/H|S .Ԯ%~+|*~(5|oq.Znjφuπj2֟g~|P3wVֳs CUPO?=?7x/^{kV?]QEj1xbR$Ӭu=$xXy84i%׏|t9Q~%G+/>~ֲ{RUxgV|Mjzm5-|eox=֘Z Kџd.X/.-sOе zi,t$RGZ-kMK/izƁfqpZiGqx,ȲͲ2"d| fC{>!Q׫cxOX|>~K}ž+Ώ./U)%ʿ: _>esxg3~'Q؏1y;?k/jhCe"ҟD[PmcJ6Ws⏂<7IAJ$s&(jv^;|aem"g:ηa?bдA*CӼO8G^i^όwNqik74Vtg⯄uE-˧G94p^Yx&6&''kko5_ |txW/g/yii?_kzDgu^/ZLk"UMM⎋ _ \_?ۿ"k^)lտ|s@,~'ƖW$>mmc-O.➯o6WڏE/=ߍ-#' oO蚛ZxLne _ muDR>/Q`|kዟx+gw$^!-k?;ߍ/tEj>sKKI[ Yo~ƾ fE "S ٫  "[z\Wb%ե|WٝYo~+_|KxŸ CA%|wgƤ&!ۯ|/RZt'D[OԼ]kj)/A>4>/|cSĚÿ?|1oOKgIr xGOχ~9u{_/t~1Ӯ-;J]V;+VD'>&Z6aROEGDoiKX<9_xG|;N:E#Ԓw+S?Byl1Q:|~5-O>|7&.? AYl> Gzv+)-^zL^IKgi0kOqz> GZ:*|\\iկb֥U]o x2_ Av:_?ewxo3~(OX1y;?t o C֭Դ츄ZShzj F!4oo|#?>cᯅz{_|i>_īNQH&ߝLL;6?cI=Ժj)U#E¿þ- c O|g/~#oM|և8E֫ki ZXj)[C迉:ύ4O7<]kڎ4x=r&|K|A}ō#ׄuXVOdNMio$9E{}ޱGZ';ͯ_Z7oOı'%Ku?yowu)|@~[ EYc@~?j|S hcKi>&]x_R~=x 4xj|=mi6ֵ}wK`QG^-eA|Km_NgÛٮ4]׾2XxvX4[qwp֧miatKԮNl+$;\|oW} OL|5_MIG| ִ<|OLl<1o>F{ OgkN} %bt=Y4GTvkdK':,?/|1s?o>"~)kƋ?M}iōΑ&!"]&*cs#Ow#?> >[Ӽd&־.>ٮjx~F^@V˱5OO ;w <5YUz|Jny ÿ]e՞MLj)$u.ý[OzR총OT?~&wvH %џZ: K`Kx͗/O/ |_u}?ş/<=kkrf>.ؽi" nz~CgoziiR}cuk,ouOiZ]Ɖwi"Ҧ\ӭo<5෽XAӤDV{|] ߅? A~ ? ~]¼_?Kz/_+f߲Ey 6[\?~sž5ߝ[7ZuCsaxڗu Ny wis mdݺ2CRkNW߳ou_|kX >|_oևCu[Aּ K%>uS]xq,ɥV5kxS5OC_?j:>{c~4{^_+㶼XފyX쭲 E/o[=rdUđm׃;g$wћm`FƇGzcMCO?o'--&X+VKM5*ﯧKtcY/'prii[d? 6> = Dx//OL'$ؾL??ifyXk? ~x],mQ^";_Wմ[IztI ]ӄhuB9"dHm /^O#hf״c|7n G?$_abcWԻ<]>fG#o.|+OV vA^ [sZ_uOCxCP ^9-@gTyZNma[Ɵ> |G >"|%ejOIS/׺?uGkg)׾.Ot_x᷀|G_ g;{Qc3jO.ii_aծn/m<^EM6 ӱCR'kY _xRoczԗQӣ\madSo+m+l1BD߀M7Mg,M<}2Ծ xoOִkzM_ Gm\IbWSNmar[c4 ^ |T%I]oNvN%dl<{\7Ǎҍů{f)y4kvkHn п|)]h8?oƞs;@&_jtWukos<ŶWg&@J ׾.O4_x᷀|GO g;{QcjO.ii_aծ./m<^EO]ym+lWSo6 xwS/+_KۧI /7~ټVK#x=bx{OզWvm%Ɖ xHMVO~W4lt--R=)C4-@`SԺ/ORzw?v/+&ajIn-|_}ws1HɣXZDcwp]n{OEljx^4w#/׼_>6bPӥҾë\\^}x-96Vw|Q ?xk;PWuCNJqq{mi.gxi[dŸ {᷀uX?lx;ú&__:OI~/%>m6&XsS=r'+k6]^^TVkw7WM?,4im-㴑h|ct:U v=W\uH;DRf5}GPMG4i$[]*0$E;w>OQF=( CžյUށxQsa6uiX᳖6Pӷcȯg/돎+İ<1 u!ڭֿ>&:ާ x$K.TexwR)kY u_xRoczԓPӣ\madSo+ 6AO#O yZii6Ǎ<_j\7Wjs<6__N e@r )]qo|DIǺզkx^NmsuyiWWGw}:ۤ5w g,^[hzhˌt1ʖ 05 x[V<;}Wz_k%CGӯu o&6uJ8͔(9 ;yŸ)F6Usk7J3xAcogFygfm^Ox6Lj$xV P-x-{:? ^i>xOy{3orhɥ^kkiO}R\ᗏuM>-&X񧀼+RK 6P4ᰎtI5w g,'o!&։ƆwS Ok;kPW?|Ai_aծ./m<^EO]|/~sžujVn? ީ-TigSΡM>'JX]6M(m$u۠A=y- CžյUցxPѴsikuiiEkw9aL1SoE J#? bhGIc4F-u+ Xwkhǧ˩MaڟI-l 4 ( ( ( ( ( ( ( ( ( ( (  h_zKh~hf2B ( ( (f?~oGZ%ÏP|P@P@3~l^MnG'v|m㧏~ԗpk O|[wsk)o6Wⶃ}_̟t 4KmK{۽51e>: o|%E~*_\[\iկF/­Bozž&7R_iJlZόO<]kºtZxz.|M|BmOG5갭ğڱݥsIkpI3{▋oCx>W/k⏈W/hߤjsIKSH[3+d/t}>_|?/O?eRR1c+O4_k3|*.|nm@?}t (Z8Ǿ>p }1 z?@A@P@OP1cP@t?:{`cӀI&AP@P@P@PL~}y@lP@P@P@={y>( ( ( ( ( ( ( ( ( ( ( ( ( (/YH)O-%њ ( ( (>gNkxF??UCSgXP@P@|cɲ|]zOE5LDZ8<ӃCKƃxWVx2e|/?@@ҏ<-+ Zi>n;i`׿g)?5io 4Z '6k[F%kM{GD~4pxwWFM?غ5-Z$mɷ]nO^&YkEItJ4- M]Ot}IӴ+J-/N ӭam8QWj@_A@98zQg@t?39#h# rxɷ61x:6n8Ock.g{n[ljKEC$ssq&jea4xLoimiatKԮN GmSԼYjMGᏆ> ⯇7i5v>6>>_Z,-nn[ԴGLJ .|Gm__xh^,ai3'Ÿ\xt֤U}ukcnx|\O *C"d3}Mx%v-GJK=~(/D_6G~8LWGo?Fx`g1i|J&cm ,hͣXx/4,G,> S w)×^WcjWkK~??j?|Q h_v7J vo cg:cW^׵[xkOÿ>(C[kY!*C~?Ǩ~'O]ZI?g.(ТյD1mii:Tlr˨{ſ}#ᕗ ~ʟ?kY|=i*_3+X>k =6O?ͲEom7BoL-orcϋ2]|Oo,um{N|e}ZOZ=|i4"ZWE_KZ]M`e$?kC<?i ~%|Gо$_ K$&>wXxK <eYug}MV)E#C~5_U&xwP4=w@MG]HKm #QhdYf m>d⎋ o _?ۿ!~~(|kտu@,>}I񥿇mOGU *͔_[oZ|2VBׇ?^(lGu_^5WZ~!S?iOk-6J+|kG KPCxkuxy9Ow7EŸC5;/0|e3N[^ҟ1\hZ[e'S~|#կ4R/XxGg;'M4OT5hfy^~W:ߍ"^eӣ/,^]dvG|H] x>_zo_j?~4/?Kjmi3 -|&mmJ*˦DGE.|eٟxC_G{~6jؾ9 i?MK[}ž,Α./]&l+%eOtH?h_b6~ 5x[OG7f4_i4o|ioas_HV^.fv %eĿ|Q_-? |7Ծ fg%wgƥh|W׀u𾝢?c.xHtMPj~.5{xOx=^TPtˍJԵMXO귚aUXeuyt& /+bV;~kCU_KOW7|BޫG~o"ƖtMM<_}\Ͷ)EYt߈(~о5πl3/k⟈|o[: "|iooxSşb%ե|W٭dP⎉ _ \_?ۿ"k~)lՆ?cO-=l.|+⿱\+ldS㯊> GOء_>;3ROBcRhX׀|A?>tO )tsĺFj^.5tk1MtxŸ>39|<#Y'gÿ ^νx^ A.lk+l{o"xm-W'l"# Zxz74_/Kkj'OEuOIg;G|<[_lz (>?x;֖uIo &HZ|l|DGE.|gٟ^|#s?=ߍz|sI~/Zz>X$դ:WZ[Cn~$|I}ںDž[ [Ÿ|[/=_9ռmZ{Z{Y'_kzwDg5^/ZHkJyv=㦩#—iA<|N7[Zk'5Kzơ|=_EI<#Gh_le[ %wo?hA=_ ^\௉W_xMï?Oޗw4kIZz E$~2sw|މOP \^]|'O}NI|!Z|D7Z3KW<z^xilt5/|Y%|QeᏃ>ck.g{n[ljKEC$ssq&jea4xLo抏_FjS$( ( ( mS;E_8ULO͟LWa@P@7&wO>kt}3?fS|;k֏Cխ,o-4ִ,n!/ͦO o Qks6+`NBj𶭮xw SľK:ߴM?]sVD:q])a71Fͽ(6r:?Znjb$e@Ja֣ -O@Uķ:T_izZ\lm,6x Ɨ~1GxOy{3orhɥ^kkiO}R\ᗏuM>-&X񧀼+RK 6P4ᰎtI5w g,'o!&։ƆwS Ok;kPW?|Ai_aծ./m<^EO]|/~sžujVn? ީ-TigSΡM>'JX]6M(m$u:c tqҐ(jNiziΉ^ZƏY:^j6ZjvG$V A> xT&O |CxIvx º/Lo47z,iwG񪵔'jMu ߅? A~ ? ~]¼_?Kz/_+f߲Ey 6[\?~sž5ߝ[7ZuCsaxڗu Ny wis mdݺ2CRkNW߳ou_|kX >|_oևCu[Aּ K%>uS]xq,ɥV5kxS5OC_?j:>{c~4{^_+㶼XފyX쭲 E/o[=rdUđm׃;g$wћm`FƇGzcMCO?o'--&X+VKM5*ﯧKtcY/'prii[d? 6> = Dx//OL'$ؾL??ifyXk? ~x],mQ^";_Wմ[IztI ]ӄhuB9"dHm /^O#hf״c|7n G?$_abcWԻ<]>fG#o.|+OV vA^ [sZ_uOCxCP ^9-@gTyZNma[Ɵ> |G >"|%ejOIS/׺?uGkg)׾.Ot_x᷀|G_ g;{Qc3jO.ii_aծn/m<^EM6 ӱCR'kY _xRoczԗQӣ\madSo+m+l1BD߀M7Mg,M<}2Ծ xoOִkzM_ Gm\IbWSNmar[c4 ^ |T%I]oNvN%dl<{\7Ǎҍů{f)y4kvkHn п|)]h8?oƞs;@&_jtWukos<ŶWg&@J ׾.O4_x᷀|GO g;{QcjO.ii_aծ./m<^EO]ym+lWSo6 xwS/+_KۧI /7~ټVK#x=bx{OզWvm%Ɖ xHMVO~W4lt--R=)C4-@`SԺ/ORzw?v/+&ajIn-|_}ws1HɣXZDcwp]n{OEljx^4w#/׼_>6bPӥҾë\\^}x-96Vw|Q ?xk;PWuCNJqq{mi.gxi[dŸ {᷀uX?lx;ú&__:OI~/%>m6&XsS=r'+k6]^^TVkw7WM?,4im-㴑h|ct:U v=W\uH;DRf5}GPMG4i$[]*0$E;w>OQF=( CžյUށxPsúͻhR;ke,?i5mo!o#^S>(|o'<5<; [>|2iŤ4|SZp\^Au<6_N"Ʋ^NCL$;X.Wt_᷀|I_ g; ~5;)uwS ˦ɻ}e {tq>ǯ4xS5?C[/6{x[~4wZO->(n,?i5mAi[dhGIc4F-u+ Xwkhǧ˩MaڟI-l MP@P@P@P@P@P@P@P@P@P@P@P@e?4/=iĿ4Tz3R!@P@P@Po?7-}Jr`?~lbx ( (l6O/ZHC[}@s>6ǿ mjK5ZmO -;W5TV[A/OzxPφV~s%׌WSS IYi\O2_GO7 |٢t?/}.X-Ů|4#NZVuig=\YKO/tmzwĭgz'ſU[.]GA:_-&>!Zhaō#τuXVOXNm9Y$|DKEmዟX+gp5x\Gs ׫z{o|iooxObҤե|WYEY1e>: xG񎱥5φQ\xiYVuksoz_ O{YF>Ǧx8y$$ z?@A@P@?@᎟מ@P@?@᎟מ@P 큏N$̚6A@P@P@P@Ph ( ( ( ( ( ( ( ( ( ( ( ( ( (/YH)O-%њ ( ( (>gNkxF??UCSgXP@P@|cɲ|]zOE5LDZ8<ӃCKƃxWVx2e|/?@@ҏ<-+ Zi>n;i`׿g)?5io 4Z '6k[F%kM{GD~4pxwWFM?غ5-Z$mɷ]nO^&YkEItJ4- M]Ot}IӴ+J-/N ӭam8QWj@_A@?Ool@m88 p@s֍C(~=cr9 ÏSh2ON9R'rZ?@( (:€ ( (oM֟})d\ZgkڮuA|9𬚕/|tZ\[x: \+=t_kKi>!{^tMcQ׃ C? kK}W$`|7>&.? |@Yl> Gzv)-^RYxV-'C.Kx8ǘzGᶝÝkxSG{j'ӿh_xF,uß~ aϦ:l<;U|{[c|#%ûwY[yOj;xU'/s፮\BVvYjZ*b[ Ev8+~߶CU_KKX'o|B櫠j?ß|/5l௄/u?CEԚhxl|?ck#o JGt>$xR<;'3xKT׍dq;~X#o+CXZ{Z:uH- s [K!$g|#?kVx^+s*~^ u;4s_uV{40KOAx/[ߎwo^> K_֞+u=RO ?߉#O4KOFF j^'PZ; -.ϋ6_Ľ>o? 1g? ~ |E|8L|cx>)hhznn$ֵ; CL},&隍牁Z6l}UFG9 l| ִ<|OLy xĺ'{mgQ3ӯ{2xV-'C.Mx>E$p_1f~: o|'Ch~*OTr swV*͊ZVwkW? k9)`-|z6ᗂ߄Z=t~3>`4>-w j:Ҽ[eo<`.G#6k^a[?;-:M5l~L|[-+߇z hw6|Scjz >_k?o,M֌O!t]el5dyLVU|QO<5k .ͤpvo km}K1;xTe(|= \kZ߄t!-Ex͗/O/ };şofҴ w^aiS`oZivM/R9TsƏ>__/ ?1½D?&ZJ. ÿCaaoi-=O_OtS jHz?CO|Ki :OK|Y񽞁[xT#m^|Jf-/O۬]xqiCYj:^E?~Yc t]orTo j 'ᾡxMOݝDžAq]]iRw~,~_(3|17WӼWk/@5v7~)hkSOl[K ^wpVJ;j.ϋV_Ľ:o? |1o? ~ E|9LcR`osspޥ:eP]9tNĦJ;l2 ?[GOW `Դ?G cI>-ŭ5K-'«_t_ x= QVZHkO~i~w,?K<=gQ j:)&-ud!k,[j?BR\G ";f2|o>~xZ}^yBηZkZτ4!1VZ?=C?W|'zZM=Xv'FǮ%mKLNKbeѤ^?Ac?]G- cTAZKOUe៉Zƹ}5_Y|8dmz({{ixZak|.F|Yyckp/ ?B4+O,ҼB-xr]Jm>k{)!]u؈>7HK+>'&U:_'\௉7?Eÿ:z^g /˫=:hJ)]j+𶑮w4/ľ/KúY'j:#E]kmIಎoC"6fЋom'?tX?h_x^6~ [GG;^?aO-;m xW:6ᗂτz=t~2Fb'Z+_ψ/>xM񅕶_Nt/ w:Z)B-N1:mƉjzGsD4{ZEi.uye؛$<Gğxcž|)S }_w{QH!}2]SkOm5xl?ak3mhQV]4=7':,/|1s/~n"|_AB>WH!CZߊ~#=w5aOX>~K[ W:Dot3Y+-%⏂ioSᾥ~ h7_zo_j?~4/?Kjmi3 -|&mmJ*˦DGE.|eٟxC_G{~6jؾ9 i?MK[}ž,Α./]&l+%eOtH?h_b6~ 5x[OG7f4_i4o|ioas_HV^.fv %eį|Q_-? |7~ f"!@LJni𾛢`YKk%4Um?RuO!Ӥ]Okǟ!ſ=/'&x?>'>WMux~oNH* uXWSd;Y[c|gĞohڽIN3`N9DlxGL /Ɵ '/|/7w^Yr&XiRx^&,zuOtKDP@P@P@P@P@P@P@P@P@P@P@Ǥ?пҖQJdP@P@P@3Ϳ~'f>.{xzߦvcMn^gz1ױ8h=z@@X_'fS|;k֏Cխ,o-4ִ,n!/ͦO o Qks6+`NBj𶭮xw OľK:^ߴM?]wV{V1mvl1Fͽm u/WITK^:ZtԓPӢӴ"Ke#k&}/\X]eޓa=+K7]vMaźmڥga%#GYa)m>f?|K7/_\x@2c7pn` k/>k" Zxž(@wih?1{cei>#ͽM&){5^?_K}s'O^=֭4]cƞuK]. 4}C\Ү;$XK(iZ'kŸ*N|6)ּ[7kzSqݝO:e4n*sqc$tY7o:qt H[<-k>S/ך煿^>mCұ;{g,?i5mo!oNǑ_~_|Wa?IbΥ/ x-SAK8=N?]iO&G%ȹ-!@Vm+lƟ |G-ψ >xZmux ¾(-.n!`5*xl#mAy!wur? j֚i5mKNt ]Ŀfj9cdd6?ؿEOF ;¾m`h>Iފ5oܛ}>ʷq/a]#^S>(|q|5<; {Scj3kG/5 :[+:n".gx#16 _|*]u-&X񧀼+VK6 C\Ү!uIk%tX6c`cmzhZaͅhfo%jz}rA}auk,o.5xWV~n.sqc$ym[PjMit [>|2iŤ4|Sv7Wi]\Can,k%4Y<+lBv<^5߆ևG=;%) Ӥɇ?G?b^уo+muxOkş à>^"wuv? zxi5}ON_k_ n|K{{.~Cj~{G%,/+IѢэ>ktxO yZi6Ǎ<oj]miW7XGw}{2ۤy; 3/bSkM??K}s?><֭,"mu ¾(-..`5&xl#mAy1ᖫ/V/^b4C5֒:tz}Z\ -۬, m`MmV6/>[HC,߃t?'Z փoI_ Ah+,Z{ͼ5.Ukl{f} ]"7Og .uq$v{<56';h2¿N#'ORz >K. iľ𬚭u&X񽖠Q}ރs"&biv? S'mx_x^wuhھ CNJqq{mY.gxi[dŸ)F6iG|_x;ú4}Bm_I :[+:٧ȹI+o Mmj ~x?`><}%+ t?%!Oؼ7`[ib|ǬAioJNґ<oiJ= ƙ G#|fe > |T%I]oNvN%dl<{\7Ǎҍů{f)y4kvkHn п|)]h8?oƞs;@&_jtWukos<ŶWg&@J ׾.O4_x᷀|GO g;{QcjO.ii_aծ./m<^EO]ym+lWSo|6 xwS?+_KۧI /7~ټVK.~ |<'^|$ewxB| ³ -pn3J])冑v=%v1crMwzN\jz~aauǪVvvΩhjZF F $kYۆD'b ~CҀcB>(K?ĩ1k\fhoVF{]F=>]Jk-4$dOk`mam67P@P@P@P@P@P@P@P@P@P@P@P@zzCaM OZ})m/ԦHP@P@P@?w_CĿ1qҪu:>:€ ( (oM|{|ת:sU4 ƛaj:_>M_^;ˉo7M7JFMX}V7ӧW%im=@Ph ((1ր( ( ( :~Ҁ?ϯ=h ( :~Ҁ?ϯ=h (( AH4m ( ( ( ( ( ( ( ( ( ( ( ( ( ( (2šR?_*=}L ( ( (ٷOQcpT90u?6}1_gNkxF??UCSgXP@P@|c1]1NwHMnת7,=zUx[Uk~hZa/Yj1i>!k 3X3or }!>$>4>-~[j>|+h_^.L:o0Msm' hF/ X7xKTn_R+ӄ~@co6x_%n?>_Z,=|9ZԬ56{4}/S(;.]8p:2Z[XK|du;w7CGh1[ypU҆o 4~EEh6y~<{xz*Cφ/ ~ ~.Yx[a<vxGow~nt+kEGp6yڟ|}z*~8ׅ!ٓgCgo5[uz]m-l`~V[uGׁw 9s[|z| ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( :{cҀ0|A}eiݭږXq 3j{QWNq)aąBB̪[GKEG7P@P@P@|6c?/&Ϧ+现 (1?ڋZ>Gg旧iNfWskVóci&(bI%uyhו,T\{|[xǚզkx55[]. 6P<3sq ww%H8Pұc.tS_ A^kp4:֡7m>=Rǚ-ƫw|2'uֻMmI#^A&$S+^Ld\ ;b[zO/$N h;'z7e,q}>ʷϓˍ|̗uo)G/|3Z/5_WLjoU6R֡hS\hWlҴe$ymoC+_)Zyp,H5QqC ;Fq^4Y?>oD=kV|ZMO(-t{4}C\y}{:ۤy>d;m Qx <-?o;k Scj3kO75V5^MMdv:/Ia5m{ͪVo $V]u좟Mn3 ( ( (>gNkxF??UCSgXP@P@Ol~=ǡdAF ӧ8?Zy!% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}_r/~t|܀?:>K@% ΏG}miGCN:QD)( ( ( mS;E_8ULO͟LWa@P@VV7<ϲZ\{o%Sipzc$oAr[WA"o|a +lh1l{c-oP]P_=??O1ecS8|'Xev4{~a /EԺrp 6Bhz' +8#>#=[[<`P_?e*?Ga',Dq(6r[#g7OYvq[]]sy5z< / Q@MifOsd{])Oc ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (ٷOQcpT90u?6}1_,/|pqpig #_hj"K*M8Zw^we۠m ]F_,o@O?Zt_|M/\gM x]b4H34T/( =P~lvGxG .5-kJψ|5 Q6EtxQ_Zj>&OCm˓S^j5{8Ln^ cIpA_xY|C]|z⟅:jZ薚Q!LЩYŵ˧kw.MޙG|s<F][>,Bo ؍&R-M֞Rb&WKQm׻}M-~>k߅>/za=:o x~|)j^hu++.5.4M[PI]#o[#ywSGѭ>,u+y /?i_>.lnmP|77Z_帙C^m~-GCq@=%@?~U O+XOf}[P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@Po?7-}Jr`?~lbx ( (89sq40[[ڤM"o YNM,BGF3;> rr%CZ5Ě?w^|G}C&uOkZG֬gu;?X0yW3tn5E$WJw]vh>-5xVWWha_7wN 3woiqgVVm$noŏ N4 /gzwx ^uO̩b^'i]ϻ4ԛ꼾oD|/|8;K~> 5׊|;= |5h(}b[F<~!MCmpҴ`Ӗ]t{;A+jb?u>,@-W3y}Pg΀ڥ*)jѻۥoOF>&&W.ZIn>87YYKKXsiZuI 2 Ux?׶|%mi GԾ̚ޙxTҵ=R]麾4v~Eh]%R74v 2ί"-PN Ffw^okӼ'-+EΛk]E4$[ó^X}sMx~%;?ge٫?]>}-<kcI/f?O߅QFLveo dtw<5~5k)%՚-{oՃ;M"֕[rJ<+eu-s:ΰgwG$ +dխm#t{E&zhç?cښvtst#s ^UwֶVo{dwtGYůRP@=|4.  `~[eY]Mһvk%5K[jk6!`u9粥kiMdV]𷧪zOl~=0?>=OQ4mM5{)7hɿh;W/KrHtnFFW~s%e+_M֯(K>aHA@<8z@A8b?@ۧӭP@duPs9pT2H?wzlLcpPsps~@O~#:qЌu3gpLj~4ï|mKȬ'g>F=ON´V`O? 3 socpQq?A}:MHce3€W]~df›24hwa#7-_~8FW}@8ω) pR8灓vudq_ٛ oDS?@<#ODA zRv_Gfo))#Lvvj? 3MO_GfqFo"4m CF{>6> x~Qh:gSk{њFc~Hp8g|-S/xO*=/f 6-x> k[Z$x cOze0v:=?u2mpd!Gԥd_Fc*0 GQ|&д_ӼEAJ𮝧x[JnhGZZIqOqn-+o:V%@i_Myw4DK>xmF"mc6j:u-vkr0gݧv0O4{wY6s;q'dydr=tAϿ< G=x8Sh?L}(x~ ޘNvAzc9289 A}FA=Gp ( ( ( ( ( ( ( ( ( ( (f?~oGZ%ÏP|P@#֛][ko ||Ii3I[i7~&K?KY|A/j,VV d&{n/s1|36<)8b(YXʔCRy7*ѣc<:R  ҧ%FU ITҊTէ8{u<i|!yU~;~0kچ]OğqU=Sx-+z}ƥk& \KUx, ~-K3RCyNW}_+ԡb<=?ʸ׆B]W5X|F4+ԅvu%%+޵ZNVk1mKFsJrek~[[%?diWo /M-yg/g ŋBCúmƳiIm4~36<ڇq dy&afUqkek-813.Q'_|\7$~6UM \#A˭ {XScXDž(yeGԲ /yYPN?ٹ69Tq3);\7Wt+P>mCLӯ5MK[y{on1 SMAFW2Gü-gPJ9nܧFzxZGz00Գjiҫ<-HΥ5N5eӛpUyTQWVNN-:[|f,>!x[K93nqPV^=e[]r]]hv:Σ%Lr>N1\L=WTLCguL.'ΞON6ecT8?7WSVԬ-Zn9IJjI7(Ɵ_fc2hW-?h9|G?xOg]| ߆vZѵ{Eڮ,ֵ>Eـͫ_Mɬ|Kf /+3l} s ,XJ0G=%K:+s5p3Jx!SبrWr==HSRP挣969E~Tpg|TI\x'>/~{:߅ur_?G4~KS`Gb\\ Vp d0ϳҖk' Lˇ\Ki[)hb?pIeU, ^E尥*BjF﹥(+a”`×^T\o+_'.MO?뺷<7[_k> m @߷Νu{zt]kZj_c?9.Sɰ|\=¸T٦&nwѯc?gFԫG6 ^Zi}^j5**Fu8rŅ}d9yy~th~)?ѬOxZ':現xK[|+;_%Xi>)U.ѬlɫKtdS_y.O&Oe| r|!5)IJWO? Kv ]5 x^O*ËTM?  W:x^.ګqp 2쿟bN>UNl-,hօ|>gF#IӌBY9FRQOh8Z6jR'~$|E3Ú|G|M#^.k)gb ;n +i>q][-B=x[iFa/-Me<~4dLֆ+ ˃= W+1O`if1Yy|!-\[J¤iBqi9+]TѸI+xh/k:<ïjjnYf/sOO 7|Mu =wQZ:6F֖]+ ׆rcL0yn$pL> $ g1s,VSR_*e,`%/UciC8U)RP,CRg(B,ez|ГOI[߈gŏ~/< 'í'X֬h?^Iiz _4ue GR~Đ>m0u$*o.$γw0yu^ _4+U<7qԱl"]:4i,ʖ[Vaj8(1e.Tt>i,Bwg nj ԣ̟=OgT>1|J=2Kxσ<[M37ďhë8,>7{ŨjڕЗDԯ5 0´ ,qG~F8RJN&I&]S K1rpx|/׭Q)IQVΕGNѥ F},s(r9^\ jxoމ_J1kkFt['ڭb l K^mX,2,+*+J8t?eZP.k%O/yr0|wRmE&[Re-nзm(G>cGxBşn_|2 hAk\>%>w:M:})V{v.h롟Έužk|"g%ZkZ֧o_i')OoHZxGB>|Ur;sBM~c&oءԞ=?C/xUQ7u|/1~j%fm]xfr/l|'=&w`a*K ~╺j_O _5Oo^/ٻ컥iڧ{}ZkK:dˤGA,ub:{bV>-e/j |yYΟ~6x7PJO SxHԾ*i-.Q-,cg퍫.JKv=~/M |6w//ֵ#Pֆƃ%ӬA GF=\]+{Kռ Ỹ!fx3xſl{BǨKrtFV}hX}fO%{ׁ+Y^(0nWC [_4/kˍ&ºoǚ|d|x_o0ʟ=o)~2ih^:W.;O⎡⫿:$\ZcIN}dW*]6/_ k q`|_?>72W_|> [CwgҵkK5cJJv={?@|feh6k w UF<:||5\}WVu-BkohH~Ij`l;=gӌHP hדGgkG45߂k o5/\/ x~oij^7ZjX<z|F|'=SM+3Cuv+t>U񶓦xGm~/A' Wǿ|)go>95/7ĭGR4m_=q^I.iPO+e#~=|?|1?xG>%6^֋?|@ҟoZxOoF}R+械 _ iPO,i萛i#麂 (16m^q^kI9敽Y#ٓ;f842}[  :duh|7hN?=WQԿiozt n>]V;M:)f^TҴ[Q3/W?o׆ c'?ITTѴCƟ x$h~4o|}hvO~}X>)T4MėWj{-J2[|5:ha> zǯRdӯмwi 4 [.]]/4v^qGPUKm.KR$M[mC6m*]6/_g |(i<㏄^6|IkG:{!?>':x<-ayƺ[xmFJZ [JJv{o j_MR4;xguK<[KY^.ҴOj:~DvV3ow/m;#z_j_s~Ο=mak¾/Þ mghj.w-ɷᴚDӴE_K0.a=sV~&uI5oU)uۯZnjf m/'Z $ $g[Tiƥkw~ w?ƥi_*xNjwR<]n(+J{h]-[|ǶR (Jkt;x|qRxzztzyZߔھßzW $?j/ؾ(x>6^'֮}aJWp]^Y[.ɫ(Ӄ7z-=}|~ H g~vq8폨? i=1Z#:TӾϣg [/x2ִ|qYoÚ-|E5 kDﴑp֏5%GEؤ&}+>x5Z|}y~7~|fDž|uB>xW|_ѬRx@`FM,>W~5|`|5@E_}.kk֋(>|HԮOPyxغږNj_#ĿW|7?ƞ 6ZV~vPx3x6᯾k~_߄q ~'\}kC& ǯ^xDi _Ƿeޅ4[ZY7clZJ\>HP@P@P@P@P@P@P@P@P@P@P@|6c?/&Ϧ+现 (~=K__=Od?=Z }O࿏NIд/ -ߧci=/ &y4[(%7rq?~'#NXNc]Ѥ3^:~3N^2Xg tΩba:VGUzyb]ؤ{\O-t O|Lø#Oczƣ-/S?qujXkWjoէm|LrN)*9o>Usb0as*ZX|^]^Lh6k6wOdо)|,ƚ⿅{[];Mrx_ mCTK[ղXԮ 9+.<|> 0har2| :zy;xVmį\#͊_7[ g3jT="X n xx֯OXuKF)b0u0E\iW(.vNQ$cj>__͏4&SCZ/;QԼGY_]kڿ]7QR$bic|.ϡR̸o'G4VFp>CʲF =,-,_C 4%)U: gZt*e6ܱ Qմh?5 kX 4k~1_<K{>$m_k7Z]Y=֍'"Ya|K<"!p}\:3<YnU9e8p[У[?=xJKQ%(ү rs#Srk׶+A7½& jdi/_|K ׺f}_Yn5K;]WS^k Fľc b{a 9s7G첬8 reVQT}_V  [1UAFpw Lfݥ;)OI4_~?tmC_ |3x_OR7+^#ֿm}gKU,]Sz7QO13dOb|~y™] Q{0aӞ,f\_lZ$<-gя%JgHTIޕu> S߇)|φBCgC\ԼGG[78ɧؾ_9Vm(8s*}[fah+CúWVz0^lDb*U:%tARWEUr&ۊ7l9o?Wko6oHt3OKfM?x@,9qig:V.KH-bK&P`8_=0~5fYo?#4}&\f7X\}j2VXJӝ^*u tJ;|wPwIbvύ^_<[Y>׾!jEjJX CY|Ae~&}p 4f>K2pfn.MN'9,˗6QG rZbVeԫ-(*eVƞ&DIket[σ? }?EX|!ߊ ҭ?Yj};~EWL]־GC8+.j%7,5/KCq>4Wj^2ol]/b=}\4*%%F\+7&-x#CgkNXƞ/*#Vu~IQFS4϶)QS$RQ(4to} ύ-|/_i^|WvR\5d%9<~07}?^t?_,OX_~%,2cƚ>Hg=m$[|O-_X^У։ெ:hޗ[x>[7PxO`MZ;=5Ͷ]=3 tfH4 Gš-/|@Żד k_:M?[N5hsgb+/ᮁL]j>Կix,[Z Տ-x#@֍um>4èsgco+x?__x/h^yX4 _i kz{ IxY@UΛ{gp}Ϋ5vkګj&V u /s~ӴI5 ot{{ΗG%:'sT:zvG uv 0"^Mg'3MnlcMW6xs/H/[we6mj-N?3QO KbJ1ݒ]E/7C+|/?L?7PKy^/Gp>u|Ϲ? n zK' Z?)T=nVo-Gg旧iNf]ͯiAc#5GH)$$gU^V-¤TiΗ_MRz/B;ஙq# Vm&*~)xῇkK./x+OcR|?i蚾ڬ:U͜2] +~gG ou|D$5O/6jn 7|Ijo:5}"O~|LK/@9kڒxvƗ7ؓT|a|O=/퍥ik#@S,[KwUU^zyK }i>Iu[>_s*|k{%"͗|?y.E)Xk~+)ݿ⏃}kM}xEҾ"xÞ l > X_t+-?EH~ߤi:E4+{HW_GukBסo/3/x?ہaB7»/Z{[mоiFwIxbݬ|CwooYO@?cz .㙯0G_?< xR }Wfoy&<' v:$L?4_k-xWMmeqc:.wquɭx#þF}Y]4zDo6?kk-<&7drÚŸk [Y|:gxK{Y[5_xM̞-}]{SS9ܻy{-:sm“[lu=_>&hz`_|UǾ/: k3g.SgSVI1Fs?e5={u@-Wh/LкO[&w@-TmQٵ&o?5kkς"͞qy~Z;Y@ѴxwRdռ7ƺ[]C^]7@u+i(kzr\Ho ĬaM ~ͷv'%k5faq֧64?PPA{m е6q:7q[|yw2ǝ0+֞E-J.H^6!xZom[jIl W<9WFJ{ iK$VkMӿS5Ӵ/h >)Ƴ@O#xW7QxAw]Z{?M^W~ǩw+} ?wt : *Lz߉n>/]qe>q7^jSAX1ۺۍB ?lt}BD/5c,׼aɤx]h͡{^mdZ0ƆU՚x/`t}K4{9/l,Yu]WTuIm-㵂]K\/.-gPhVPK4 kv_д}f΍ik Q}eЯ/[/F+F}XL{ <;H?>-_GD|hW?䥏폱yc?gWk0KAw1G}.(/p% }|܂b] Q_r >KAw1G}.(/p?^z{QD:3G}}^MenTlV`% }|܂cnO$k3u~l!P@?@1zדրP~c`86Ѐ2Ϡɣom~-<ΏG}_r/~t|܀?:>K@% ΏtJH玀g:QX[yXwǮ{:{)6}(Ep .Qw=L`'k1Qv6}(1Ep8稤@Or89ǦO~RP@P@P@P@P@P@P@P@P@P@P@P@P@P@P? A浤hqTm[RӖvúJc]3JqҪ 0u?6{,?)_M:?x/& ;((HA^XvRR7x!,/&ϥi>?&xZϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&Zϭ얟o~ch%ߘ>i>?&>oNkxF??UCSgXP@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@?w_CĿ1qҪu:>:€ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (f?~oGZ%ÏP|P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@|6c?/&Ϧ+现 ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (ٷOQcpT90u?6}1_wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ap@Fٮ"B[| wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χwx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>^ G:dg, '=%@ ~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>wx!€i߉Ox ?~'χ?Al(>dNkxF??UCSgXP@t?~QGǞ6)ǷMi Gƞ -g<--{J>?W1}Uǻn\?B_F!ӭf]Zjzu"Xe1 w:6BMrxہy9?.{l@N1ب`G?Lt<>]o|ojZ>&Ce[{5};lFq n\t1鎝HS9"vӧr1 (>@@3@r@z@2ƳljٟCA7ևY8:X\f}>w4k$j`?)^*g@`zP@`zP@`zP3q@ψi}xch^Ҵ_}Ai]AmS,K?i?5$?ho?-g)4&Q#IٵhtyltXIKIg,KX}Oۿ>,Ɵ_$n4A/wh4yo>vh]-`Ix3lKž8Ӿ5wKօMp1YB϶Ӊ|Ni}_.[?ueb.m=>g57WKuh`fnbzkflf^\O]ڏvW4ݭݝ嬯ͭݴImq ѼrE"FBK(]-`Qvcl6N+;+R{KKvhmD6du!Д7u R EofY+MF^涹JZ 1e ; 'Ӡj|i~lo4 CV.YkK H-JC}g!ݞ0pJʬş`]6 AU׍i}J]6G^W Guk+|-kUMu/^/m궞,vu9ͶikI Nexc(ϳ4;Ox(Qs]ҵϰ}/ƛy/E摿 P,k[\:ۥܬRWpVœKXzi G?0)LJyҁW؂jfI"q杔m j2g|40,/L]>,-~/ͯ(% >-x'5 c>oåGo4j`3+<jy>W5J>Ǭ^xgO^տi>9$0ڜ~yv]?kunb}>Ӷ_$MM[Wn/ᯏۯxF'Ed|?c⯉Zw!]=>$0M,lL$ OOk?)|LlOym1xvIeѵm|wյA[Pl˦ݥ_i/?U.S`4Z\xbTb,p+Fe/%a5|+t w~ d'ᦂ|9XDԭךtfgmR7*ʖrUgb]l.}2DAKh"kcK$$`x?7]ĺwZxV5mއs˯^0_xׁ'xGHѼ]hznWTȻ%J1Eax>gy=OEk kt{MC/\~ԭ5>|MCgcaEwes?k ĺkveg>̊v?u/ P@P@P@`c?:K֙aĭ+N?g[xϴ,!=/Ķttl̈́Yg*?tK j:>?bxvw~6&u]GSZ5t+tˇYtZZ WL Ŋ8=9ZlCּ1~ζzORћiKiysfn8fg²e]7㥑_l+ >!=N(ѴKooL|PϦ.  ?΍E=WLGؿƚ6v\ukM55V̓<6HQ W{%_jy音~~Uo ?CWLOΩz=evmkBYZyO5φ_l.uﭾ"4S񗋵ǫڏn<1{c}u ՕnVݠ^D=yI@F>N~xOdI=YOn @ʤo+P@P@P@SP}\ ( ( =n4 %ֿRxSIAgx7K&> Mt.4eW3$-iFyTc\|cW~6дϏY:ΟxUO^++;4PO[R޵{ )+z~=j|XӼQaxgMt5-olu%%S{Ӌۥ=W&.e$º8֌Hճ~~< ݺ Z5ImwZG2մ[x[>=↿+[>Z|px+DڱVv'.. kikam34Լk;_oxQMbZ5Mk7٥}CLռ?R"qZpc"vhz%?Mxݷ ^h^+U.? 4#2?hxbNLҬ-&_&VRZ_]<[ N7mx\xK/#_Lݑ{n/.Hf>G:%tm&OL~٦i_$b-VE3h.~"w tuإAͷxf~ ( ( ( v?u/ P@P@P@Bׯ ~'m<7%闉\hi.rۘZhDNC",5^L:Džuٓ si~Vi;j72Ҽ/YɧxBkKWWEEr_5~1оx~_qy_6.ܓ[]Y[2$Zܑ/kkwc!AxU}C&h׵>^l<9S@43Q6wzLWiOp8A$fX{py #|[ZmMn+Yya[K xjK=ueB]ѭmj+WN_x>*axƲ^zu4_?}zmXBDG|7>_\]x.Ş;_{em*Z{e5Ɖ\o y'2jt8Mx^O6֝'I@P@P@Po?7-}Jr`?~lbx ( 8߷:gmAOj6 :{}8~m OcN~l49##m{ LvG@E;_( ( (1?\gڹJi{|s|Qz [yX62=nm z{tǧ=pA `AtN `v8݆[h?@ ( ( (CKP@P@P@=cmY}hVOç|c'֍nӎugɣ`qNGb '>pO| `NF vL|Aq=(R ( ( (ٷOQcpT90u?6}1_rʖqu"4Xɣ`U? < #@2hxW?|CG?3¿ =]Ql?W4x<+̍ʾǟ_dhU? < #@2hxW?|CG?3¿ HnmC)ͫS68srr)iF2vI);sHikƏu[G|;kWhHL}ŞC4,P$ #<=\T!zjQ'uB(ܥ גRTQY6煮Ri5x_=l6qۯN5[%% +;-Zmګ4e+3<)A#N?e-OWr[6/n9;:~xZɮ|)gmI#x@^HLu; w-V6STbjJ&U8]Qqz*2*U6Z&*tl.Jt*niE,q,9urQ;t6#aJ`n/#UjH5_xkZVүv[7T"Ij`kO YoJ8kzS!RTjl^3ȝBVtT*5)[9-υPoo,ڃKqhtjq9[)DMU %LE,]X5`hƼKŸUnN:: Zm{m薶kxJz&7gN*=7@Ў8-momm$-/B=>v;ٻ˪KS7t I{ ƽO[I4ķ v;ift5MZHv-tap:aRISJ+JU =%iFtJ:NU.խ+[ɥ+ZRb)ťgt;~5aҼa :e1jwtMVn. -o#Sf֔%[Z 7N+N\ݓgk6ui:5n]fԚv[$y=~1<~̒^[eˮw~׻w$}mhKpO^%+wv\wn o~^_vvK[9xNσ)]"+m'U1oL젿㵻eXnfA"I(-ьO`R9JRp$qJN-h4JT*ԣ?s)r[sF\ֱ\2N7\ϕsW /Xxz֣}ֳ^!|PmxWB[4_ \l6תV"X ҥ s.m*ap8(ʾ+n\Qgk SK1pU)R^F&_RG`qA^^EylZIݥ^M.m̓J=՛VE5Y>{=0q8:B[.]cNZI%toNfZ]]vZ-Zw{k[9s:wOi7s[H{Xxm<6Fڝ'ڜjbgDQby&z8ʰj0эy"($ 7vhJt5UF^\*6M'5tTv?xÞ.|E&ZS [RƑkiix?E5vhw$H%gtX<#*+麕]j0ЧiNg:/էNQI*Wr^9JNPyzm)N +|M5vĺEAY^|AOWQos=ZVM>lMohPJ)Η-*ԫ]}ڔ*T+_W .Wg̣3' r=c(}񔑳XP@Pz{Q-CBtぎN3h'6GlӵEݷnwFw4[}_=7Ětk/Os ) Υy)Vtiq3O<E4er3*\G̰%8PYD^.Q9V"q)TU7UTu!F!59Մ#'*RlB5:]rM]GX7s{q3Gˁ <qmʯfWv{imj귻z2FMkwӧen?k_>g5#Ҵ0|+/^bBְahjEg^>W)ܿb+^`*\,'(P5RPZ1KxOl#}А18Rh A7=p?6z6.ҴwWizjzm~,Z=k;Iu+>_w({Fo|M6kڼ!],$v4_y^%״xI~zuCQzko;xZ3E4ځHx [ktcӜq}p'OA %On2l=1g^11GN6 ots:ր} a=7KdriG~"hVxgƚ= [ӡԠ/I Pθr@jv05/~~1P4V|"wl$x_>"K6VՓ_ YA'BWx_<048ukzo,|Yh? [ijjޛŢsgexוK~FM/Ծ"xPn]vڴV]/JѤ}N \j_:]q4,VBKd4|{ᯇr߶ƚ}Q@t4k=bwt'X㴿i&;fZj9N}>FN!=9\dBg<4xC|,|A/mu$>.ռ"~|TѾ%]߇x.j+ P.mXx'vkkw\ྵwMSǏ|I[1kuqik~gI$ H!,MH1=r8n8:<2q:duFp;$gwWMs<-d[iڥgDF{[[Xu;ThmOgll|f?~oGZ%ÏP|P@iEkvyE+kn౏~N:'l&aYng1$֍y'$߼߳Dwu1wOi4G< x^զkoC.5Ko?rwZtA:[{|v}H ( (zt@l>_C~1x:KE<=mBְ,WFO?l|DlY9}iU=wA=ǹwV㇭, FS>oh,os-+*ua%z%tQU#߅G:r1Tݢ4quɻJIŦq2:e*tV6* >L^p_XjFsJQkBR<*\<'xWQx-?;GUt 毡;x; Jm5 _ZkKc^RIⱔaO%\SU2TjʄFn!RUiU^ ZvSxƛg98Pڝ8Yu xφ|Ib>&o>#M/AOچwzx_vbZsf{Ox3A\bW뎝5(FW98a*B mF>\iΫRRQiF gjErryԔ rQ8;Fj_şΥwxrnSƚOȊ=gNѼ-ᖷ{kӥ^Hԋېkhb1'JFGNүN.3pn85J|΃71JT%נRRJ`!NxѦӗ$V|+8sbݥ?,🃮X|5g khh:|Oi<խ#d{.$Hxl4-daɒɱۆ/ }L^"" UJ,7jNX(RxyNY6J+TkʽIqNqa*iFVsvj5x/ ŧh=w=xtEρ4l.MAq2=FĚOFxV5p~|W+n嵛.{e/kQ*WtO _/fZ^ӝJ+',4R0zi7oM8:TWiMwUŏ4h]>w-?f߈\6~ {lj75=7N{.C_Mj'F0:qxp 8R[ aNZucQaFstE:tgQJ?CJ )t*%*q//JU=xTWXX¤(RRt]?sqcG|/W촟~y>,jz??wkZ6k'?>.Y/SЭu;L-OS;n39k3|.*(,U xLe3tcTVP[|exJ)S*KN %JTgҜ"ȩr/~J/)Koj炼KGK_^<x]k:;>wHo6xğ ^@9} Ú׏qi"nUԡEӬ<6iGN|Fy F_W>,fN5aB*jUxyLbeJWUjʦ"} eIJОFONQiBmrsO2nSSRQ¯ YX?~>/#7nO)nZeXw /o^ӻ_VsؙNy 8":RBeR\F5χ9a_+k|BoSUaS>Z1)Qé{pg¯ ?~>/#37n|M(vznwMEkwvk}M}Bq5'KĶDhV;_?y?k Pg5IileIO 0~6 i /,5 Y4 hVtYjQ .,I_eayp֧†a$b*F,dӫ:2uq\&ԒZN+4%?'N FΤiNZ~sM)_w+c VC)| :/4NdioFj˺UIf^|cՙƪk4U#Gda^)du=xe~|':%oOso/g xb >>3OhGh⯳R5eO%R0cnjՙƪk4?Wײq<2=c_]{X_,Ki}cQ=m;JO-hW wᥝOi~t|7G_âO6w_ŐuîKkڤv\S&9x1aK(!5_NX hJqOѕSBӎ"qS5\jFr5TwNk+*gI^ GMi|nmk /sxtxS^|eWǷ溞0)fv7Yn53WN]B:>BΕz3ԍ|)* úԱxJ4iօ iJ/,>)*:ѧMNF xbqج5 * }[i`v4hqBJO8?:R&?JRgof%)MKfE>J|Ry{>^!iYπ&Ҽi|7x#CisM4FYmC?|UʦkYN3g\~3sxb2fyF6kcʯWeK^Q va޸F<=EOSj J'#šޗ>s 0'->'wE4Yuv=F-%utg~<#xKP~/n߅oƯ?}S|GɬG ;/h7֯miZ5S>6_/obQ W/Oi/|NK孴~Ǻ=Ʊǫ[ڝ[H.kkT5Kv9V{:=֗|owWwU啧3~BZ?e_DxIulVՒVK]\gi>azWUg֔Ǎ]Oĺ&ζjZDiEkx/;HIhvMvZeU惦h:kao- /3ÍgmޓoSOிMgÂFxvvLj4?R1eӮ2_ޟyᏈ< K _j% x}a z?wuM;F'%֑u$7u-7NJ,ƕac%h[QxL> j? F<'ПƿADu*M.y.x)WvWޓ/aEk[՞φmk?eOF~RM%e=&NIf]gm>$mxZ]mֳۡѤ?zZ/?zNW:vjt-GLtcm ^7є:">E'xg&OU=|]q/~j_8hiZw _v.<7\x*/6sdۻlhoNKx]~8;ǀ|>F3_ڳ|vs@M7o>ě.co hlDGn mR~#jzUR_<+3Rt{x\ojz7Z𕞫az}ݞ>S5 }&;_/~%k'x^x 0h%u⯌!|j孆|Mk VMO j:e΋ 8/jיRK񮙠xO5OZ|D{m/?j_ Z5-3`]>4oC@J<x:}v'GzίtA W?Mn-ӧo+)tEyT CW??l5 u>辷bͣ5c%$ LZyX!X ~3e㿄z=џo^խ+>oz&{KU%ioNMȺ|1\Y:=}5I?:ߊ>L𾇩&1mec[x X΅X,BѮg7o? |}vDF_ e_~\iOx-~xO4Smҭ5fZvO]~/?~XZ| <'~=> V~qi ia ѼGgGi&z{f:r\/+ ?cvuO | Α<2+|;N޳~tO& +—.xsSc4K *}/NKFvOM,xo|Gj7G9 QLx[?1?x'_na-=;zo{t9-7NJ,ƕac%hWPK> ?u\$w:q4Vq^LֈognZE0u8jIJY:!hArӌae'yI9T49NR}#Fbn\jjm\v?C'Qq<YM-v\nڥk[o wп FBGCG 7xKG ZEk+)l;~h_яO^ DB0V1^K-6V^Areu˷j_wk+l;QG/z/?a@Xrm}6jg_tCbzqGß016[議}n#VWSeZۧm-k wgьe4Nq3օ9.f ]Mt@9eukK<]O\I?RI=V0I;[&meEh-uY㶺XՕi&*jyTd劭'e8TN.SnW<gOq,$gzn^ms‹;?Yx+'8'L3k59aF_۩_n%cYc++Zf߭BL3ēn=}h\C?}sid^v^O;C>v? @?. W[iMY{K+m]%jii;~hgь2PB'8G +&"{K7;]NK;;k=_>3?eࡎsh?Zgb<EnutjwKA ? ;Y.`G]굩{_e{>~nOkQru(ҫokJyvemNײZ>8foټͿ|?t(/Sj `U_dfټͿ|ҏc8?(_ EZ-oïWO폱kMڟ?ؖm)|W'ٷ}y|S s^!Xio<~PjUTʧ'iN7TI;]B1hZ]۵ϚP@P@P@P@P@P@P~Fx}1:c u` clcҀ==ûH<1cd&okڏɮ/c\1K}jzwg4.&ht+G*ߧ`۱q#ۧ=!mӏ~:s@l?8|o+ ;ߧOZa:q8=>mmt^3Qg=M89tQf?~oGZ%ÏP|P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@ӷ@ =ON12AY{t'l'@;ym ( ( ( ( ( ( ( ( mS;E_8ULO͟LWa@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P_|oSW^o6l/<⥵<3V4*/%s e`]KP񦕥jԵ u? t úZ[V%\:<{x th5] ~2^6}gxzKse趓-7dǮHP@P@P@P@P@P@P@P@?w_CĿ1qҪu:>:€ (}|>P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@P@O@L0=6Olgǿz` ( ( ( ( ( ( ( (f?~oGZ%ÏP|P@?@6kZQG]Fy:","v62!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{_;?@$:5?!9 Acw~Huk3CsX ?g{CmS;E_8ULO͟LWa@P1h_t=ZH<+Ju:ao{{xoJvT4 ]8lHMOG) _ͩijo>1b>gJ6PoSsU7)Vůh=@-DŽ47}sIiG4tN=FOm@\ZIX5Ƹ\_,J0cb7JHaVhЩT*5'NBTΪp:Rn1R:sb9*FJNKrڧ>{rTy7L&t Ak~}xoMceo:MKb4[ E`שf_B8e֩RQn9(J֜<;4YKZ;(J.;[m.((9:9l)xSm&Vn7W^!:; ZClj>˪mma}mqvu-1#M^IҠBabtZ5aW^RՅ9Ÿ[ׄc)SQN1Npreh)(EZ|ҩnWӵjϸ!gϣ> |Ɨ;sĹ?vdI l5j:Tї4Z(֫uU,no<cJZF4dA6tosZo>רm6iˮ^HmV]cHƷ9֚Vӆ[}S@=Ac7]l G.!Ry>#F JԪʜ0t;{_oYIsƮMi$XaRxYFMFmyT$ysR)B|JtWaZŵƼ|k=kAާ H]sHn<;tj %CM\U+W pTV"/ BlE:tV'kףOEUJt8^5ͣKJ(S~u)SB)T7:惕jΖxFY_[Ol*4JPV]Zԥ K{E:};գJ>[y7SѳOG-wQ4Mo<5m+@LU|7ןh͵dʾQҩV*IQԯBPJj7%S({*ӅԔ`xZԠ% G*9NHBnpi4cQo#|3g5ʬ1YYfj5ɳ-Rn㴴+KY^;[+mfxӡVUӜ}UB%jTa9#F+惨¤9Ir57ӃJ)|'6fw/|5m+@ j=7NɾEuO|RXR^Ǘӆ#VPhRxʆKljuMHGAsn4ߴ\^Z$nJaM*ڝ VOD_Gxl;Kũ^k.{_ ;kmc-HrʶaOVx%z(תê.*FXXʯԅJNgN$0jpQM&7:r=Ə..m',4/Wo7V:Nwiio^L%2?yrI$CXY< yîU ֥$TgzP^1tZwNXMSS9Tn6عF.7;EIIiV&a@P@ j-?.??hZcLҵmY/dz??c}Icg{y$V6yw K[Yɲ,?TeJT);h['Be X_>oÿx?z:Ljn6hJZ[Oy'g>*+sLqb^s\Ae4mG%agKṱQ^ 0nov4{(wNi{9G9^sMrkv=F|AxR OC״}kMmEдgZmwW 5,mZ6Zaap ;pْe[fiNxlf`2&33.*_cVX,Uvx\F_uxԚRNq(2҂pq3qiGT֒|5kDR;Ӽ-xyn-ukOh:5Z~ɥ_l.VpujcpK/`ՕWͳ#^(uaj#c+R23 ]r#Yk9EBwZǕ9G)n|BOՖ{1擪iu=ՏIG%ز*糰YOq>+&O VרYObt14c0f9u5V 5Z 8VYWFg2JTB7|% |.ܮWWqmr~xk&,; {B]c¾2{4WVƟ/G19_Rx<( ?X0y5%t~&mYvqK5PΖ.Vvp)%(TRISc'7iI=c6z׆@?ﴴFJH^~զ{hmWNaWL6r8 /˨xҖ: eY<,v;e8n? k!a_ Ɇoj¥IIQq:kvN]%uVvg .XVsy嶐\%5+#DΟh͈oTs9[_(W˱Upk<3scج dR*~7 Q-ju"4n(];I'[JկZd%2=$i_[s&̵Z(?ű;^%Gx;ZM_Pּ1/zw4='] O^G"=^KԮVTm@MkX鵏ΔP|Ixi;>ՌDn\w\_^g^^y0ٷb{\kOÞ :€ (ۧJ7㮛 V>Ow|P]xD ]|9 y&MTz|if,5Kȥe)5uhυ5,/.ǧ_z`\3xū;Ohu֋_dE 2?D q,]1}^]O ^%ayE*1urjX)Ib'{k(UY*u#MLE4i~ΊV5F8N=4y7)6ԯғaXr)yugYG~-1폛{b:7V.7}:[ƁqqiMM:)gUpoy^a\|R̝)wvޫ|v>8:?¼0-J#-hIy5'zv&/uiZv }6?P5mk}-aH&u%[ C =JN"N,\%,΄:*ъ)JT!WJ#:(`)J0QcQFF6gy\3ӵ4o ~)A7WNNO:›];I~ˬ]5{ GoiͯXGֵ.lGhO[$5eX9PẑJ1yu|,!OS:19t!r:0~ړC5UNQ>VU΋*DJ7MS^IIE%:|˗?3xĞ,/V/}7s@x",u-\i'5K|$dX< /r(IfRM(BG =+ǸTB*|Ӎ<= uh;/iS g6ڦuoi7RJ^}ZwM ;E">Co,:\%ѵVLhkOW1aS=Tj'4WtLv+UbgRgEQ}%Uԏ*^'Ft?o >i ~:>*&?cmᏴkr:ֵKdI7N7 9PҞ+='a}2:%xfӝ~J#F7y7V\%Vdu!8Tξ BWi,KrTUk{g{XѼI}/\~|t5e^}eۯ66ZNy-};{ª9a4jbokBa–G>JͪOVh|/Yr>n8 L^ v qĹ>IAE>X7~unG[^ [(4/׵?D4H^ e G|2zǁ-.Eew,Ūvi\7zGeu1XN CugD9SRiuJx|st^Vⱎ1ҭJT״[:JqZ.I7BRQZzai_ -|xXYuo_Z<#o|k[V5'>$Tu{˧{ɥtq†z\O5'5cp1R_4BQMVS%h(NU/5)ӆ* SVTJx/gS(=gp&Ŀg"_Gr>GFgt#Q^r}EiG/X2ooe,ڼ?%R:QkعQ ϕ{/;'MCrTp&ģŃ_,/F_àc1oŵ Nbni[Zm,UPrStO)HkhS<7 Aèzݏ4/hZEt7V,nӬGa=)hN~RI9үĺΕP,%\D)T2j2 jVu xP6YK&?Ğ χ\1x%nMԚCYF$O;l<[bx\,r9n+[50+LOiW$N>R 8\g6.\yN |gݺ..TՔmy&3[m b+]^ baO:_~gn,| ×#֜:+ٿK\.ø|~_pgO<;Kqm<ʆ]/QrQ,LΜŋF8fPU}UG-.~¤jJJ fZhDI?> ~ѺQ5}@5پi:fТhnl,'x&5]O2tpٮRbj񟇘\f/4xLeLÉ|e1cVaxxU b{TjM+{rj2gW! x[:_GjW7zWjNnm>FŏVw /eY?>kfxlF[ľf|.OCC93RY:gN1b6j$p%´S&ABq,M>GM΢Vk+-~ ^c<1+鿳;|%FԵ_f铟i/AI#Rk?4,n+r|,p9f}Ĺ592pì %sRkT\+(֩Vj9Nc SқIIJVKCǾ[~Լkxw]Ι[i lMi]Goퟆ>iK[A[q"ǩ~qEF767`[5fhbxbs+XK5)=\Ɇ,::8]llA?e:J.ι\k>dh!% ι\Œ>}Nu|);WvYӬ$܁!`2n.2|_ `'x &SxKYTigyFk 5~.<0QfX8ͫ96 RМZj)K>f8%*wsrȻ3ľ'ҼlZ|JQ MRh׉tBK8?'W~!gGiR1七%ɣ^T eOGOS̲i6<7gP:]<}JsytsӧQB5]XIP7u_ -(sEm>f#;^#Zo&|(Tr閒]BUӶdr&_e9Se'q\W]pg1u_uOp1*JݣG8YoxxJ2jsI(xڟ`hz͟4]bVJb\е KlCc  ʹ>df9;fu|0eغ:;,;ZN2˫u$SNZL( ÛlB)F/֯8a@P@P@P@P@P@Pm/c$a O`4* /|4XEM-7<#x{Ꮑ-u U~KZ&/ Yx?F֣YGssig\i^-hGKi]>#2o|Wsm:G}#OnxOz4i>ԼoWHY׭WGToE%5|d~DZ_}I>όw׃¾ermƫk]|}9Tٴ/Z5:C{|Rj1K▱]𽮛e]OQ~OFk?;]2w7ڟ_P{tշt2]?ijڶZxwXOy ^ ͢ Y w-;ob585|6._ C~w6~1Z_>?~"Cx]5nEǟi^\ڨ/90/4{_~=AF_.}=xNs]y#󍼬>ec-m5O{߁5O >ꗿ|{?+P;^.te%IF$zMoDIwIvv| 1jV|(wjZfk5j^ѭ/,/9l`)mE7u"['R$( ( ( ( ( (:ٷOQcpT90u?6}1_cniڷNlٵݫ݆_?88NXcnh[^wﻹ~7@zq?ZkMݶmV-m~m62p1olO+%[Y+iWAwt]׮۠dz;ڝݵoӢ^Y{?ZޯPsH{4Kݷe]Y=I.3ǻ dZ;6-=VZZ-m7o[ziAqǮ؏׽=ow׶Gu mhmkZtws z qqw8;ݧkmw{yֿP;מ1lvZ6ֵWkVz6/0ϿLu,z :%*W][6ӳk>aw$ t'>ߒɭ-U[ GA}ޟZ[id{JZY; V^D-H2=A=9օYYy+N-lٴ v{uNݬ۾3ב=?*kFZ;[EE[_M}=:ucN:O4-KN]-^DvCN{mi-ۥp13Qr쬶Y]&o–[r?_$ ?:vYZ;+|o-JM[[2\@s8qN.ޞ^_߽kgnq9#o^Kz~ioVj7II_~;89(}/ϻK-<8ҍg߯NM߻ekY[}m]8K9\wEV^Jӷ[6mo[[w݆qu99 g+F=uO=Z.mzmn[e;?>zb{.[|{[[[k Q4dz#~Mh@G2=G?_& q3c1۰ݭz6=nZ;}8qҞ5ovMv7waqc88\qЮռOKݷN[ۺoT{zzEm˺]͍6w}~a;Q[%OQ/uY$=^ps>>vVh_aݫt>rGBx=}ϭ5kd}|۶ڽ5b/NEh 8ya qzR[]]=Y.|N v z q>լڲj˲![~38?wovUC#~M!G2=G?_& Q4dz#~Mh@G2=G?_& Q4dz#~Mh@G2=G?_& Q4dz#~Mh@q }F1Pӌ䁏A>\~k@ c1}፽;uc1ǶaJo+@A`c#q61P`d:0x@mq~^zV@遏1z6Ltsm ;A@lhdz#~Mh@G2=G?_& Q4dz#~Mh@G2=G?_& Q4dz#~Mh@G2=G?_&?PKPƥjRGRPK.AOEBPS/img/family.gifIGIF89a:ضȿݐЭ???YYY"""fffCCCXXX&&&;;;GGGnnnwwwDDD...{{{bbbᩩًHHH!!!333jjjTTT\\\zzz)))²---666杝%%%vvvLLLeeeˠ>>> ]]]aaa222PPP777sss***UUUdddժxxxrrr^^^iii,: ͜ ꠸皯6p+߾~TPC>h!oŋQBn"8R$H<PՀF&:@!ϟ?_(!ntBLӧP>5@ +,LJPBBbٳbrB L.+nAjUŠ C.י2M0F5*^l7pȱ!qUtcȒgVe]Ȱ@b۸;%aAA1$鸉:+"<7UGNQ5SRT@}i$vH%ԦZNS#ժ~ɣ<]OU`uNZͪH ŬhMՊV8k}k-ְG,JVI]:X@}PbIš,],$ ʆ<5 ̮]3KX.u,B&nXUYخCeN]VNVjU~5g}[0ӹyrK;6ȭu-rHt>MxJw8pEp}FIzK xLB\G1uTz\ $gF!%$M2HB 8Ay21xD !/Rc{oC_c cc˘O*|ahfА@ 4bVAho*APd@ 0[(*`<6M i~L  DiSP!~%&i)2 BPA@8|ҝ2=z$ITQuBτ&Ѕ+a\)@ dޥX!F  t`~W`+;|S_' pA\BRA"M C B|<`[Bkׂ>w 'e2! ?+4 2 C#uv 0[ )tevgǂ2L_<g(b]׋Be`T #O,P:,B)cg>Y运;g%%l-R&F0>'5b |OD*%]@[s 0 65@OTh {6DA{`BaHOPB{k,}!`&8 @rBLADpJvCl~BOPL yNf0 bQ pI}37 a@1pc :q37u1+2C3Ё 3SKq 3*( ,8cCl`@se6p 6qS;V 1s&=$C?ǣv0P&eT kfpW H:q??YRC ɶ$xA@@֏k R@;KN0&h`LQKЀ%K1noY )A~TG3ao14Cq8%K9M HP 2DCI) WɈ`4dy8 N@'P;`@1|'}I*PCA  131PBKDKaKLFoG9b,Fnp9I )#apq}Yh2@quCXy%'I;P'uTvCT EAIJ$@M`NJJY ) (IԝSWu*+IdQh   Z z 2`{5@e@pfࢦ@ȗc `fࣿP/);PKC}PK.A!OEBPS/img_text/spgateway_xslt.htm  Description of the illustration spgateway_xslt.jpg

This screen capture shows the XSLT Management Page, which is described in text before the illustration.

PKUqM>% PK.A$OEBPS/img_text/spgateway_index_2.htm Description of the illustration spgateway_index_2.jpg

This screen capture shows the XML format for the response to the query.

PKvPK.A"OEBPS/img_text/cancer_ontology.htmx Description of the illustration cancer_ontology.eps

This is a text description of Figure 2-1, "Cancer Ontology Example". It illustrates a cancer ontology, as described in text after the illustration.

PKLq$}xPK.A$OEBPS/img_text/spgateway_index_1.htm/ Description of the illustration spgateway_index_1.jpg

This screen capture includes the following controls:

  • SPARQL Endpoint

  • Location selector (here showing "local")

  • SPARQL SELECT Query Body

  • Submit Query button

  • Show Advanced Options link

PK8PK.A%OEBPS/img_text/spgateway_browse_3.htmp Description of the illustration spgateway_browse_3.jpg

This screen capture shows the query and the response from clicking in the specified URI link that appears in the preceding Figure 7-5. The response is shown in table form.

PKtLPK.AOEBPS/img_text/family.htm Description of the illustration family.eps

This is a text description of Figure 1-3, "Family Tree for RDF Example". It illustrates the family tree for the example that uses an RDF model for family information. The family relationships are explained in text before the illustration and in comment lines before the INSERT statements in the example.

PKsKPK.A%OEBPS/img_text/spgateway_browse_1.htmo Description of the illustration spgateway_browse_1.jpg

This screen capture includes the following controls:

  • SPARQL Endpoint

  • Location selector (here showing "sparql.org")

  • SPARQL SELECT Query Body and selector control

  • Submit Query button

  • Hide Advanced Options link

  • Timeout (ms) (here showing "3000") and selector control

  • Best Effort check box (here not selected)

PKú_toPK.A(OEBPS/img_text/spgateway_sparql_mgmt.htm' Description of the illustration spgateway_sparql_mgmt.jpg

This screen capture shows the XSLT Management Page, which is described in text before the illustration.

PK7& ,'PK.A"OEBPS/img_text/shortest_a_to_d.htm2 Description of the illustration shortest_a_to_d.jpg

This illustration is an image of the output from the analytical functions used in Example 7-3 in Section 7.9, "Analytical Functions for RDF Data". The image shows four nodes, each representing a person, and lines with arrowheads showing the relationships between the persons (expressed as the predicates differs, likes, dislikes, and knows).

PKsYjPK.A&OEBPS/img_text/spgateway_browse_1a.htm< Description of the illustration spgateway_browse_1a.jpg

This screen capture shows the query response in table form. The column headings are Row Count, Subject, Predicate, and Object.

PKf2A<PK.AOEBPS/img_text/sem_overview.htm( Description of the illustration sem_overview.eps

This is a text description of Figure 1-1, "Oracle Semantic Capabilities". It illustrates the storage, query, and inferencing capabilities of Oracle Database. These capabilities are explained in text after the illustration.

The illustration contains four boxes, labeled INFER, QUERY, STORE, and Database.

The INFER box contains boxes labeled "OWL subset", "RDF/S", and "User-defined", with an arrow pointing to "RDF/OWL data and ontologies" in the Database box.

The QUERY box contains two boxes: one labeled "Query RDF/OWL data and ontologies and the other labeled "Ontology-assisted query of relational data", with an arrow pointing from each to "RDF/OWL data and ontologies" in the Database box, and an arrow pointing from "Ontology-assisted query of relational data" to "Enterprise (relational) data" in the Database box.

The STORE box contains boxes labeled "Bulk Load" and "Incremental Load and DML", with an arrow pointing to "RDF/OWL data and ontologies" in the Database box.

PK<ٿPK.A%OEBPS/img_text/wm_rdf_rules_index.htmc Description of the illustration wm_rdf_rules_index.gif

This is a text description of Figure 6-1, "Physical Versioning of Entailment (Rules Index)", which is described before and after the illustration.

The illustration uses boxes to show the following: The RDF model includes some differences from workspace W1 and some differences from workspace W2. For the rules index, there are three private copies maintained: one for the LIVE workspace, one for the W1 workspace, and one for the W2 workspace. Each of the private copies for workspaces W1 and W2 includes the information for the LIVE workspace, plus information about differences from the associated other workspace (W1 or W2).

PKתhcPK.AOEBPS/img_text/inferencing.htm2 Description of the illustration inferencing.eps

This is a text description of Figure 1-2, "Inferencing". It illustrates the inferencing capabilities of Oracle Database, as explained in text before and after the illustration.

The illustration uses boxes and arrows to show the following: Inferred Triple Set 1 is derived from Model 1 using rules in Rulebase 1; Inferred Triple Set 2 is derived from Model 2 using rules in Rulebase 1 and Rulebase 2; and the database can have additional models, rulebases, and inferred triple sets.

PKPK.A,OEBPS/img_text/obiee_impmeta_metaobjects.htm# Description of the illustration obiee_impmeta_metaobjects.jpg

This illustration shows the Select Metadata Objects page of the Import Metadata wizard. It contains a Find box for finding metadata objects, and a Datasource View pane on the left and a Repository View box on the right. Each view contains an expandable hierarchy tree, with a check box option to Show Complete Structure (checked in this illustration).

PKƒ(#PK.A+OEBPS/img_text/obiee_impmeta_datasource.htmb Description of the illustration obiee_impmeta_datasource.jpg

This illustration shows the Select Data Source page of the Import Metadata wizard. After this illustration are instructions for fields that you should fill in.

PK;,gbPK.A*OEBPS/img_text/obiee_impmeta_metatypes.htm Description of the illustration obiee_impmeta_metatypes.jpg

This illustration shows the Select Metadata Types page of the Import Metadata wizard. The following metadata types are shows as selected: Tables, Keys, and Foreign Keys. The following metadata types are shown as deselected (not selected): Metadata From CRM Tables, System Tables, Aliases, Synonyms, and Views.

PK"PK.AOEBPS/sem_rdfctx_ref.htm9Bƽ SEM_RDFCTX Package Subprograms

12 SEM_RDFCTX Package Subprograms

The SEM_RDFCTX package contains subprograms (functions and procedures) to manage extractor policies and semantic indexes created for documents. To use the subprograms in this chapter, you should understand the conceptual and usage information in Chapter 4, "Semantic Indexing for Documents".

This chapter provides reference information about the subprograms, listed in alphabetical order.


SEM_RDFCTX.ADD_DEPENDENT_POLICY

Format

SEM_RDFCTX.ADD_DEPENDENT_POLICY(

     index_name IN VARCHAR2,

     policy_name IN VARCHAR2,

     partition_name IN VARCHAR2 DEFAULT NULL);

Description

Adds a dependent policy to an (already created) index or index partition.

Parameters

index_name

Name of the index.

policy_name

Name of the dependent policy.

partition_name

If the specified index is local, the name of the target partition. (Otherwise, must be null.)

Usage Notes

The base policy corresponding to the new dependent policy must already be a part of the index.

Examples

The following example adds a new dependent policy SEM_EXTR_PLUS_GEOONT to the index ArticleIndex.

begin
  sem_rdfctx.add_dependent_policy (index_name  => 'ArticleIndex',
                                   policy_name => 'SEM_EXTR_PLUS_GEOONT');
end;
/

SEM_RDFCTX.CREATE_POLICY

Format

SEM_RDFCTX.CREATE_POLICY(

     policy_name IN VARCHAR2,

     extractor mdsys.rdfctx_extractor,

     preferences sys.XMLType DEFAULT NULL);

or

SEM_RDFCTX.CREATE_POLICY(

     policy_name IN VARCHAR2,

     base_policy IN VARCHAR2,

     user_models IN SEM_MODELS DEFAULT NULL,

     user_entailments IN SEM_MODELS DEFAULT NULL);

Description

Creates an extractor policy. (The first format is for a base policy; the second format is for a policy that is dependent on a base policy.)

Parameters

policy_name

Name of the extractor policy.

extractor

An instance of a subtype of the RDFCTX_EXTRACTOR type that encapsulates the extraction logic for the information extractor.

preferences

Any preferences associated with the policy.

base_policy

Base extractor policy for a dependent policy.

user_models

List of user models for a dependent policy.

user_entailments

List of user entailments for a dependent policy.

Usage Notes

An extractor policy created using this procedure determines the characteristics of a semantic index that is created using the policy. Each extractor policy refers to an instance of an extractor type, either directly or indirectly. An extractor policy with a direct reference to an extractor type instance can be used to compose other extractor policies that include additional RDF models for ontologies.

An instance of the extractor type assigned to the extractor parameter must be an instance of a direct or indirect subtype of type mdsys.rdfctx_extractor.

The RDF models specified in the user_models parameter must be accessible to the user that is creating the policy.

The RDF entailments specified in the user_entailments parameter must be accessible to the user that is creating the policy. Note that the RDF models underlying the entailments do not get automatically included in the dependent policy. To include one or more of those underlying RDF models, you need to include the models in the user_models parameter.

The preferences specified for extractor policy determine the type of repository used for the documents to be indexed and other relevant information. For more information, see Section 4.8, "Indexing External Documents".

Examples

The following example creates an extractor policy using the gatenlp_extractor extractor type, which is included with the Oracle Database support for semantic indexing.

begin
  sem_rdfctx.create_policy (policy_name => 'SEM_EXTR',
                            extractor   => mdsys.gatenlp_extractor());
end;
/

The following example creates a dependent policy for the previously created extractor policy, and it adds the user-defined RDF model geo_ontology to the dependent policy.

begin
  sem_rdfctx.create_policy (policy_name => 'SEM_EXTR_PLUS_GEOONT',
                            base_policy => 'SEM_EXTR',
                            user_models => SEM_MODELS ('geo_ontology'));
end;
/

SEM_RDFCTX.DROP_POLICY

Format

SEM_RDFCTX.DROP_POLICY(

     policy_name IN VARCHAR2);

Description

Deletes (drops) an unused extractor policy.

Parameters

policy_name

Name of the extractor policy.

Usage Notes

An exception is generated if the specified policy being is used for a semantic index for documents or if a dependent extractor policy exists for the specified policy.

Examples

The following example drops the SEM_EXTR_PLUS_GEOONT extractor policy.

begin
  sem_rdfctx.drop_policy (policy_name => 'SSEM_EXTR_PLUS_GEOONT');
end;
/

SEM_RDFCTX.MAINTAIN_TRIPLES

Format

SEM_RDFCTX.MAINTAIN_TRIPLES(

     index_name IN VARCHAR2,

     where_clause IN VARCHAR2,

     rdfxml_content sys.XMLType,

     policy_name IN VARCHAR2 DEFAULT NULL,

     action IN VARCHAR2 DEFAULT 'ADD');

Description

Adds one or more triples to graphs that contain information extracted from specific documents.

Parameters

index_name

Name of the semantic index for documents.

where_clause

A SQL predicate (WHERE clause text without the WHERE keyword) on the table in which the documents are stored, to identify the rows for which to maintain the index.

rdfxml_content

Triples, in the form of an RDF/XML document, to be added to the individual graphs corresponding to the documents.

policy_name

Name of the extractor policy. If policy_name is null (the default), the triples are added to the information extracted by the default (or the only) extractor policy for the index; if you specify a policy name, the triples are added to the information extracted by that policy.

action

Type of maintenance operation to perform on the triples. The only value currently supported in ADD (the default), which adds the triples that are specified in the rdfxml_content parameter.

Usage Notes

The information extracted from the semantically indexed documents may be incomplete and lacking in proper context. This procedure enables a domain expect to add triples to individual graphs pertaining to specific semantically indexed documents, so that all subsequent SEM_CONTAINS queries can consider these triples in their document search criteria.

This procedure accepts the index name and WHERE clause text to identify the specific documents to be annotated with the additional triples. For example, the where_clause might be specified as a simple predicate involving numeric data, such as 'docId IN (1,2,3)'.

Examples

The following example annotates a specific document with the semantic index ArticleIndex by adding triples to the corresponding individual graph.

begin
  sem_rdfctx.maintain_triples(
     index_name      => 'ArticleIndex',
     where_clause    => 'docid = 15',  
     rdfxml_content => sys.xmltype(
      '<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" 
                xmlns:rdfs="http://www.w3.org/2000/01/rdf-schema#"
                xmlns:pred="http://myorg.com/pred/">
       <rdf:Description rdf:about=" http://newscorp.com/Org/ExampleCorp">
         <pred:hasShortName 
               rdf:datatype="http://www.w3.org/2001/XMLSchema#string">
             Example
         </pred:hasShortName>
     </rdf:Description> 
    </rdf:RDF>'));
end;
/

SEM_RDFCTX.SET_DEFAULT_POLICY

Format

SEM_RDFCTX.SET_DEFAULT_POLICY(

     index_name IN VARCHAR2,

     policy_name IN VARCHAR2);

Description

Sets the default extractor policy for a semantic index that is configured with multiple extractor policies.

Parameters

index_name

Name of the semantic index for documents.

policy_name

Name of the extractor policy to be used as the default extractor policy for the specified semantic index. Must be one of the extractor policies listed in the PARAMETERS clause of the CREATE INDEX statement that created index_name.

Usage Notes

When you create a semantic index for documents, you can specify multiple extractor policies as a space-separated list of names in the PARAMETERS clause of the CREATE INDEX statement. As explained in Section 4.3, "Semantically Indexing Documents", the first policy from this list is used as the default extractor policy for all SEM_CONTAINS queries that do not identify an extractor policy by name. You can use the SEM_RDFCTX.SET_DEFAULT_POLICY procedure to set a different default policy for the index.

Examples

The following example sets CITY_EXTR as the default extractor policy for the ArticleIndex index.

begin
  sem_rdfctx.set_default_policy (index_name => 'ArticleIndex',
                                 policy_name => 'CITY_EXTR');
end;
/

SEM_RDFCTX.SET_EXTRACTOR_PARAM

Format

SEM_RDFCTX.SET_EXTRACTOR_PARAM(

     param_key IN VARCHAR2,

     patam_value IN VARCHAR2,

     param_desc IN VARCHAR2);

Description

Configures the Oracle Database semantic indexing support to work with external information extractors, such as Calais and GATE.

Parameters

param_key

Key for the parameter to be set.

param_value

Value for the parameter to be set.

param_desc

Short description for the parameter to be set.

Usage Notes

You must have the SYSDBA role to use this procedure.

To work with the Calais extractor type (see Section 4.9), you must specify values for the following parameters:

  • CALAIS_WS_ENDPOINT: Web service end point for Calais.

  • CALAIS_KEY: License key for Calais.

  • CALAIS_WS_SOAPACTION: SOAP action for the Calais Web service.

To work with the General Architecture for Text Engineering (GATE) extractor type (see Section 4.10), you must specify values for the following parameters:

  • GATE_NLP_HOST: Host for the GATE NLP Listener.

  • GATE_NLP_PORT: Port for the GATE NLP Listener.

In addition to these parameters, you may need to specify a value for the HTTP_PROXY parameter to work with information extractors or index documents that are outside the firewall.

A database instance only has one set of values for these parameters, and they are used for all instances of semantic indexes using the corresponding information extractor. You can use this procedure if you need to change the existing values of any of the parameters.

Examples

For examples, see the following sections:

PKJ>B9BPK.A OEBPS/toc.ncx Oracle® Database Semantic Technologies Developer's Guide, 11g Release 2 (11.2) Cover Table of Contents List of Examples List of Figures List of Tables Oracle Database Semantic Technologies Developer's Guide, 11g Release 2 (11.2) Preface What's New in Semantic Technologies? Conceptual and Usage Information Oracle Database Semantic Technologies Overview OWL Concepts Simple Knowledge Organization System (SKOS) Support Semantic Indexing for Documents Fine-Grained Access Control for RDF Data Workspace Manager Support for RDF Data Jena Adapter for Oracle Database Sesame Adapter for Oracle Database Reference and Supplementary Information SEM_APIS Package Subprograms SEM_OLS Package Subprograms SEM_PERF Package Subprograms SEM_RDFCTX Package Subprograms SEM_RDFSA Package Subprograms Enabling, Downgrading, or Removing Semantic Technologies Support SEM_MATCH Support for Spatial Queries Glossary Index Copyright PKLCPK.AOEBPS/content.opf3` Oracle® Database Semantic Technologies Developer's Guide, 11g Release 2 (11.2) en-US E25609-03 Oracle Corporation Oracle Corporation Oracle® Database Semantic Technologies Developer's Guide, 11g Release 2 (11.2) 2012-05-01T06:29:17Z Provides usage and reference information about Oracle Database support for semantic technologies, including storage, inference, and query capabilities for data and ontologies based on Resource Description Framework (RDF), RDF Schema (RDFS), and Web Ontology Language (OWL). PKMN33PK.A OEBPS/lof.htms List of Figures PKj wx s PK.AOEBPS/dcommon/prodbig.gif GIF87a!!!)))111BBBZZZsss{{ZRRcZZ!!1!91)JB9B9)kkcJJB991ssc絽Zcc!!{祽BZc!9B!c{!)c{9{Z{{cZB1)sJk{{Z{kBsZJ91)Z{!{BcsRsBc{9ZZk甽kBkR!BZ9c)JJc{!))BZks{BcR{JsBk9k)Zck!!BZ1k!ZcRBZcZJkBk1Z9c!R!c9kZRZRBZ9{99!R1{99R{1!1)c1J)1B!BJRkk{ƽ絵ތkk絵RRs{{{{JJsssBBkkk!!9ss{{ZZssccJJZZRRccRRZZ))cBBJJ99JJ!!c11991199Z11!c!!))Z!!!1BRck{)!cJBkZRZ,HP)XRÇEZ֬4jJ0 @ "8pYҴESY3CƊ@*U:lY0_0#  5tX1E: C_xޘeKTV%ȣOΏ9??:a"\fSrğjAsKJ:nOzO=}E1-I)3(QEQEQEQEQEQEQE֝Hza<["2"pO#f8M[RL(,?g93QSZ uy"lx4h`O!LŏʨXZvq& c՚]+: ǵ@+J]tQ]~[[eϸ (]6A&>ܫ~+כzmZ^(<57KsHf妬Ϧmnẁ&F!:-`b\/(tF*Bֳ ~V{WxxfCnMvF=;5_,6%S>}cQQjsOO5=)Ot [W9 /{^tyNg#ЄGsֿ1-4ooTZ?K Gc+oyڙoNuh^iSo5{\ܹ3Yos}$.nQ-~n,-zr~-|K4R"8a{]^;I<ȤL5"EԤP7_j>OoK;*U.at*K[fym3ii^#wcC'IIkIp$󿉵|CtĈpW¹l{9>⪦׺*ͯj.LfGߍԁw] |WW18>w.ӯ! VӃ :#1~ +މ=;5c__b@W@ +^]ևՃ7 n&g2I8Lw7uҭ$"&"b eZ":8)D'%{}5{; w]iu;_dLʳ4R-,2H6>½HLKܹR ~foZKZ࿷1[oZ7׫Z7R¢?«'y?A}C_iG5s_~^ J5?œ tp]X/c'r%eܺA|4ծ-Ե+ْe1M38Ǯ `|Kյ OVڅu;"d56, X5kYR<̭CiطXԮ];Oy)OcWj֩}=܅s۸QZ*<~%뺃ȶp f~Bðzb\ݳzW*y{=[ C/Ak oXCkt_s}{'y?AmCjޓ{ WRV7r. g~Q"7&͹+c<=,dJ1V߁=T)TR՜*N4 ^Bڥ%B+=@fE5ka}ędܤFH^i1k\Sgdk> ֤aOM\_\T)8靠㡮3ģR: jj,pk/K!t,=ϯZ6(((((((49 xn_kLk&f9sK`zx{{y8H 8b4>ÇНE|7v(z/]k7IxM}8!ycZRQ pKVr(RPEr?^}'ðh{x+ՀLW154cK@Ng C)rr9+c:׹b Жf*s^ fKS7^} *{zq_@8# pF~ [VPe(nw0MW=3#kȵz晨cy PpG#W:%drMh]3HH<\]ԁ|_W HHҡb}P>k {ZErxMX@8C&qskLۙOnO^sCk7ql2XCw5VG.S~H8=(s1~cV5z %v|U2QF=NoW]ո?<`~׮}=ӬfԵ,=;"~Iy7K#g{ñJ?5$y` zz@-~m7mG宝Gٱ>G&K#]؃y1$$t>wqjstX.b̐{Wej)Dxfc:8)=$y|L`xV8ߙ~E)HkwW$J0uʟk>6Sgp~;4֌W+חc"=|ř9bc5> *rg {~cj1rnI#G|8v4wĿhFb><^ pJLm[Dl1;Vx5IZ:1*p)إ1ZbAK(1ׅ|S&5{^ KG^5r>;X׻K^? s fk^8O/"J)3K]N)iL?5!ƾq:G_=X- i,vi2N3 |03Qas ! 7}kZU781M,->e;@Qz T(GK(ah(((((((Y[×j2F}o־oYYq $+]%$ v^rϭ`nax,ZEuWSܽ,g%~"MrsrY~Ҿ"Fت;8{ѰxYEfP^;WPwqbB:c?zp<7;SBfZ)dϛ; 7s^>}⍱x?Bix^#hf,*P9S{w[]GF?1Z_nG~]kk)9Sc5Ո<<6J-ϛ}xUi>ux#ţc'{ᛲq?Oo?x&mѱ'#^t)ϲbb0 F«kIVmVsv@}kҡ!ˍUTtxO̧]ORb|2yԵk܊{sPIc_?ħ:Ig)=Z~' "\M2VSSMyLsl⺿U~"C7\hz_ Rs$~? TAi<lO*>U}+'f>7_K N s8g1^CeКÿE ;{+Y\ O5|Y{/o+ LVcO;7Zx-Ek&dpzbӱ+TaB0gNy׭ 3^c T\$⫫?F33?t._Q~Nln:U/Ceb1-im WʸQM+VpafR3d׫é|Aү-q*I P7:y&]hX^Fbtpܩ?|Wu󭏤ʫxJ3ߴm"(uqA}j.+?S wV ~ [B&<^U?rϜ_OH\'.;|.%pw/ZZG'1j(#0UT` Wzw}>_*9m>󑓀F?EL3"zpubzΕ$+0܉&3zڶ+jyr1QE ( ( ( ( ( ( ( (UIdC0EZm+]Y6^![ ԯsmܶ捆?+me+ZE29)B[;я*wGxsK7;5w)}gH~.Ɣx?X\ߚ}A@tQ(:ͧ|Iq(CT?v[sKG+*רqҍck <#Ljα5݈`8cXP6T5i.K!xX*p&ќZǓϘ7 *oƽ:wlຈ:Q5yIEA/2*2jAҐe}k%K$N9R2?7ýKMV!{W9\PA+c4w` Wx=Ze\X{}yXI Ү!aOÎ{]Qx)#D@9E:*NJ}b|Z>_k7:d$z >&Vv󃏽WlR:RqJfGإd9Tm(ҝEtO}1O[xxEYt8,3v bFF )ǙrPNE8=O#V*Cc𹾾&l&cmCh<.P{ʦ&ۣY+Gxs~k5$> ӥPquŽўZt~Tl>Q.g> %k#ú:Kn'&{[yWQGqF}AЅ׮/}<;VYZa$wQg!$;_ $NKS}“_{MY|w7G!"\JtRy+贾d|o/;5jz_6fHwk<ѰJ#]kAȎ J =YNu%dxRwwbEQEQEQEQEQEQEQEQEQE'fLQZ(1F)hQ@X1KEQE-Q@ 1KE3h=iPb(((1GjZ(-ʹRPbR@ 1KE7`bڒyS0(-&)P+ ڎԴP11F)h&:LRmQ@Q@Š(((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((?l:ϊw "{{-3j3%{sj~2= 7 ~MڅKrHb|P3 r=Ҁ +Ş/$iu7=q2dԂxn⸷9$l]H #WI񯄴;\[ݚD8C3p&0U9^AnK vI+!I8>5(zqj03Y.X ,@85ߛ8>pq8=} \xmm常8` $Q@$v7zwp]ɝA GX;y_]覮O&4 SPtY.X),@84U=7Vuv K4,$g{@<+uqtiGw3; I@ORմn5MBp%8'ƫ%u6uBJrHRN2@ϸ J(9i[[my1*ڏV99jnqko{Z[Ov-eF9`0=Gs~։{C$Ҥ-!#NȻ`<1bF:k6puQ-'UfzoOۼ?-ϕv83^Y>_ A-GM",6Jr_y NK|6rWKO![4˭?deHGC6gYO^,kQH]*x6 LI{"#bRencMO=Hxg&t*/!# =AGtg@k5MRS%n6$IUOvuhH!Av7\܃`gugqqG4 t: ԿO[?n0g}Oqq3w]^u \FW |ozR#~Mh=0py.|wZu7ڼ*l$Fpz@򏈩Tүƞ)7=&Vd0܀Dk `+sJoj:Env8+qif6i2s°Lぐy{~0_ExXnb0<: V,|5P)#|OM 9G0sץ$|[O[ښe^,S,$Eac>dym>^oOqui<m( W|]nap'&"8_vkw%9NWeJ9ROcWuyZ&`\'`2eF":7~;<qeu%^$r2h$8<3''~C?Ua ⅏n4/ZF{9_i`W!H-\_}a<]x_BIqK< #1pqm(ѯtЖ!ot?6щd-P7`c!Fs#:6px\u# ˥DtgH[ 2qˎNˋVcp!o՗1tp)?W6N:0I:5#wi'үc*YFRCJ q]$:DtῆOt{oxOzBFo%$qC*q% @Wohtr{yK$#(?1V r~cyZ襯?'u@|=ໍf^ε$vUfeYqScŝO9<)OksBʧ08 vI6/ v=.+(KTll2TasO^']_J#K}[gC \y͹w'@|: NJhLr>wFh K`3i3nu-~I4K.cc0#  @_W?8>/ ,gHZ#kK Q8`ݹgDӯ]6ݟd{/̚%~v;IaXT|gkAH)bfFc`{Meoֿ?Zjv}wPTz>x3+v)Y[U%UB6w 9+%Onjn<ixmm\uX{*:rs~ -/}#ۻ jvWQ@'MѭK,`g.±)l3*P?7<]w3"NxUp{[xV决mAo}aCj"FH ɑO⮂t=/:sj^ڶNɓ;InS[7¾d]Iɔ$0GUuP7__\kLwS#HsQ#9 'MSYMK@-n'>dG܊vGNQ@ؚw#gg>oo;6mݝw>|,MޏiϠ@2A"|v;Kc'J(~ׄ|áZI&CMB2ܖ83Zciwfm͝l'#@#U(?>;|9șY&CG(T8ֺ 6euK^?X HtZآ9CƇi[}d[7m`Õ @<Т+Xu~]I )<:}DžkZx{=F.<miwV_0 {M r쌄2px7gt OD$8FVF œ*3a]hw4x4&0nϷs<$I [F%gKY%32y&\ ˒z(b'oikkQ2 ε'ym"h%BG"WR0A%Z7\ 9UIn m8lb5 hv~}֗/fkS)=+b+~x'_^Q {wI!I$C1$'ֻ (м)fmt=2 ($r͍'( ( ( ( ( ( ( ( ( ( ( ( ( ( (fDGn7wgmhc#4J|ՕdT+1V hH> Z/hmqHWEkc#(~jl_5%j<YwH^N᱋Mw eJm.[vFӮ2)A]ڀ9Q6Ʊ3CէZkH{7SˈA5,+xW^mu=6'4f5FުѪ\C (_׈nPFwŠ=+xK/x~̟ڥG)Rm̎W?iW~'}H\GpY<8׼AῈUOn&V̠nsR͕-Bx`?0j>%_rSº&u/*M|#a- ە|_^u/뚐[5;;ž @ J9y%H`x4msV2qgCI\$P** +Es4|%j^wW*YrviI SmIxC]mtwəZ$ Q^w^$C>{Y#Đ6$K 1 X<3x?5q,{ B$*rpv2>?|2zz6-#iLDŔ s]QEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEgmk+VcGOvu qo,l1sݯRz<wjom.9R(nP bN@r8?9玑WnM^[Qwx`qs@=2KsUr@cIo~^{V[6$eTd(9 ZM|cp KPBDWW!Tyjl`5+ E̊"Jm$>g>yfnvp~(+ꏩM݅ܮ;ʨ&cYYX9P2XkC_ 9;Thm#%N*(q='CGנxIuVcGOvu qo,l1sݯRzcVatk.X/m䷑ 0WR3h%$B(IaY?TUݟ|9iI<< r@O^ |Ms ptQ @$O*Z^j Z[4?d1tb8Uxƾ |yCW pjX+.a9]Cuo-QRHPF Ab1#jX2%@&_q\ ѷ#pEy/%h9΁pn2ѰJ3n)5~ o- $B޸@WYOs4m:8)K>ysRE0f8v,>,xKuHoqZMikr;\ =JuZ-+WiZ 3}&0$a!Vb5(~x#גiw<ocmIFS)Qʻq !пJ$qyVOv I=CӮ<=mHmHBO }AҬx+Zoi.{YUڪ1h/<]3:ޖnNE Fg!D#8 r88o?[yn.5}f"BI%*dLxs^^5ӖZ| I 0*89R@8> yoc\ #DVt:i_n]R{K=s'BnQ`*^>Not˜bC,|, 9ȮC=Ĉ-JnB;ssK}En5-=1Z܆Rr~c+q ߃xD j1<0#[N&FSH C< ^\>s[Hro.|p(ڹ ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( +.6=/v3[I :;;|sC+W|AeOxßWWVvrju`ܶ +P@X((((ZT.]NDY.iX/,II85QEQEQEwu" #Yz~bv9k?FYQK;@.i$a$f Z(+xY׼e Nϵ yHRy e@'^@EPEPEy&SR_^Z9Qwp‚ pJ >yeOhӺ!e8 7@ONHPQ\?|,q:~Ex=~)w^:͊Z뺂)6Š!z5jjOоoH2xQ2}Mr~'5ۏO]]akh\HPHRNn*XP/^9!w*p~&𖟨i5XaFaǜ18#+CX#ji:Jt1ظ >7GssxVA~=ٳ+Hqѱw& y c x-h%@WR2#5|SWEeջ,^c2>hؕž:w Z飃^*(ftIC,Lc+E-wy 5Uφ6PrD1v$ʇc$yH(_Λ^)"X0 9?IaY?TTxž%gYB {xjJt!zoE @&IaY?TU_h{Ⱦnݰ.t $yb+;1@-.zg.dm͖Q_# 7a-e OMK6sy˥\Cmh.&݌1+ 9T\l=&@- ' H ,n-?g?M<{#hwFO>daϧ(>$D+/}poy%ĒIݷP!׶Ÿi^$ק|'Kh] `ǴpXM |lo~XX[%݃ʗPy]#Щ@h~9uN'0 U‚$p|D!ZRZW[i`!"ed/$7dp ~$qC^}_M["m9(#l7Mܚ}6KZ\*āޠV O⫫x-/3A*H̬dD s@>:W.;.[+v7f92[q&k,͡i~&ݝ!`AH b<9Dk'/˂#<{J3J*Ac,:fN A |?f;Mo[\\@Vl}xoZowTI\ΥQ;a0Rz (V?|TkvI-g3ܶē Ђ7X|$|OU4'I - IC: ˩GLAIBv8`pٯ3uZ\&𧊮MY4>ض${eRF𮯪[m=,X2=EIj~5o'uW>V7m3T_iw "{o%$dC@wFj0V,H@ gh pYk?;ocV'2In7}8^7ĩo#Uh-5}"XcP҂;XIvרjz_U/v.83P?"xWڌ32}XN]@O]յ(h iqjiͦ;h0m,?*GaY~?i_ZAik!1.E8Q$5'',q! v 0U,@ PWյ-gw}:r"u3J|bN2IǹL4<[Xii;QA,p98'jEE@>jږ㯈Kh晝a_6qF AQJ?oQc_SD)&~Lx8 {N[(*?mGk<:;,vi; LBqЖ@Hu_|g&7`ɦD&X(i-vC²訫5]VC5=N;k;t,G$$I+jEE@s?|C}6Z|gՕk2bt VS0g7ë J.s))*|i#8ڹ<3⫫ym>*k3A*9VWR0A|GP޷50ya/-R F|ɸu*SjW7 x#Lm"GxP* a?\ (<???6޽???6޽ Q6gmh+oc[ƚo _YTv02n$ a#v7{x<:vxbX֬9^{BTG !qǨ9?92cF&;{9w>(Pwm R}su_\\K$˦[ \\D+Ƴi8lgE.RZGia,mυ兜~]q )0QuOVatk.X/m䷑ 0WR3h>*]w:GtOj^.Xgv+*, H?*tl|w~GoIk涻w3A@ 30]˒ vݟ|9iI<< r@ձ@]*\4N;;,OчG F'n_t8<oj6xzko$ngr 9`#mmⷷ8`GjQ@TQ@~xs%%^۞[ BCÁP7+A+:7`9?|9QHM\>]w8P3s3Q@wAӠi7[NvxeRG(#Kk:VFtR%~6`0|8/+\i??o'տ~w+|6;c8=KA ]Zo6,!ː5Q@G+눗~8u>'KHm,8a߬q_W~]ϛyX"sQ@㟄_5kv-/w :R8!@qԯk_DȘ6# ,[{鷮x( e!t&Ȳ(Ǔqhh Ix#xR/^a!eV?t{{×ze>u6͗g|Xuc2@#[P| ׵ [߉:I9Bfʜ9һx^5K]%t25Q@x>->ydq'z+^TTNɲ[q+3#^ =+(o^Լ/Mex]U s*O_,EK= :̥Ċ2%*X}6mºFpѴVPђT RF@8_௄<-& Wwq:}*yUUPO PFD((((((((((7xGQ凇nyd%ڱ˃@u&N+V7ҥGo(2:X&&c =*?k㯆 jlI#TQ-$s/#+OoiOc" H"ǯa^|}@ѤKycԲ1Pr0rP < WY^?E./oxYmbzoPH9uzKkQsh9 _rĀ1Nwg65*F"!.'eeQYw/M_Ot쵭ȼXb$$Q z7u>!M=&AI۩H?k]go4dZ\çZ*3~-H r@iExӾ<]ͨ gJȺhHK )ysd5|5W:;-kG/,m`y#A7~H-6cTVePU_G;ῑu IV;,cKQA# PBh^ 0OZIo*鴖_g@='Ưi垻V$y@BRactEqx3%wn/1"'?~*ëH8PnPmB Ua˸z\n{H"  ! GJM'RYѬuKu`5]CpH ~*玓wKW -»DbIq(1WQ@/q3/ƞ tIɇQH>Č7Sӭ-nI}nFdָƗ|5Նb֖[NJxJ鷎p 5Dӯ˺XW y`@!pA¹xx:YseoL A#?1j+_I~hS]C?0M9^w6f#>3px uC'=ܗycCrT\ ۹+O<{O x/_f o!E,;pf?u(M]KǞ5H-!+׷UFU*'s1\qyVO{x~SM>Ҵ]u2@)hةޣ<瞸 jSk>5KgE2IOOow:< $_ K@Q@'e⿉^K- E<}쫫|Kec :n[ ( rGWu"l`|CuvDD00tb!I"N02徣w^RhӐ$Xm'N 85-Nj.U޻Jֺ,7 S9\@ +?i[3ZO (y0r2y`((((((((((((((((((6X^GZj ɸ[zV#]Ž&v鈮Y Mw!b= hxZѦΛ5n).|<Ʃ}x\5MLZƈ\"ds@9='CG?+}we`/K>f&J,C ˯JVc\>m]B2n1OQjx_43a~<ַh{w=Ǫ |v ==#7G>*l`ųWU,xڳƍ5 t٭#vIve,H,^9@pw 5K[Wz桨:od4BQ'h# 1]w5B$,!w A`]X`N7X>E#R-p \p7L|`{du!_k K{io-9 >RIS ,$?/5h˰5)nf*dl PxgJ>hZfm͝ƏjyI#@#W7hh?minRilyx9IOM{Yemc!WRpH8#;?s_ř<ۣ$ ï>%|\P6"W !  4hڕAIkzwc{] 2;vG(pk6 ^]kzƯ>ؖ)v! /ʏ¨y^\xQc [|eamĶ8@$cCV\^xC'd܌aoOJ~4H$w Jһõqq7檓Y_-,3i˸0 eR"HolMzãh:]HY[oHAbA8gP?sjU|*/+Ȓ@?_ |eˬYixia!7lfL=O xcTC_hhn{y_\j:cl;/' v_s="mW};\]I:dg#S|V6e6ϳ2-7m`ÕZHF- t vg \iZƷ ;@7H2*qZ叇=FZE*͸rA9<"|DUV7Kv`9۞w+Ƒy/W7Y}G!#nWnMC?Jk#џVd(o.{1!lnz#"?G9ߕz G&/diM,}oi7yv<>vXrp^_^u{.Z4l ep(ʃzQ@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@Q@'MѭK,`g.±)l3*PEPEPEPEP{ =NK;H.dgHʞ "@TP0%QEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEQEPKPp yyPK.AOEBPS/dcommon/contbig.gif`GIF87a!!!111999BBBJJJRRRccckkksss{{{skk{{ZRRRJJƽ{sZRJRJB91)kcZB9)sskZRJ1޽ƽ{{ssskkkcƵZZRccZRRJJJB{BB9991ssckkZccR))!RRB!!JJ1))99!11ƌ)1R)k֔)s1RZJR{BJs9R1J!11J1J9k{csZk!1J!)cBR9J1B)91B!cRs{!)s!){1B!k!s!{ksksckckZc9B)1!)!)BJ9B1919έƌ!!)JJcZZ{!!!1RR{JJsBBkJJ{!!9BB{1!!J9)!!Z!!c1!!kR!!s9Z!BckJs)19!!c!!ZRZ,H rrxB(Kh" DժuICiи@S z$G3TTʖ&7!f b`D 0!A  k,>SO[!\ *_t  Exr%*_}!#U #4 & ֩3|b]L ]t b+Da&R_2lEٱZ`aC)/яmvUkS r(-iPE Vv_{z GLt\2s!F A#葡JY r|AA,hB}q|B`du }00(䡆<pb,G+oB C0p/x$…– ]7 @2HFc ) @AD \0 LHG',(A` `@SC)_" PH`}Y+_|1.K8pAKMA @?3҄$[JPA)+NH I ,@8G0/@R T,`pF8Ѓ)$^$ DDTDlA@ s;PKPK.AOEBPS/dcommon/darbbook.cssPKPK.A!OEBPS/dcommon/O_signature_clr.JPG"(JFIF``C    $.' ",#(7),01444'9=82<.342C  2!!22222222222222222222222222222222222222222222222222" }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz w!1AQaq"2B #3Rbr $4%&'()*56789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz ?( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( ( (?O '~MQ$Vz;OlJi8L%\]UFjޙ%ԯS;rA]5ފ<׈]j7Ouyq$z'TQuw7Ŀ KX߁M2=S'TQt?.5w'97;~pq=" ~k?`'9q6 E|yayM^Om'fkC&<5x' ?A?Zx'jß={=SßM gVC.5+Hd֪xc^)Җufz{Cީ|D Vkznq|+Xa+{50rx{|OG.OϞ~f/ xxX[2H )c+#jpUOZYX\=SG ߨC|K@;_߆'e?LT?]:?>w ڔ`D^So~xo[Ӡ3i7B:Q8 Vc-ďoi:FM292~y_*_闱YN\Fr=xZ3鳎OwW_QEzW~c]REeaSM}}Hӏ4&.E]u=gMѠ+mF`rNn$w9gMa꺢nTuhf2Xv>އ a(Û6߭?<=>z'TQuw7Ŀ KX߁M2=S'TQt?.5Kko\.8S$TOX߀Gw?Zx汴X)C7~.i6(Щ=+4{mGӭ¸-]&'t_kV*I<1)4thtIsqpQJ+> \m^[aJ5)ny:4o&QEnyAEPEEss 72,PDۢ׃K W{Wjr+wگ iM/;pd?~&?@;7E4gv8 $l'z'TQuw7Ŀ Gֱ=ɿ&G?. iR(5W*$|?w᫼gkmIbHe/_t>tg%y.l}N5[]+Mk0ĠeHdPrsst'UiC,y8`V%9ZIia|ܪvi מYG,o}+kk{YbyIeb*sAtի82zWoEK5z*o-eo;n(P u-I)4Š(HQEQEQEQEhz(X/Đ?}Bk˩ ݏrk0]4>8XzV? }6$}d^F>nU K ?Bտk_9׾x~w'ߞ  uDŽtL ؈5c-E/"|_Oo.IH쐍=i*Iw5(ںw?t5s.)+tQ2dUt5Vĺ.jZ"@IRrZƅY4ߡ_;}ų(KyQf1Aǵt?sZg+?F5_oQR&Dg߿]6FuRD u>ڿxl7?IT8'shj^=.=J1rj1Wl$얲cPx;E,p$֟ˏkw qg"45(ǛkV/=+ũ)bYl~K#˝J_כ5&\F'I#8/|wʾ_Xj Q:os^T1.M_|TO.;?_  jF?g N 8nA2F%i =qW,G=5OU u8]Rq?wr'˻S+۾.ܼ 87Q^elo/T*?L|ۚ<%<,/v_OKs B5f/29n0=zqQq(ª=VX@*J(э(f5qJN_EVǞQEOuoѕOuoa5}gO?:߂8Wא|cڽ~]N&O( (<]>͠@VQ=^~U ̴m&\խ5i:}|}r~9՝f}_>'vVֲ$~^f30^in{\_.O F8to}?${φ|#x^#^n~w=~k~?'KRtO.㌡h![3Zu*ٷճ(ԟ]z_/W1(ԟ]v~g|Yq<ז0 ; b8֮s,w9\?uEyStKaª@\,)) (!EPEPEPEPEPzѧts{v>C/"N6`d*J2gGӧWqBq_1ZuΓ\X]r?=Ey88Mp&pKtO-"wR2 K^-Z< \c>V0^@O7x2WFjs<׻kZ(<Т(OFw/6$1[:ޯԯ#q~4|,LVPem=@=YLUxӃV}AUbcUB.Ds5*kٸAeG>PJxt͝ b88?*$~@ׯD VkraiJs}Q.20x&mXξ,Z]“A-J#`+-E/"<]\a'tZGy.(|lދ~gMK OZdxDŽU9T6ϯ^<Ϡt5CZ]].t۫S=s`ڳ%8iVK:nqe+#<.T6U>zWoy3^I {F?J~=G}k)K$$;$de8*G Uӟ4Ocºw}|]4=ݣ\x$ʠms?q^ipw\"ȿPs^Z Q_0GڼU.t}ROM[G#]8wٞ ӫ87}Cgw vHȩBM55vof =A_٭`Ygx[6 P,5}>蚊(0(+?>+?> k|TuXq6_ +szk :u_ Z߶Ak_U}Jc2u/1[_»ݸG41-bሬ۴}}Eȹפ_c?5gi @cL\L<68hF_Ih>X4K7UТ sMj =J7CKo>Օ5s:߀t ~ηaٿ?|gdL8+gG%o?x`دOqȱwc¨&TW_V_aI=dpG!wu۞սZ1yL50$(l3(:~'ַo A}a3N*[0ǭ HKQV}G@֜$ 9of$ArNqUOgË05#m?D)^_h//5_/<?4}Jį+GkpG4"$ r| >S4Ђ"S 1%R:ȝ 8;PKPz PK.AOEBPS/dcommon/feedback.gif7GIF89a'%(hp|fdx?AN5:dfeDGHɾTdQc`g*6DC\?ؘ||{;=E6JUՄfeA= >@,4`H.|`a (Q 9:&[|ځ,4p Y&BDb,!2@, $wPA'ܠǃ@CO~/d.`I @8ArHx9H75j L 3B/` P#qD*s 3A:3,H70P,R@ p!(F oԥ D;"0 ,6QBRɄHhI@@VDLCk8@NBBL2&pClA?DAk%$`I2 #Q+l7 "=&dL&PRSLIP)PɼirqМ'N8[_}w;PK-PK.AOEBPS/dcommon/booklist.gifGIF89a1޵֥΄kZ{Jk1Rs!BZ)B),@I9Z͓Ca % Dz8Ȁ0FZЌ0P !x8!eL8aWȠFD(~@p+rMS|ӛR$ v "Z:]ZJJEc{*=AP  BiA ']j4$*   & 9q sMiO?jQ = , YFg4.778c&$c%9;PKː5PK.AOEBPS/dcommon/cpyr.htm1 Oracle Legal Notices

Oracle Legal Notices

Copyright Notice

Copyright © 1994-2012, Oracle and/or its affiliates. All rights reserved.

Trademark Notice

Oracle and Java are registered trademarks of Oracle and/or its affiliates. Other names may be trademarks of their respective owners.

Intel and Intel Xeon are trademarks or registered trademarks of Intel Corporation. All SPARC trademarks are used under license and are trademarks or registered trademarks of SPARC International, Inc. AMD, Opteron, the AMD logo, and the AMD Opteron logo are trademarks or registered trademarks of Advanced Micro Devices. UNIX is a registered trademark of The Open Group.

License Restrictions Warranty/Consequential Damages Disclaimer

This software and related documentation are provided under a license agreement containing restrictions on use and disclosure and are protected by intellectual property laws. Except as expressly permitted in your license agreement or allowed by law, you may not use, copy, reproduce, translate, broadcast, modify, license, transmit, distribute, exhibit, perform, publish, or display any part, in any form, or by any means. Reverse engineering, disassembly, or decompilation of this software, unless required by law for interoperability, is prohibited.

Warranty Disclaimer

The information contained herein is subject to change without notice and is not warranted to be error-free. If you find any errors, please report them to us in writing.

Restricted Rights Notice

If this is software or related documentation that is delivered to the U.S. Government or anyone licensing it on behalf of the U.S. Government, the following notice is applicable:

U.S. GOVERNMENT RIGHTS Programs, software, databases, and related documentation and technical data delivered to U.S. Government customers are "commercial computer software" or "commercial technical data" pursuant to the applicable Federal Acquisition Regulation and agency-specific supplemental regulations. As such, the use, duplication, disclosure, modification, and adaptation shall be subject to the restrictions and license terms set forth in the applicable Government contract, and, to the extent applicable by the terms of the Government contract, the additional rights set forth in FAR 52.227-19, Commercial Computer Software License (December 2007). Oracle America, Inc., 500 Oracle Parkway, Redwood City, CA 94065.

Hazardous Applications Notice

This software or hardware is developed for general use in a variety of information management applications. It is not developed or intended for use in any inherently dangerous applications, including applications that may create a risk of personal injury. If you use this software or hardware in dangerous applications, then you shall be responsible to take all appropriate fail-safe, backup, redundancy, and other measures to ensure its safe use. Oracle Corporation and its affiliates disclaim any liability for any damages caused by use of this software or hardware in dangerous applications.

Third-Party Content, Products, and Services Disclaimer

This software or hardware and documentation may provide access to or information on content, products, and services from third parties. Oracle Corporation and its affiliates are not responsible for and expressly disclaim all warranties of any kind with respect to third-party content, products, and services. Oracle Corporation and its affiliates will not be responsible for any loss, costs, or damages incurred due to your access to or use of third-party content, products, or services.

Alpha and Beta Draft Documentation Notice

If this document is in prerelease status:

This documentation is in prerelease status and is intended for demonstration and preliminary use only. It may not be specific to the hardware on which you are using the software. Oracle Corporation and its affiliates are not responsible for and expressly disclaim all warranties of any kind with respect to this documentation and will not be responsible for any loss, costs, or damages incurred due to the use of this documentation.

Oracle Logo

PKN61PK.AOEBPS/dcommon/masterix.gif.GIF89a1ޜΌscJk1Rs!Bc1J),@IS@0"1 Ѿb$b08PbL,acr B@(fDn Jx11+\%1 p { display: none; } /* Class Selectors */ .ProductTitle { font-family: sans-serif; } .BookTitle { font-family: sans-serif; } .VersionNumber { font-family: sans-serif; } .PrintDate { font-family: sans-serif; font-size: small; } .PartNumber { font-family: sans-serif; font-size: small; } PKeӺ1,PK.AOEBPS/dcommon/larrow.gif#GIF87a絵ƌֵƽ{{ss֜ƔZZ{{{{ZZssZZccJJJJRRBBJJJJ991111))!!{,@pH,Ȥrl:ШtpHc`  өb[.64ꑈ53=Z]'yuLG*)g^!8C?-6(29K"Ĩ0Яl;U+K9^u2,@@ (\Ȱ Ë $P`lj 8x I$4H *(@͉0dа8tA  DсSP v"TUH PhP"Y1bxDǕ̧_=$I /& .)+ 60D)bB~=0#'& *D+l1MG CL1&+D`.1qVG ( "D2QL,p.;u. |r$p+5qBNl<TzB"\9e0u )@D,¹ 2@C~KU 'L6a9 /;<`P!D#Tal6XTYhn[p]݅ 7}B a&AƮe{EɲƮiEp#G}D#xTIzGFǂEc^q}) Y# (tۮNeGL*@/%UB:&k0{ &SdDnBQ^("@q #` @1B4i@ aNȅ@[\B >e007V[N(vpyFe Gb/&|aHZj@""~ӎ)t ? $ EQ.սJ$C,l]A `8A o B C?8cyA @Nz|`:`~7-G|yQ AqA6OzPbZ`>~#8=./edGA2nrBYR@ W h'j4p'!k 00 MT RNF6̙ m` (7%ꑀ;PKl-OJPK.AOEBPS/dcommon/index.gifGIF89a1޵ΥΥ{sc{BZs,@IM" AD B0 3.R~[D"0, ]ШpRNC  /& H&[%7TM/`vS+-+ q D go@" 4o'Uxcxcc&k/ qp zUm(UHDDJBGMԃ;PK(PK.AOEBPS/dcommon/bookbig.gif +GIF89a$!!!)))111999BBBJJJRRRZZZccckkksss{{{skkB991)))!!B11))1!JB9B9!!cZ9ƭƽssk{ZZRccZRRJJJBBB9c!!ν)1)k{s絽ƌkssֽZccJRRBJJ{9BB)11)99!!))11!!k!JZ!)RcJccBcs)1c)JZ!BR!)BZ)99J!Rk9!c11B)Z{)9Bkc1kB9BZ!Z{9Rs)Jkksk9kB1s1Jk9Rƥc{k9s)Z{1k91)s1Rk)Jc1J!))BZ!1k{csc{)19B!)Bcsc{ksc{kZs!RkJkJkքc{9Zks{ck9R)Bks9R9R1J!)Z1B!)c)9)99BR19kksBBJcc{ccBBZ))9kk!!199c11ZBB{9!!R!!Z!!c))!!kR!!s!!BcksRZ1c9B)R91c1)Z!R9B9k1)RcZ{)!1B9JB9B)!)J9B!& Imported from GIF image: bookbig.gif,$!!!)))111999BBBJJJRRRZZZccckkksss{{{skkB991)))!!B11))1!JB9B9!!cZ9ƭƽssk{ZZRccZRRJJJBBB9c!!ν)1)k{s絽ƌkssֽZccJRRBJJ{9BB)11)99!!))11!!k!JZ!)RcJccBcs)1c)JZ!BR!)BZ)99J!Rk9!c11B)Z{)9Bkc1kB9BZ!Z{9Rs)Jkksk9kB1s1Jk9Rƥc{k9s)Z{1k91)s1Rk)Jc1J!))BZ!1k{csc{)19B!)Bcsc{ksc{kZs!RkJkJkքc{9Zks{ck9R)Bks9R9R1J!)Z1B!)c)9)99BR19kksBBJcc{ccBBZ))9kk!!199c11ZBB{9!!R!!Z!!c))!!kR!!s!!BcksRZ1c9B)R91c1)Z!R9B9k1)RcZ{)!1B9JB9B)!)J9BH`\Ȑ:pظа"A6DBH,V@Dڹ'G"v Æ ܥ;n;!;>xAܽ[G.\rQC wr}BŊQ A9ᾑ#5Y0VȒj0l-GqF>ZpM rb ;=.ސW-WѻWo ha!}~ْ ; t 53 :\ 4PcD,0 4*_l0K3-`l.j!c Aa|2L4/1C`@@md;(H*80L0L(h*҇҆o#N84pC (xO@ A)J6rVlF r  fry†$r_pl5xhA+@A=F rGU a 1х4s&H Bdzt x#H%Rr (Ѐ7P`#Rщ'x" #0`@~i `HA'Tk?3!$`-A@1l"P LhʖRG&8A`0DcBH sq@AXB4@&yQhPAppxCQ(rBW00@DP1E?@lP1%T` 0 WB~nQ@;PKGC PK.AOEBPS/dcommon/rarrow.gif/GIF87a絵ƌֵƽ{{ss֜ƔZZ{{{{ZZssZZccJJJJRRBBJJJJ991111))!!{,@pH,Ȥrl:ШLlԸ NCqWEd)#34vwwpN|0yhX!'+-[F 'n5 H $/14w3% C .90" qF 7&E "D mnB|,c96) I @0BW{ᢦdN p!5"D`0 T 0-]ʜ$;PKJV^PK.AOEBPS/dcommon/mix.gifkGIF89aZZZBBBJJJkkk999sss!!!111cccֽ{{{RRR)))猌ƭ{s{sks!,@@pH,B$ 8 t:<8 *'ntPP DQ@rIBJLNPTVEMOQUWfj^!  hhG H  kCúk_a Ǥ^ h`B BeH mm  #F` I lpǎ,p B J\Y!T\(dǏ!Gdˆ R53ټ R;iʲ)G=@-xn.4Y BuU(*BL0PX v`[D! | >!/;xP` (Jj"M6 ;PK枰pkPK.AOEBPS/dcommon/doccd_epub.jsM /* Copyright 2006, 2012, Oracle and/or its affiliates. All rights reserved. Author: Robert Crews Version: 2012.3.17 */ function addLoadEvent(func) { var oldOnload = window.onload; if (typeof(window.onload) != "function") window.onload = func; else window.onload = function() { oldOnload(); func(); } } function compactLists() { var lists = []; var ul = document.getElementsByTagName("ul"); for (var i = 0; i < ul.length; i++) lists.push(ul[i]); var ol = document.getElementsByTagName("ol"); for (var i = 0; i < ol.length; i++) lists.push(ol[i]); for (var i = 0; i < lists.length; i++) { var collapsible = true, c = []; var li = lists[i].getElementsByTagName("li"); for (var j = 0; j < li.length; j++) { var p = li[j].getElementsByTagName("p"); if (p.length > 1) collapsible = false; for (var k = 0; k < p.length; k++) { if ( getTextContent(p[k]).split(" ").length > 12 ) collapsible = false; c.push(p[k]); } } if (collapsible) { for (var j = 0; j < c.length; j++) { c[j].style.margin = "0"; } } } function getTextContent(e) { if (e.textContent) return e.textContent; if (e.innerText) return e.innerText; } } addLoadEvent(compactLists); function processIndex() { try { if (!/\/index.htm(?:|#.*)$/.test(window.location.href)) return false; } catch(e) {} var shortcut = []; lastPrefix = ""; var dd = document.getElementsByTagName("dd"); for (var i = 0; i < dd.length; i++) { if (dd[i].className != 'l1ix') continue; var prefix = getTextContent(dd[i]).substring(0, 2).toUpperCase(); if (!prefix.match(/^([A-Z0-9]{2})/)) continue; if (prefix == lastPrefix) continue; dd[i].id = prefix; var s = document.createElement("a"); s.href = "#" + prefix; s.appendChild(document.createTextNode(prefix)); shortcut.push(s); lastPrefix = prefix; } var h2 = document.getElementsByTagName("h2"); for (var i = 0; i < h2.length; i++) { var nav = document.createElement("div"); nav.style.position = "relative"; nav.style.top = "-1.5ex"; nav.style.left = "1.5em"; nav.style.width = "90%"; while (shortcut[0] && shortcut[0].toString().charAt(shortcut[0].toString().length - 2) == getTextContent(h2[i])) { nav.appendChild(shortcut.shift()); nav.appendChild(document.createTextNode("\u00A0 ")); } h2[i].parentNode.insertBefore(nav, h2[i].nextSibling); } function getTextContent(e) { if (e.textContent) return e.textContent; if (e.innerText) return e.innerText; } } addLoadEvent(processIndex); PKo"nR M PK.AOEBPS/dcommon/toc.gifGIF89a1ΥΥ{c{Z{JkJk1Rk,@IK% 0| eJB,K-1i']Bt9dz0&pZ1o'q(؟dQ=3S SZC8db f&3v2@VPsuk2Gsiw`"IzE%< C !.hC IQ 3o?39T ҍ;PKv I PK.AOEBPS/dcommon/topnav.gifGIF89a1ֽ筽ޭƔkZZk{Bc{,@ ) l)-'KR$&84 SI) XF P8te NRtHPp;Q%Q@'#rR4P fSQ o0MX[) v + `i9gda/&L9i*1$#"%+ ( E' n7Ȇ(,҅(L@(Q$\x 8=6 'נ9tJ&"[Epljt p#ѣHb :f F`A =l|;&9lDP2ncH R `qtp!dȐYH›+?$4mBA9 i@@ ]@ꃤFxAD*^Ŵ#,(ε  $H}F.xf,BD Z;PK1FAPK.AOEBPS/dcommon/bp_layout.css# @charset "utf-8"; /* bp_layout.css Copyright 2007, Oracle and/or its affiliates. All rights reserved. */ body { margin: 0ex; padding: 0ex; } h1 { display: none; } #FOOTER { border-top: #0d4988 solid 10px; background-color: inherit; color: #e4edf3; clear: both; } #FOOTER p { font-size: 80%; margin-top: 0em; margin-left: 1em; } #FOOTER a { background-color: inherit; color: gray; } #LEFTCOLUMN { float: left; width: 50%; } #RIGHTCOLUMN { float: right; width: 50%; clear: right; /* IE hack */ } #LEFTCOLUMN div.portlet { margin-left: 2ex; margin-right: 1ex; } #RIGHTCOLUMN div.portlet { margin-left: 1ex; margin-right: 2ex; } div.portlet { margin: 2ex 1ex; padding-left: 0.5em; padding-right: 0.5em; border: 1px #bcc solid; background-color: #f6f6ff; color: black; } div.portlet h2 { margin-top: 0.5ex; margin-bottom: 0ex; font-size: 110%; } div.portlet p { margin-top: 0ex; } div.portlet ul { list-style-type: none; padding-left: 0em; margin-left: 0em; /* IE Hack */ } div.portlet li { text-align: right; } div.portlet li cite { font-style: normal; float: left; } div.portlet li a { margin: 0px 0.2ex; padding: 0px 0.2ex; font-size: 95%; } #NAME { margin: 0em; padding: 0em; position: relative; top: 0.6ex; left: 10px; width: 80%; } #PRODUCT { font-size: 180%; } #LIBRARY { color: #0b3d73; background: inherit; font-size: 180%; font-family: serif; } #RELEASE { position: absolute; top: 28px; font-size: 80%; font-weight: bold; } #TOOLS { list-style-type: none; position: absolute; top: 1ex; right: 2em; margin: 0em; padding: 0em; background: inherit; color: black; } #TOOLS a { background: inherit; color: black; } #NAV { float: left; width: 96%; margin: 3ex 0em 0ex 0em; padding: 2ex 0em 0ex 4%; /* Avoiding horizontal scroll bars. */ list-style-type: none; background: transparent url(../gifs/nav_bg.gif) repeat-x bottom; } #NAV li { float: left; margin: 0ex 0.1em 0ex 0em; padding: 0ex 0em 0ex 0em; } #NAV li a { display: block; margin: 0em; padding: 3px 0.7em; border-top: 1px solid gray; border-right: 1px solid gray; border-bottom: none; border-left: 1px solid gray; background-color: #a6b3c8; color: #333; } #SUBNAV { float: right; width: 96%; margin: 0ex 0em 0ex 0em; padding: 0.1ex 4% 0.2ex 0em; /* Avoiding horizontal scroll bars. */ list-style-type: none; background-color: #0d4988; color: #e4edf3; } #SUBNAV li { float: right; } #SUBNAV li a { display: block; margin: 0em; padding: 0ex 0.5em; background-color: inherit; color: #e4edf3; } #SIMPLESEARCH { position: absolute; top: 5ex; right: 1em; } #CONTENT { clear: both; } #NAV a:hover, #PORTAL_1 #OVERVIEW a, #PORTAL_2 #OVERVIEW a, #PORTAL_3 #OVERVIEW a, #PORTAL_4 #ADMINISTRATION a, #PORTAL_5 #DEVELOPMENT a, #PORTAL_6 #DEVELOPMENT a, #PORTAL_7 #DEVELOPMENT a, #PORTAL_11 #INSTALLATION a, #PORTAL_15 #ADMINISTRATION a, #PORTAL_16 #ADMINISTRATION a { background-color: #0d4988; color: #e4edf3; padding-bottom: 4px; border-color: gray; } #SUBNAV a:hover, #PORTAL_2 #SEARCH a, #PORTAL_3 #BOOKS a, #PORTAL_6 #WAREHOUSING a, #PORTAL_7 #UNSTRUCTURED a, #PORTAL_15 #INTEGRATION a, #PORTAL_16 #GRID a { position: relative; top: 2px; background-color: white; color: #0a4e89; } PK3( # PK.AOEBPS/dcommon/bookicon.gif:GIF87a!!!)))111999BBBJJJRRRZZZccckkksss{{{ޭ{{ZRRcZZRJJJBB)!!skRB9{sν{skskcZRJ1)!֭ƽ{ZZRccZJJBBB999111)JJ9BB1ZZB!!ﭵBJJ9BB!!))Jk{)1!)BRZJ{BsR!RRJsJ!J{s!JsBkks{RsB{J{c1RBs1ZB{9BJ9JZ!1BJRRs!9R!!9Z9!1)J19JJRk19R1Z)!1B9R1RB!)J!J1R)J119!9J91!9BkksBBJ119BBR!))9!!!JB1JJ!)19BJRZckތ1)1J9B,H*\hp >"p`ƒFF "a"E|ժOC&xCRz OBtX>XE*O>tdqAJ +,WxP!CYpQ HQzDHP)T njJM2ꔀJ2T0d#+I:<жk 'ꤱF AB @@nh Wz' H|-7f\A#yNR5 /PM09u UjćT|q~Yq@&0YZAPa`EzI /$AD Al!AAal 2H@$ PVAB&c*ؠ p @% p-`@b`uBa l&`3Ap8槖X~ vX$Eh`.JhAepA\"Bl, :Hk;PKx[?:PK.AOEBPS/dcommon/conticon.gif^GIF87a!!!)))111999BBBJJJRRRZZZccckkksss{{{ZRR޽{{ssskkkcccZ991ccRZZBBJJZck)19ZcsBJZ19J!k{k)Z1RZs1!B)!J91{k{)J!B!B911)k{cs!1s!9)s!9!B!k)k1c!)Z!R{9BJcckZZcBBJ99B119{{!!)BBRBBZ!))999R99Z!!999c1!9!)19B1)!B9R,  oua\h2SYPa aowwxYi 9SwyyxxyYSd $'^qYȵYvh ч,/?g{н.J5fe{ڶyY#%/}‚e,Z|pAܠ `KYx,ĉ&@iX9|`p ]lR1khٜ'E 6ÅB0J;t X b RP(*MÄ!2cLhPC <0Ⴁ  $4!B 6lHC%<1e H 4p" L`P!/,m*1F`#D0D^!AO@..(``_؅QWK>_*OY0J@pw'tVh;PKp*c^PK.AOEBPS/dcommon/blafdoc.cssL@charset "utf-8"; /* Copyright 2002, 2011, Oracle and/or its affiliates. All rights reserved. Author: Robert Crews Version: 2011.10.7 */ body { font-family: Tahoma, sans-serif; /* line-height: 125%; */ color: black; background-color: white; font-size: small; } * html body { /* http://www.info.com.ph/~etan/w3pantheon/style/modifiedsbmh.html */ font-size: x-small; /* for IE5.x/win */ f\ont-size: small; /* for other IE versions */ } h1 { font-size: 165%; font-weight: bold; border-bottom: 1px solid #ddd; width: 100%; } h2 { font-size: 152%; font-weight: bold; } h3 { font-size: 139%; font-weight: bold; } h4 { font-size: 126%; font-weight: bold; } h5 { font-size: 113%; font-weight: bold; display: inline; } h6 { font-size: 100%; font-weight: bold; font-style: italic; display: inline; } a:link { color: #039; background: inherit; } a:visited { color: #72007C; background: inherit; } a:hover { text-decoration: underline; } a img, img[usemap] { border-style: none; } code, pre, samp, tt { font-family: monospace; font-size: 110%; } caption { text-align: center; font-weight: bold; width: auto; } dt { font-weight: bold; } table { font-size: small; /* for ICEBrowser */ } td { vertical-align: top; } th { font-weight: bold; text-align: left; vertical-align: bottom; } ol ol { list-style-type: lower-alpha; } ol ol ol { list-style-type: lower-roman; } td p:first-child, td pre:first-child { margin-top: 0px; margin-bottom: 0px; } table.table-border { border-collapse: collapse; border-top: 1px solid #ccc; border-left: 1px solid #ccc; } table.table-border th { padding: 0.5ex 0.25em; color: black; background-color: #f7f7ea; border-right: 1px solid #ccc; border-bottom: 1px solid #ccc; } table.table-border td { padding: 0.5ex 0.25em; border-right: 1px solid #ccc; border-bottom: 1px solid #ccc; } span.gui-object, span.gui-object-action { font-weight: bold; } span.gui-object-title { } p.horizontal-rule { width: 100%; border: solid #cc9; border-width: 0px 0px 1px 0px; margin-bottom: 4ex; } div.zz-skip-header { display: none; } td.zz-nav-header-cell { text-align: left; font-size: 95%; width: 99%; color: black; background: inherit; font-weight: normal; vertical-align: top; margin-top: 0ex; padding-top: 0ex; } a.zz-nav-header-link { font-size: 95%; } td.zz-nav-button-cell { white-space: nowrap; text-align: center; width: 1%; vertical-align: top; padding-left: 4px; padding-right: 4px; margin-top: 0ex; padding-top: 0ex; } a.zz-nav-button-link { font-size: 90%; } div.zz-nav-footer-menu { width: 100%; text-align: center; margin-top: 2ex; margin-bottom: 4ex; } p.zz-legal-notice, a.zz-legal-notice-link { font-size: 85%; /* display: none; */ /* Uncomment to hide legal notice */ } /*************************************/ /* Begin DARB Formats */ /*************************************/ .bold, .codeinlinebold, .syntaxinlinebold, .term, .glossterm, .seghead, .glossaryterm, .keyword, .msg, .msgexplankw, .msgactionkw, .notep1, .xreftitlebold { font-weight: bold; } .italic, .codeinlineitalic, .syntaxinlineitalic, .variable, .xreftitleitalic { font-style: italic; } .bolditalic, .codeinlineboldital, .syntaxinlineboldital, .titleinfigure, .titleinexample, .titleintable, .titleinequation, .xreftitleboldital { font-weight: bold; font-style: italic; } .itemizedlisttitle, .orderedlisttitle, .segmentedlisttitle, .variablelisttitle { font-weight: bold; } .bridgehead, .titleinrefsubsect3 { font-weight: bold; } .titleinrefsubsect { font-size: 126%; font-weight: bold; } .titleinrefsubsect2 { font-size: 113%; font-weight: bold; } .subhead1 { display: block; font-size: 139%; font-weight: bold; } .subhead2 { display: block; font-weight: bold; } .subhead3 { font-weight: bold; } .underline { text-decoration: underline; } .superscript { vertical-align: super; } .subscript { vertical-align: sub; } .listofeft { border: none; } .betadraft, .alphabetanotice, .revenuerecognitionnotice { color: #e00; background: inherit; } .betadraftsubtitle { text-align: center; font-weight: bold; color: #e00; background: inherit; } .comment { color: #080; background: inherit; font-weight: bold; } .copyrightlogo { text-align: center; font-size: 85%; } .tocsubheader { list-style-type: none; } table.icons td { padding-left: 6px; padding-right: 6px; } .l1ix dd, dd dl.l2ix, dd dl.l3ix { margin-top: 0ex; margin-bottom: 0ex; } div.infoboxnote, div.infoboxnotewarn, div.infoboxnotealso { margin-top: 4ex; margin-right: 10%; margin-left: 10%; margin-bottom: 4ex; padding: 0.25em; border-top: 1pt solid gray; border-bottom: 1pt solid gray; } p.notep1 { margin-top: 0px; margin-bottom: 0px; } .tahiti-highlight-example { background: #ff9; text-decoration: inherit; } .tahiti-highlight-search { background: #9cf; text-decoration: inherit; } .tahiti-sidebar-heading { font-size: 110%; margin-bottom: 0px; padding-bottom: 0px; } /*************************************/ /* End DARB Formats */ /*************************************/ @media all { /* * * { line-height: 120%; } */ dd { margin-bottom: 2ex; } dl:first-child { margin-top: 2ex; } } @media print { body { font-size: 11pt; padding: 0px !important; } a:link, a:visited { color: black; background: inherit; } code, pre, samp, tt { font-size: 10pt; } #nav, #search_this_book, #comment_form, #comment_announcement, #flipNav, .noprint { display: none !important; } body#left-nav-present { overflow: visible !important; } } PKʍPK.AOEBPS/dcommon/rightnav.gif&GIF89a1ֽ筽ޭƔkZZk{Bc{,@ ) l)- $CҠҀ ! D1 #:aS( c4B0 AC8 ְ9!%MLj Z * ctypJBa H t>#Sb(clhUԂ̗4DztSԙ9ZQҀEPEPEPEPEPEPEPM=iԍP Gii c*yF 1׆@\&o!QY00_rlgV;)DGhCq7~..p&1c:u֫{fI>fJL$}BBP?JRWc<^j+χ5b[hֿ- 5_j?POkeQ^hֿ1L^ H ?Qi?z?+_xɔŪ\썽O]χ>)xxV/s)e6MI7*ߊޛv֗2J,;~E4yi3[nI`Ѱe9@zXF*W +]7QJ$$=&`a۾?]N T䏟'X)Ɣkf:j |>NBWzYx0t!* _KkoTZ?K Gc+UyڹgNuh^iSo5{\ܹ3Yos}.>if FqR5\/TӮ#]HS0DKu{($"2xִ{SBJ8=}Y=.|Tsц2UЫ%.InaegKo z ݎ3ֹxxwM&2S%';+I',kW&-"_¿_ Vq^ܫ6pfT2RV A^6RKetto^[{w\jPZ@ޢN4/XN#\42j\(z'j =~-I#:q[Eh|X:sp* bifp$TspZ-}NM*B-bb&*xUr#*$M|QWY ~p~- fTED6O.#$m+t$˙H"Gk=t9r娮Y? CzE[/*-{c*[w~o_?%ƔxZ:/5𨴟q}/]22p qD\H"K]ZMKR&\C3zĽ[PJm]AS)Ia^km M@dК)fT[ijW*hnu Ͳiw/bkExG£@f?Zu.s0(<`0ֹoxOaDx\zT-^ѧʧ_1+CP/p[w 9~U^[U<[tĽwPv[yzD1W='u$Oeak[^ |Gk2xv#2?¹TkSݕ| rݞ[Vi _Kz*{\c(Ck_܏|?u jVڔ6f t?3nmZ6f%QAjJf9Rq _j7Z-y.pG$Xb]0')[_k;$̭?&"0FOew7 z-cIX岛;$u=\an$ zmrILu uٞ% _1xcUW%dtÀx885Y^gn;}ӭ)場QEQ@Q@Q@Q@Q@Q@!4xPm3w*]b`F_931˜[ן+(> E ly;<;MF-qst+}DH @YKlLmؤciN<|]IU)Lw(8t9FS(=>og<\Z~u_+X1ylsj'eՃ*U3`C!N9Q_WܱhKc93^ua>H ƕGk=8~e#_?{ǀe-[2ٔ7;=&K挑5zsLdx(e8#{1wS+ΝVkXq9>&yஏh$zq^0~/j@:/«Vnce$$uoPp}MC{$-akH@ɫ1O !8R9s5ԦYmϧ'OUṡ5T,!Ԛ+s#1Veo=[)g>#< s)ƽُA^䠮ωFUj(ǩ|N3Jڷ睁ϱuږZYGOTsI<&drav?A^_f׻B$,O__ԿC`it{6>G׈C~&$y؎v1q9Sc1fH[ѽ>,gG'0'@Vw,BO [#>ﱺg5ΒFVD%Yr:O5 Tu+O멃]ی38Ze}R&ѝ_xzc1DXgس;<,_,{ƽY'AS#oF.M#~cBuEx7G+Y)(5q+GCV;qF+CLQ)qEC&6z𿊘z}?&w=+)??&\g{;V??׻xGœdٿ׼-Nc')3K]N)iLTӿCdb7Q^a N sd>Fz[0S^s'Zi 77D}kWus ab~~H(>.fif9,~|Jk;YN3H8Y(t6Q݉k͇_÷Z+2߄&[ +Tr^藺97~c܎=[f1RrBǓ^kEMhxYVm<[џ6| kqbѱ| YA{G8p?\UM7Z66 g1U1igU69 u5Pƪ:VVZC=[@ҹ¨$kSmɳО\vFz~i3^a Osŧυ9Q}_3 όO{/wgoet39 vO2ea;Ύ7$U#?k+Ek&dpzbӱ+TaB0gN{[N7Gי}U7&@?>Fz~E!a@s ?'67XxO*!?qi]֏TQN@tI+\^s8l0)2k!!iW8F$(yOּT.k,/#1:}8uT˾+5=O/`IW G֯b.-<= HOm;~so~hW5+kS8s.zwE| ?4ӿw/K N 9?j(#0UT` Wzw}:_*9m>󑓀F?ELzv=8q:=WgJ`nDr Zе<ֹ](Q@Q@Q@Q@Q@Q@Q@Q@ 'IdC0EYJVcMty_~u+Sw-aO n<[YJgL#6i g5ЖDZ14cʝ!!\/M}/_AYR__>oC? _?7_G#RERW쏞KB}JxGSkǕA pƱơP m]hwB7U$Zq M95"3q1ioATߚ{g.t uu2k=;h#YB= fgS :TdLԃ!44mFK{Hrd^7oz|BVr<{)6AXգV»|>*/hS܏z͆OM=Εq (s|s׊LKQI :9NJ)P+!ʣoAF>+=@I}"x/}۠1aנc¹4emC:>p_xWKX` >R3_S½èųp3޺u3N e یbmͺ<_ mnݮ1Op?Gm)Qb%N585'%Ahs\6yw!"&Ɨ._wk)}GP;Z!#\"< *oƾ\)}N>"լ/~]Lg}pBG X?<zZ#x69S=6) jzx=y9O&>+e!!? ?s~k5Gʏ)?*ce7Ox~k5􇔾Q/e7/Ԑ#3OgNC0] ;_FiRl>Q.g>!%k#ú:Kn'&}?U@\pџPtp)v<{_i}Oվֲ3XIYIx~b<D?(=_JXH=bbi=Oh?_ C_O)}oW쏜? %Ƶ;-RYFi`wۭ{ϖZMtQ$"c_+ԃx1*0b;ԕ݋ESQEQEQEQEQEQEQEQEQEQZ(1F)h1K@XLRE&9P (bf{RӨ&)PEPEPbԴPGKZ(iإbn(:A%S0(-&)P+ ڎԴP11F)h&:LRmQ@Q@Š(((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((((PKje88PK.AOEBPS/dcommon/help.gif!GIF89a1εֵ֜֜{kZsBc{,@ )sƠTQ$8(4ʔ%ŌCK$A HP`$h8ŒSd+ɡ\ H@%' 6M HO3SJM /:Zi[7 \( R9r ERI%  N=aq   qƦs *q-n/Sqj D XZ;PKއ{&!PK.AOEBPS/sdo_rdf_concepts.htm Oracle Database Semantic Technologies Overview

1 Oracle Database Semantic Technologies Overview

This chapter describes the support in Oracle Database Enterprise Edition for semantic technologies, specifically Resource Description Framework (RDF) and a subset of the Web Ontology Language (OWL). It assumes that you are familiar with the major concepts associated with RDF and OWL, such as {subject, predicate, object} triples, URIs, blank nodes, plain and typed literals, and ontologies. This chapter does not explain these concepts in detail, but focuses instead on how the concepts are implemented in Oracle.

The PL/SQL subprograms for working with semantic data are in the SEM_APIS package, which is documented in Chapter 9.

The RDF and OWL support are features of Oracle Spatial, which must be installed for these features to be used. However, the use of RDF and OWL is not restricted to spatial data.

This chapter contains the following major sections:

For information about OWL concepts and the Oracle Database support for OWL capabilities, see Chapter 2.


Required Actions to Enable Semantic Technologies Support:

Before performing any operations described in this guide, you must enable semantic technologies support in the database and meet other prerequisites, as explained in Section A.1, "Enabling Semantic Technologies Support".


Release 11.2.0.2: Required Actions if Semantic Technologies Installation is Invalid:

Further action may be required if your Semantic Technologies installation is invalid after upgrading to Release 11.2.0.2.0. For information, see Section A.1.4.

1.1 Introduction to Oracle Semantic Technologies

Oracle Database enables you to store semantic data and ontologies, to query semantic data and to perform ontology-assisted query of enterprise relational data, and to use supplied or user-defined inferencing to expand the power of querying on semantic data. Figure 1-1 shows how these capabilities interact.

Figure 1-1 Oracle Semantic Capabilities

Description of Figure 1-1 follows
Description of "Figure 1-1 Oracle Semantic Capabilities"

As shown in Figure 1-1, the database contains semantic data and ontologies (RDF/OWL models), as well as traditional relational data. To load semantic data, bulk loading is the most efficient approach, although you can load data incrementally using transactional INSERT statements.


Note:

If you want to use existing semantic data from a release before Oracle Database 11.1, the data must be upgraded as described in Section A.1.

You can query semantic data and ontologies, and you can also perform ontology-assisted queries of semantic and traditional relational data to find semantic relationships. To perform ontology-assisted queries, use the SEM_RELATED operator, which is described in Section 2.3.

You can expand the power of queries on semantic data by using inferencing, which uses rules in rulebases. Inferencing enables you to make logical deductions based on the data and the rules. For information about using rules and rulebases for inferencing, see Section 1.3.6.

1.2 Semantic Data Modeling

In addition to its formal semantics, semantic data has a simple data structure that is effectively modeled using a directed graph. The metadata statements are represented as triples: nodes are used to represent two parts of the triple, and the third part is represented by a directed link that describes the relationship between the nodes. The triples are stored in a semantic data network. In addition, information is maintained about specific semantic data models created by database users. A user-created model has a model name, and refers to triples stored in a specified table column.

Statements are expressed in triples: {subject or resource, predicate or property, object or value}. In this manual, {subject, property, object} is used to describe a triple, and the terms statement and triple may sometimes be used interchangeably. Each triple is a complete and unique fact about a specific domain, and can be represented by a link in a directed graph.

1.3 Semantic Data in the Database

There is one universe for all semantic data stored in the database. All triples are parsed and stored in the system as entries in tables under the MDSYS schema. A triple {subject, property, object} is treated as one database object. As a result, a single document containing multiple triples results in multiple database objects.

All the subjects and objects of triples are mapped to nodes in a semantic data network, and properties are mapped to network links that have their start node and end node as subject and object, respectively. The possible node types are blank nodes, URIs, plain literals, and typed literals.

The following requirements apply to the specifications of URIs and the storage of semantic data in the database:

  • A subject must be a URI or a blank node.

  • A property must be a URI.

  • An object can be any type, such as a URI, a blank node, or a literal. (However, null values and null strings are not supported.)

1.3.1 Metadata for Models

The MDSYS.SEM_MODEL$ view contains information about all models defined in the database. When you create a model using the SEM_APIS.CREATE_SEM_MODEL procedure, you specify a name for the model, as well as a table and column to hold references to the semantic data, and the system automatically generates a model ID.

Oracle maintains the MDSYS.SEM_MODEL$ view automatically when you create and drop models. Users should never modify this view directly. For example, do not use SQL INSERT, UPDATE, or DELETE statements with this view.

The MDSYS.SEM_MODEL$ view contains the columns shown in Table 1-1.

Table 1-1 MDSYS.SEM_MODEL$ View Columns

Column NameData TypeDescription

OWNER

VARCHAR2(30)

Schema of the owner of the model.

MODEL_ID

NUMBER

Unique model ID number, automatically generated.

MODEL_NAME

VARCHAR2(25)

Name of the model.

TABLE_NAME

VARCHAR2(30)

Name of the table to hold references to semantic data for the model.

COLUMN_NAME

VARCHAR2(30)

Name of the column of type SDO_RDF_TRIPLE_S in the table to hold references to semantic data for the model.

MODEL_TABLESPACE_NAME

VARCHAR2(30)

Name of the tablespace to be used for storing the triples for this model.


When you create a model, a view for the triples associated with the model is also created under the MDSYS schema. This view has a name in the format SEMM_model-name, and it is visible only to the owner of the model and to users with suitable privileges. Each MDSYS.SEMM_model-name view contains a row for each triple (stored as a link in a network), and it has the columns shown in Table 1-2.

Table 1-2 MDSYS.SEMM_model-name View Columns

Column NameData TypeDescription

P_VALUE_ID

NUMBER

The VALUE_ID for the text value of the predicate of the triple. Part of the primary key.

START_NODE_ID

NUMBER

The VALUE_ID for the text value of the subject of the triple. Also part of the primary key.

CANON_END_NODE_ID

NUMBER

The VALUE_ID for the text value of the canonical form of the object of the triple. Also part of the primary key.

END_NODE_ID

NUMBER

The VALUE_ID for the text value of the object of the triple

MODEL_ID

NUMBER

The ID for the RDF model to which the triple belongs.

COST

NUMBER

(Reserved for future use)

CTXT1

NUMBER

(Reserved column; can be used for fine-grained access control)

CTXT2

VARCHAR2(4000)

(Reserved for future use)

DISTANCE

NUMBER

(Reserved for future use)

EXPLAIN

VARCHAR2(4000)

(Reserved for future use)

PATH

VARCHAR2(4000)

(Reserved for future use)

G_ID

NUMBER

The VALUE_ID for the text value of the graph name for the triple. Null indicates the default graph (see Section 1.3.9, "Named Graphs").

LINK_ID

VARCHAR2(71)

Unique triple identifier value. (It is currently a computed column, and its definition may change in a future release.)



Note:

In Table 1-2, for columns P_VALUE_ID, START_NODE_ID, END_NODE_ID, CANON_END_NODE_ID, and G_ID, the actual ID values are computed from the corresponding lexical values. However, a lexical value may not always map to the same ID value.

1.3.2 Statements

The MDSYS.RDF_VALUE$ table contains information about the subjects, properties, and objects used to represent RDF statements. It uniquely stores the text values (URIs or literals) for these three pieces of information, using a separate row for each part of each triple.

Oracle maintains the MDSYS.RDF_VALUE$ table automatically. Users should never modify this view directly. For example, do not use SQL INSERT, UPDATE, or DELETE statements with this view.

The RDF_VALUE$ table contains the columns shown in Table 1-3.

Table 1-3 MDSYS.RDF_VALUE$ Table Columns

Column NameData TypeDescription

VALUE_ID

NUMBER

Unique value ID number, automatically generated.

VALUE_TYPE

VARCHAR2(10)

The type of text information stored in the VALUE_NAME column. Possible values: UR for URI, BN for blank node, PL for plain literal, PL@ for plain literal with a language tag, PLL for plain long literal, PLL@ for plain long literal with a language tag, TL for typed literal, or TLL for typed long literal. A long literal is a literal with more than 4000 bytes.

VNAME_PREFIX

VARCHAR2(4000)

If the length of the lexical value is 4000 bytes or less, this column stores a prefix of a portion of the lexical value. The SEM_APIS.VALUE_NAME_PREFIX function can be used for prefix computation. For example, the prefix for the portion of the lexical value <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> without the angle brackets is http://www.w3.org/1999/02/22-rdf-syntax-ns#.

VNAME_SUFFIX

VARCHAR2(512)

If the length of the lexical value is 4000 bytes or less, this column stores a suffix of a portion of the lexical value. The SEM_APIS.VALUE_NAME_SUFFIX function can be used for suffix computation. For the lexical value mentioned in the description of the VNAME_PREFIX column, the suffix is type.

LITERAL_TYPE

VARCHAR2(4000)

For typed literals, the type information; otherwise, null. For example, for a row representing a creation date of 1999-08-16, the VALUE_TYPE column can contain TL, and the LITERAL_TYPE column can contain http://www.w3.org/2001/XMLSchema#date.

LANGUAGE_TYPE

VARCHAR2(80)

Language tag (for example, fr for French) for a literal with a language tag (that is, if VALUE_TYPE is PL@ or PLL@). Otherwise, this column has a null value.

CANON_ID

NUMBER

The ID for the canonical lexical value for the current lexical value. (The use of this column may change in a future release.)

COLLISION_EXT

VARCHAR2(64)

Used for collision handling for the lexical value. (The use of this column may change in a future release.)

CANON_COLLISION_EXT

VARCHAR2(64)

Used for collision handling for the canonical lexical value. (The use of this column may change in a future release.)

LONG_VALUE

CLOB

The character string if the length of the lexical value is greater than 4000 bytes. Otherwise, this column has a null value.

VALUE_NAME

VARCHAR2(4000)

This is a computed column. If length of the lexical value is 4000 bytes or less, the value of this column is the concatenation of the values of VNAME_PREFIX column and the VNAME_SUFFIX column.


1.3.2.1 Triple Uniqueness and Data Types for Literals

Duplicate triples are not stored in the database. To check if a triple is a duplicate of an existing triple, the subject, property, and object of the incoming triple are checked against triple values in the specified model. If the incoming subject, property, and object are all URIs, an exact match of their values determines a duplicate. However, if the object of incoming triple is a literal, an exact match of the subject and property, and a value (canonical) match of the object, determine a duplicate. For example, the following two triples are duplicates:

<eg:a> <eg:b> "123"^^http://www.w3.org/2001/XMLSchema#int
<eg:a> <eg:b> "123"^^http://www.w3.org/2001/XMLSchema#unsignedByte

The second triple is treated as a duplicate of the first, because "123"^^http://www.w3.org/2001/XMLSchema#int has an equivalent value (is canonically equivalent) to "123"^^http://www.w3.org/2001/XMLSchema#unsignedByte. Two entities are canonically equivalent if they can be reduced to the same value.

To use a non-RDF example, A*(B-C), A*B-C*A, (B-C)*A, and -A*C+A*B all convert into the same canonical form.

Value-based matching of lexical forms is supported for the following data types:

  • STRING: plain literal, xsd:string and some of its XML Schema subtypes

  • NUMERIC: xsd:decimal and its XML Schema subtypes, xsd:float, and xsd:double. (Support is not provided for float/double INF, -INF, and NaN values.)

  • DATETIME: xsd:datetime, with support for time zone. (Without time zone there are still multiple representations for a single value, for example, "2004-02-18T15:12:54" and "2004-02-18T15:12:54.0000".)

  • DATE: xsd:date, with or without time zone

  • OTHER: Everything else. (No attempt is made to match different representations).

Canonicalization is performed when the time zone is present for literals of type xsd:time and xsd:dateTime.

The following namespace definition is used: xmlns:xsd="http://www.w3.org/2001/XMLSchema"

The first occurrence of a literal in the RDF_VALUE$ table is taken as the canonical form and given the VALUE_TYPE value of CPL, CPL@, CTL, CPLL, CPLL@, or CTLL as appropriate; that is, a C for canonical is prefixed to the actual value type. If a literal with the same canonical form (but a different lexical representation) as a previously inserted literal is inserted into the RDF_VALUE$ table, the VALUE_TYPE value assigned to the new insert is PL, PL@, TL, PLL, PLL@, or TLL as appropriate.

Canonically equivalent text values having different lexical representations are thus stored in the RDF_VALUE$ table; however, canonically equivalent triples are not stored in the database.

1.3.3 Subjects and Objects

RDF subjects and objects are mapped to nodes in a semantic data network. Subject nodes are the start nodes of links, and object nodes are the end nodes of links. Non-literal nodes (that is, URIs and blank nodes) can be used as both subject and object nodes. Literals can be used only as object nodes.

1.3.4 Blank Nodes

Blank nodes can be used as subject and object nodes in the semantic network. Blank node identifiers are different from URIs in that they are scoped within a semantic model. Thus, although multiple occurrences of the same blank node identifier within a single semantic model necessarily refer to the same resource, occurrences of the same blank node identifier in two different semantic models do not refer to the same resource.

In an Oracle semantic network, this behavior is modeled by requiring that blank nodes are always reused (that is, are used to represent the same resource if the same blank node identifier is used) within a semantic model, and never reused between two different models. Thus, when inserting triples involving blank nodes into a model, you must use the SDO_RDF_TRIPLE_S constructor that supports reuse of blank nodes.

1.3.5 Properties

Properties are mapped to links that have their start node and end node as subjects and objects, respectively. Therefore, a link represents a complete triple.

When a triple is inserted into a model, the subject, property, and object text values are checked to see if they already exist in the database. If they already exist (due to previous statements in other models), no new entries are made; if they do not exist, three new rows are inserted into the RDF_VALUE$ table (described in Section 1.3.2).

1.3.6 Inferencing: Rules and Rulebases

Inferencing is the ability to make logical deductions based on rules. Inferencing enables you to construct queries that perform semantic matching based on meaningful relationships among pieces of data, as opposed to just syntactic matching based on string or other values. Inferencing involves the use of rules, either supplied by Oracle or user-defined, placed in rulebases.

Figure 1-2 shows triple sets being inferred from model data and the application of rules in one or more rulebases. In this illustration, the database can have any number of semantic models, rulebases, and inferred triple sets, and an inferred triple set can be derived using rules in one or more rulebases.

A rule is an object that can be applied to draw inferences from semantic data. A rule is identified by a name and consists of:

  • An IF side pattern for the antecedents

  • An optional filter condition that further restricts the subgraphs matched by the IF side pattern

  • A THEN side pattern for the consequents

For example, the rule that a chairperson of a conference is also a reviewer of the conference could be represented as follows:

('chairpersonRule', -- rule name
 '(?r :ChairPersonOf ?c)', -- IF side pattern
 NULL,  -- filter condition
 '(?r :ReviewerOf ?c)', -- THEN side pattern
 SEM_ALIASES (SEM_ALIAS('', 'http://some.org/test/'))
)

In this case, the rule does not have a filter condition, so that component of the representation is NULL. For best performance, use a single-triple pattern on the THEN side of the rule. If a rule has multiple triple patterns on the THEN side, you can easily break it into multiple rules, each with a single-triple pattern, on the THEN side.

A rulebase is an object that contains rules. The following Oracle-supplied rulebases are provided:

  • RDFS

  • RDF (a subset of RDFS)

  • OWLSIF (empty)

  • RDFS++ (empty)

  • OWL2RL (empty)

  • OWLPrime (empty)

  • SKOSCORE (empty)

The RDFS and RDF rulebases are created when you call the SEM_APIS.CREATE_SEM_NETWORK procedure to add RDF support to the database. The RDFS rulebase implements the RDFS entailment rules, as described in the World Wide Web Consortium (W3C) RDF Semantics document at http://www.w3.org/TR/rdf-mt/. The RDF rulebase represents the RDF entailment rules, which are a subset of the RDFS entailment rules. You can see the contents of these rulebases by examining the MDSYS.SEMR_RDFS and MDSYS.SEMR_RDF views.

You can also create user-defined rulebases using the SEM_APIS.CREATE_RULEBASE procedure. User-defined rulebases enable you to provide additional specialized inferencing capabilities.

For each rulebase, a system table is created to hold rules in the rulebase, along with a system view with a name in the format MDSYS.SEMR_rulebase-name (for example, MDSYS.SEMR_FAMILY_RB for a rulebase named FAMILY_RB). You must use this view to insert, delete, and modify rules in the rulebase. Each MDSYS.SEMR_rulebase-name view has the columns shown in Table 1-4.

Table 1-4 MDSYS.SEMR_rulebase-name View Columns

Column NameData TypeDescription

RULE_NAME

VARCHAR2(30)

Name of the rule

ANTECEDENTS

VARCHAR2(4000)

IF side pattern for the antecedents

FILTER

VARCHAR2(4000)

Filter condition that further restricts the subgraphs matched by the IF side pattern. Null indicates no filter condition is to be applied.

CONSEQUENTS

VARCHAR2(4000)

THEN side pattern for the consequents

ALIASES

SEM_ALIASES

One or more namespaces to be used. (The SEM_ALIASES data type is described in Section 1.6.)


Information about all rulebases is maintained in the MDSYS.SEM_RULEBASE_INFO view, which has the columns shown in Table 1-5 and one row for each rulebase.

Table 1-5 MDSYS.SEM_RULEBASE_INFO View Columns

Column NameData TypeDescription

OWNER

VARCHAR2(30)

Owner of the rulebase

RULEBASE_NAME

VARCHAR2(25)

Name of the rulebase

RULEBASE_VIEW_NAME

VARCHAR2(30)

Name of the view that you must use for any SQL statements that insert, delete, or modify rules in the rulebase

STATUS

VARCHAR2(30)

Contains VALID if the rulebase is valid, INPROGRESS if the rulebase is being created, or FAILED if a system failure occurred during the creation of the rulebase.


Example 1-1 creates a rulebase named family_rb, and then inserts a rule named grandparent_rule into the family_rb rulebase. This rule says that if a person is the parent of a child who is the parent of a child, that person is a grandparent of (that is, has the grandParentOf relationship with respect to) his or her child's child. It also specifies a namespace to be used. (This example is an excerpt from Example 1-43 in Section 1.11.2.)

Example 1-1 Inserting a Rule into a Rulebase

EXECUTE SEM_APIS.CREATE_RULEBASE('family_rb');

INSERT INTO mdsys.semr_family_rb VALUES(
  'grandparent_rule',
  '(?x :parentOf ?y) (?y :parentOf ?z)',
  NULL,
  '(?x :grandParentOf ?z)', 
  SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')));

Note that the kind of grandparent rule shown in Example 1-1 can be implemented using the OWL 2 property chain construct. For information about property chain handling, see Section 3.2.2.

You can specify one or more rulebases when calling the SEM_MATCH table function (described in Section 1.6), to control the behavior of queries against semantic data. Example 1-2 refers to the family_rb rulebase and to the grandParentOf relationship created in Example 1-1, to find all grandfathers (grandparents who are male) and their grandchildren. (This example is an excerpt from Example 1-43 in Section 1.11.2.)

Example 1-2 Using Rulebases for Inferencing

-- Select all grandfathers and their grandchildren from the family model.
-- Use inferencing from both the RDFS and family_rb rulebases.
SELECT x, y
  FROM TABLE(SEM_MATCH(
    '(?x :grandParentOf ?y) (?x rdf:type :Male)',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

For information about support for native OWL inferencing, see Section 2.2.

1.3.7 Entailments (Rules Indexes)

An entailment (rules index) is an object containing precomputed triples that can be inferred from applying a specified set of rulebases to a specified set of models. If a SEM_MATCH query refers to any rulebases, an entailment must exist for each rulebase-model combination in the query.

To create an entailment, use the SEM_APIS.CREATE_ENTAILMENT procedure. To drop (delete) an entailment, use the SEM_APIS.DROP_ENTAILMENT procedure.

When you create an entailment, a view for the triples associated with the entailment is also created under the MDSYS schema. This view has a name in the format SEMI_entailment-name, and it is visible only to the owner of the entailment and to users with suitable privileges. Each MDSYS.SEMI_entailment-name view contains a row for each triple (stored as a link in a network), and it has the same columns as the SEMM_model-name view, which is described in Table 1-2 in Section 1.3.1.

Information about all entailments is maintained in the MDSYS.SEM_RULES_INDEX_INFO view, which has the columns shown in Table 1-6 and one row for each entailment.

Table 1-6 MDSYS.SEM_RULES_INDEX_INFO View Columns

Column NameData TypeDescription

OWNER

VARCHAR2(30)

Owner of the entailment

INDEX_NAME

VARCHAR2(25)

Name of the entailment

INDEX_VIEW_NAME

VARCHAR2(30)

Name of the view that you must use for any SQL statements that insert, delete, or modify rules in the entailment

STATUS

VARCHAR2(30)

Contains VALID if the entailment is valid, INVALID if the entailment is not valid, INCOMPLETE if the entailment is incomplete (similar to INVALID but requiring less time to re-create), INPROGRESS if the entailment is being created, or FAILED if a system failure occurred during the creation of the entailment.

MODEL_COUNT

NUMBER

Number of models included in the entailment

RULEBASE_COUNT

NUMBER

Number of rulebases included in the entailment


Information about all database objects, such as models and rulebases, related to entailments is maintained in the MDSYS.SEM_RULES_INDEX_DATASETS view. This view has the columns shown in Table 1-7 and one row for each unique combination of values of all the columns.

Table 1-7 MDSYS.SEM_RULES_INDEX_DATASETS View Columns

Column NameData TypeDescription

INDEX_NAME

VARCHAR2(25)

Name of the entailment

DATA_TYPE

VARCHAR2(8)

Type of data included in the entailment. Examples: MODEL and RULEBASE

DATA_NAME

VARCHAR2(25)

Name of the object of the type in the DATA_TYPE column


Example 1-3 creates an entailment named family_rb_rix_family, using the family model and the RDFS and family_rb rulebases. (This example is an excerpt from Example 1-43 in Section 1.11.2.)

Example 1-3 Creating an Entailment

BEGIN
  SEM_APIS.CREATE_ENTAILMENT(
    'rdfs_rix_family',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'));
END;
/

1.3.8 Virtual Models

A virtual model is a logical graph that can be used in a SEM_MATCH query. A virtual model is the result of a UNION or UNION ALL operation on one or more models and optionally the corresponding entailment.

Using a virtual model can provide several benefits:

  • It can simplify management of access privileges for semantic data. For example, assume that you have created three semantic models and one entailment based on the three models and the OWLPrime rulebase. Without a virtual model, you must individually grant and revoke access privileges for each model and the entailment. However, if you create a virtual model that contains the three models and the entailment, you will only need to grant and revoke access privileges for the single virtual model.

  • It can facilitate rapid updates to semantic models. For example, assume that virtual model VM1 contains model M1 and entailment R1 (that is, VM1 = M1 UNION ALL R1), and assume that semantic model M1_UPD is a copy of M1 that has been updated with additional triples and that R1_UPD is an entailment created for M1_UPD. Now, to have user queries over VM1 go to the updated model and entailment, you can redefine virtual model VM1 (that is, VM1 = M1_UPD UNION ALL R1_UPD).

  • It can simplify query specification because querying a virtual model is equivalent to querying multiple models in a SEM_MATCH query. For example, assume that models m1, m2, and m3 already exist, and that an entailment has been created for m1, m2 ,and m3 using the OWLPrime rulebase. You could create a virtual model vm1 as follows:

    EXECUTE sem_apis.create_virtual_model('vm1', sem_models('m1', 'm2', 'm3'), 
                                          sem_rulebases('OWLPRIME'));
    

    To query the virtual model, use the virtual model name as if it were a model in a SEM_MATCH query. For example, the following query on the virtual model:

    SELECT * FROM TABLE (sem_match('{…}', sem_models('vm1'), null, …));
    

    is equivalent to the following query on all the individual models:

    SELECT * FROM TABLE (sem_match('{…}', sem_models('m1', 'm2', 'm3'), 
                                          sem_rulebases('OWLPRIME'), …));
    

    A SEM_MATCH query over a virtual model will query either the SEMV or SEMU view (SEMU by default and SEMV if the 'ALLOW_DUP=T' option is specified) rather than querying the UNION or UNION ALL of each model and entailment. For information about these views and options, see the reference section for the SEM_APIS.CREATE_VIRTUAL_MODEL procedure.

Note that you cannot use Oracle Workspace Manager version-enabling on a model that participates in a virtual model. (Workspace Manager support for RDF data is described in Chapter 6.)

Virtual models use views (described later in this section) and add some metadata entries, but do not significantly increase system storage requirements.

To create a virtual model, use the SEM_APIS.CREATE_VIRTUAL_MODEL procedure. To drop (delete) a virtual model, use the SEM_APIS.DROP_VIRTUAL_MODEL procedure. A virtual model is dropped automatically if any of its component models, rulebases, or entailment are dropped.

To query a virtual model, specify the virtual model name in the models parameter of the SEM_MATCH table function, as shown in Example 1-4.

Example 1-4 Querying a Virtual Model

SELECT COUNT(protein)
  FROM TABLE (SEM_MATCH (
    '(?protein rdf:type :Protein) 
     (?protein :citation ?citation) 
     (?citation :author "Bairoch A.")',
    SEM_MODELS('UNIPROT_VM'), 
    NULL, 
    SEM_ALIASES(SEM_ALIAS('', 'http://purl.uniprot.org/core/')),
    NULL, 
    NULL, 
    'ALLOW_DUP=T'));

For information about the SEM_MATCH table function, see Section 1.6, which includes information using certain attributes when querying a virtual model.

When you create a virtual model, an entry is created for it in the MDSYS.SEM_MODEL$ view, which is described in Table 1-1 in Section 1.3.1. However, the values in several of the columns are different for virtual models as opposed to semantic models, as explained in Table 1-8.

Table 1-8 MDSYS.SEM_MODEL$ View Column Explanations for Virtual Models

Column NameData TypeDescription

OWNER

VARCHAR2(30)

Schema of the owner of the virtual model

MODEL_ID

NUMBER

Unique model ID number, automatically generated. Will be a negative number, to indicate that this is a virtual model.

MODEL_NAME

VARCHAR2(25)

Name of the virtual model

TABLE_NAME

VARCHAR2(30)

Null for a virtual model

COLUMN_NAME

VARCHAR2(30)

Null for a virtual model

MODEL_TABLESPACE_NAME

VARCHAR2(30)

Null for a virtual model


Information about all virtual models is maintained in the MDSYS.SEM_VMODEL_INFO view, which has the columns shown in Table 1-9 and one row for each virtual model.

Table 1-9 MDSYS.SEM_VMODEL_INFO View Columns

Column NameData TypeDescription

OWNER

VARCHAR2(30)

Owner of the virtual model

VIRTUAL_MODEL_NAME

VARCHAR2(25)

Name of the virtual model

UNIQUE_VIEW_NAME

VARCHAR2(30)

Name of the view that contains unique triples in the virtual model, or null if the view was not created

DUPLICATE_VIEW_NAME

VARCHAR2(30)

Name of the view that contains duplicate triples (if any) in the virtual model

STATUS

VARCHAR2(30)

Contains VALID if the associated entailment is valid, INVALID if the entailment is not valid, INCOMPLETE if the entailment is incomplete (similar to INVALID but requiring less time to re-create), INPROGRESS if the entailment is being created, FAILED if a system failure occurred during the creation of the entailment, or NORIDX if no entailment is associated with the virtual model.

MODEL_COUNT

NUMBER

Number of models in the virtual model

RULEBASE_COUNT

NUMBER

Number of rulebases used for the virtual model

RULES_INDEX_COUNT

NUMBER

Number of entailments in the virtual model


Information about all objects (models, rulebases, and entailment) related to virtual models is maintained in the MDSYS.SEM_VMODEL_DATASETS view. This view has the columns shown in Table 1-10 and one row for each unique combination of values of all the columns.

Table 1-10 MDSYS.SEM_VMODEL_DATASETS View Columns

Column NameData TypeDescription

VIRTUAL_MODEL_NAME

VARCHAR2(25)

Name of the virtual model

DATA_TYPE

VARCHAR2(8)

Type of object included in the virtual model. Examples: MODEL for a semantic model, RULEBASE for a rulebase, or RULEIDX for an entailment

DATA_NAME

VARCHAR2(25)

Name of the object of the type in the DATA_TYPE column


1.3.9 Named Graphs

Oracle Database Semantic Technologies supports the use of named graphs, which are described in the "RDF Dataset" section of the W3C SPARQL Query Language for RDF recommendation (http://www.w3.org/TR/rdf-sparql-query/#rdfDataset).

This support is provided by extending an RDF triple consisting of the traditional subject, predicate, and object, to include an additional component to represent a graph name. The extended RDF triple, despite having four components, will continue to be referred to as an RDF triple in this document. In addition, the following terms are sometimes used:

  • N-Triple is a format that does not allow extended triples. Thus, n-triples can include only triples with three components.

  • N-Quad is a format that allows both "regular" triples (three components) and extended triples (four components, including the graph name). For more information, see N-Quads: Extending N-Triples with Context (http://sw.deri.org/2008/07/n-quads/).

    To load a file containing extended triples (possibly mixed with regular triples) into an Oracle database, the input file must be in N-Quad format.

The graph name component of an RDF triple must either be null or a URI. If it is null, the RDF triple is said to belong to a default graph; otherwise it is said to belong to a named graph whose name is designated by the URI.

Additionally, to support named graphs in SDO_RDF_TRIPLE_S object type (described in Section 1.5), a new syntax is provided for specifying a model-graph, that is, a combination of model and graph (if any) together, and the RDF_M_ID attribute holds the identifier for a model-graph: a combination of model ID and value ID for the graph (if any). The name of a model-graph is specified as model_name, and if a graph is present, followed by the colon (:) separator character and the graph name (which must be a URI and enclosed within angle brackets < >).

For example, in a medical data set the named graph component for each RDF triple might be a URI based on patient identifier, so there could be as many named graphs as there are unique patients, with each named graph consisting of data for a specific patient.

For information about performing specific operations with named graphs, see the following:

1.3.9.1 Data Formats Related to Named Graph Support

TriG (http://www4.wiwiss.fu-berlin.de/bizer/trig/) and N-QUADS (http://sw.deri.org/2008/07/n-quads/) are two popular data formats that provide graph names (or context) to triple data. (Note that as on November 2011, neither format is a standard.) The graph names (context) can be used in a variety of different ways. Typical usage includes, but is not limited to, the grouping of triples for ease of management, localized query, localized inference, and provenance.

Example 1-5 shows an RDF data set encoded in TriG format. It contains a default graph and a named graph.

Example 1-5 RDF Data Encoded in TriG Format

@prefix foaf: <http://xmlns.com/foaf/0.1/> .
@prefix dc: <http://purl.org/dc/elements/1.1/> .
 
# Default graph
{
  <http://my.com/John> dc:publisher <http://publisher/Xyz> .
}
 
# A named graph
<http://my.com/John> {
  <http://my.com/John> foaf:name "John Doe" .
}

When loading the TriG file from Example 1-5 into a DatasetGraphOracleSem object (for example, using Example 7-10 in Section 7.11, "Bulk Loading Using the Jena Adapter", but replacing the constant "N-QUADS" with "TRIG"), the triples in the default graph will be loaded into Oracle Database as triples with null graph names, and the triples in the named graphs will be loaded into Oracle Database with the designated graph names.

N-QUADS format is a simple extension of the existing N-TRIPLES format by adding an optional fourth column (graph name or context). Example 1-6 shows the N-QUADS format representation of the TriG file from Example 1-5.

Example 1-6 N-QUADS Format Representation

<http://my.com/John> <http://purl.org/dc/elements/1.1/publisher> <http://publisher/Xyz> .
<http://my.com/John> <http://xmlns.com/foaf/0.1/name> "John Doe" <http://my.com/John>

When loading an N-QUADS file into a DatasetGraphOracleSem object (see Example 7-10), lines without the fourth column will be loaded into Oracle Database as triples with null graph names, and lines with a fourth column will be loaded into Oracle Database with the designated graph names.

1.3.10 Semantic Data Security Considerations

The following database security considerations apply to the use of semantic data:

  • When a model or entailment is created, the owner gets the SELECT privilege with the GRANT option on the associated view. Users that have the SELECT privilege on these views can perform SEM_MATCH queries against the associated model or entailment.

  • When a rulebase is created, the owner gets the SELECT, INSERT, UPDATE, and DELETE privileges on the rulebase, with the GRANT option. Users that have the SELECT privilege on a rulebase can create an entailment that includes the rulebase. The INSERT, UPDATE, and DELETE privileges control which users can modify the rulebase and how they can modify it.

  • To perform data manipulation language (DML) operations on a model, a user must have DML privileges for the corresponding base table.

  • The creator of the base table corresponding to a model can grant privileges to other users.

  • To perform data manipulation language (DML) operations on a rulebase, a user must have the appropriate privileges on the corresponding database view.

  • The creator of a model can grant SELECT privileges on the corresponding database view to other users.

  • A user can query only those models for which that user has SELECT privileges to the corresponding database views.

  • Only the creator of a model or a rulebase can drop it.

1.4 Semantic Metadata Tables and Views

Oracle Database maintains several tables and views in the MDSYS schema to hold metadata related to semantic data. (Some of these tables and views are created by the SEM_APIS.CREATE_SEM_NETWORK procedure, as explained in Section 1.10, and some are created only as needed.) Table 1-11 lists the tables and views in alphabetical order. (In addition, several tables and views are created for Oracle internal use, and these are accessible only by users with DBA privileges.)

Table 1-11 Semantic Metadata Tables and Views

NameContains Information AboutDescribed In

RDF_VALUE$

Subjects, properties, and objects used to represent statements

Section 1.3.2


RDFOLS_SECURE_RESOURCE

Resources secured with Oracle Label Security (OLS) policies and the sensitivity labels associated with these resources

Section 5.2.2.6


RDFVPD_MODELS

RDF models and their associated VPD policies

Section 5.1.5


RDFVPD_POLICIES

All VPD policies defined in the schema or the policies to which the user has FULL access

Section 5.1.4


RDFVPD_POLICY_CONSTRAINTS

Constraints defined in the VPD policy that are accessible to the current user

Section 5.1.6


RDFVPD_PREDICATE_MDATA

Predicate metadata associated with a VPD policy

Section 5.1.7


RDFVPD_RESOURCE_REL

Subclass, subproperty, and equivalence property relationships that are defined between resources in a VPD policy

Section 5.1.8


SEM_DTYPE_INDEX_INFO

All data type indexes in the network

Section 1.9


SEM_MODEL$

All models defined in the database

Section 1.3.1


SEM_NETWORK_INDEX_INFO$

Semantic network indexes

Section 1.8.1


SEM_RULEBASE_INFO

Rulebases

Section 1.3.6


SEM_RULES_INDEX_DATASETS

Database objects used in entailments

Section 1.3.7


SEM_RULES_INDEX_INFO

Entailments (rules indexes)

Section 1.3.7


SEM_VMODEL_INFO

Virtual models

Section 1.3.8


SEM_VMODEL_DATASETS

Database objects used in virtual models

Section 1.3.8


SEMCL_entailment-name

owl:sameAs clique members and canonical representatives

Section 2.2.8


SEMI_entailment-name

Triples in the specified entailment

Section 1.3.7


SEMM_model-name

Triples in the specified model

Section 1.3.1


SEMR_rulebase-name

Rules in the specified rulebase

Section 1.3.6


SEMU_virtual-model-name

Unique triples in the virtual model

Section 1.3.8


SEMV_virtual-model-name

Triples in the virtual model

Section 1.3.8



1.5 Semantic Data Types, Constructors, and Methods

The SDO_RDF_TRIPLE object type represents semantic data in triple format, and the SDO_RDF_TRIPLE_S object type (the _S for storage) stores persistent semantic data in the database. The SDO_RDF_TRIPLE_S type has references to the data, because the actual semantic data is stored only in the central RDF schema. This type has methods to retrieve the entire triple or part of the triple.


Note:

Blank nodes are always reused within an RDF model and cannot be reused across models

The SDO_RDF_TRIPLE type is used to display triples, whereas the SDO_RDF_TRIPLE_S type is used to store the triples in database tables.

The SDO_RDF_TRIPLE object type has the following attributes:

SDO_RDF_TRIPLE (
  subject VARCHAR2(4000), 
  property VARCHAR2(4000), 
  object VARCHAR2(10000))

The SDO_RDF_TRIPLE_S object type has the following attributes:

SDO_RDF_TRIPLE_S (
  RDF_C_ID  NUMBER,  -- Canonical object value ID
  RDF_M_ID NUMBER,  -- Model (or Model-Graph) ID 
  RDF_S_ID  NUMBER,  -- Subject value ID
  RDF_P_ID NUMBER, -- Property value ID
  RDF_O_ID  NUMBER)  -- Object value ID

The SDO_RDF_TRIPLE_S type has the following methods that retrieve the name of the RDF model (or model-graph), a triple, or a part (subject, property, or object) of a triple:

GET_MODEL() RETURNS VARCHAR2
GET_TRIPLE() RETURNS SDO_RDF_TRIPLE
GET_SUBJECT() RETURNS VARCHAR2
GET_PROPERTY() RETURNS VARCHAR2
GET_OBJECT() RETURNS CLOB

Example 1-7 shows the SDO_RDF_TRIPLE_S methods.

Example 1-7 SDO_RDF_TRIPLE_S Methods

SELECT a.triple.GET_MODEL() AS model_graph, a.triple.GET_TRIPLE() AS triple
  FROM articles_rdf_data a WHERE a.id = 99;
 
MODEL_GRAPH
--------------------------------------------------------------------------------
TRIPLE(SUBJECT, PROPERTY, OBJECT)
--------------------------------------------------------------------------------
ARTICLES:<http://examples.com/ns#Graph1>
SDO_RDF_TRIPLE('<http://nature.example.com/Article101>', '<http://purl.org/dc/elements/1.1/creator>', '"John Smith"')

SELECT a.triple.GET_TRIPLE() AS triple
  FROM articles_rdf_data a WHERE a.id = 1;
 
TRIPLE(SUBJECT, PROPERTY, OBJECT)
--------------------------------------------------------------------------------
SDO_RDF_TRIPLE('<http://nature.example.com/Article1>', '<http://purl.org/dc/elem
ents/1.1/title>', '<All about XYZ>')
 
SELECT a.triple.GET_SUBJECT() AS subject
  FROM articles_rdf_data a WHERE a.id = 1;
 
SUBJECT                                                                         
--------------------------------------------------------------------------------
<http://nature.example.com/Article1>                                           
 
SELECT a.triple.GET_PROPERTY() AS property
  FROM articles_rdf_data a WHERE a.id = 1;
 
PROPERTY                                                                        
--------------------------------------------------------------------------------
<http://purl.org/dc/elements/1.1/title>                                         
 
SELECT a.triple.GET_OBJECT() AS object
  FROM articles_rdf_data a WHERE a.id = 1;
 
OBJECT                                                                          
--------------------------------------------------------------------------------
<All about XYZ>

1.5.1 Constructors for Inserting Triples

The following constructor formats are available for inserting triples into a model table. The only difference is that in the second format the data type for the object is CLOB, to accommodate very long literals.

SDO_RDF_TRIPLE_S (
  model_name VARCHAR2, -- Model name
  subject    VARCHAR2, -- Subject
  property   VARCHAR2, -- Property
  object     VARCHAR2) -- Object
  RETURN     SELF;

SDO_RDF_TRIPLE_S (
  model_name VARCHAR2, -- Model name
  subject    VARCHAR2, -- Subject
  property   VARCHAR2, -- Property
  object     CLOB) -- Object
  RETURN SELF;

GET_OBJ_VALUE() RETURN VARCHAR2;

Example 1-8 uses the first constructor format to insert several triples.

Example 1-8 SDO_RDF_TRIPLE_S Constructor to Insert Triples

INSERT INTO articles_rdf_data VALUES (2,
  SDO_RDF_TRIPLE_S ('articles','<http://nature.example.com/Article1>',
    '<http://purl.org/dc/elements/1.1/creator>',
    '"Jane Smith"'));

INSERT INTO articles_rdf_data VALUES (98,
  SDO_RDF_TRIPLE_S ('articles:<http://examples.com/ns#Graph1>',
    '<http://nature.example.com/Article102>',
    '<http://purl.org/dc/elements/1.1/creator>',
    '_:b1'));
 
INSERT INTO articles_rdf_data VALUES (97,
  SDO_RDF_TRIPLE_S ('articles:<http://examples.com/ns#Graph1>',
    '_:b2',
    '<http://purl.org/dc/elements/1.1/creator>',
    '_:b1'));

1.6 Using the SEM_MATCH Table Function to Query Semantic Data

To query semantic data, use the SEM_MATCH table function. This function has the following attributes:

SEM_MATCH(
  query        VARCHAR2,
  models       SEM_MODELS,
  rulebases    SEM_RULEBASES,
  aliases      SEM_ALIASES,
  filter       VARCHAR2,
  index_status VARCHAR2,
  options      VARCHAR2,
  graphs       SEM_GRAPHS,
  named_graphs SEM_GRAPHS
 ) RETURN ANYDATASET;

The query attribute is required. The other attributes are optional (that is, each can be a null value).

The query attribute is a string literal (or concatenation of string literals) with one or more triple patterns, usually containing variables. (The query attribute cannot be a bind variable or an expression involving a bind variable.) A triple pattern is a triple of atoms followed by a period. Each atom can be a variable (for example, ?x), a qualified name (for example, rdf:type) that is expanded based on the default namespaces and the value of the aliases attribute, or a full URI (for example, <http://www.example.org/family/Male>). In addition, the third atom can be a numeric literal (for example, 3.14), a plain literal (for example, "Herman"), a language-tagged plain literal (for example, "Herman"@en), or a typed literal (for example, "123"^^xsd:int).

For example, the following query attribute specifies three triple patterns to find grandfathers (that is, grandparents who are also male) and the height of each of their grandchildren:

'{ ?x :grandParentOf ?y . ?x rdf:type :Male . ?y :height ?h }'

The models attribute identifies the model or models to use. Its data type is SEM_MODELS, which has the following definition: TABLE OF VARCHAR2(25). If you are querying a virtual model, specify only the name of the virtual model and no other models. (Virtual models are explained in Section 1.3.8.)

The rulebases attribute identifies one or more rulebases whose rules are to be applied to the query. Its data type is SDO_RDF_RULEBASES, which has the following definition: TABLE OF VARCHAR2(25). If you are querying a virtual model, this attribute must be null.

The aliases attribute identifies one or more namespaces, in addition to the default namespaces, to be used for expansion of qualified names in the query pattern. Its data type is SEM_ALIASES, which has the following definition: TABLE OF SEM_ALIAS, where each SEM_ALIAS element identifies a namespace ID and namespace value. The SEM_ALIAS data type has the following definition: (namespace_id VARCHAR2(30), namespace_val VARCHAR2(4000))

The following default namespaces (namespace_id and namespace_val attributes) are used by the SEM_MATCH table function and the SEM_CONTAINS and SEM_RELATED operators:

('orardf', 'http://xmlns.oracle.com/rdf/')
('orageo', 'http://xmlns.oracle.com/rdf/geo/')
('owl',    'http://www.w3.org/2002/07/owl#')
('rdf',    'http://www.w3.org/1999/02/22-rdf-syntax-ns#')
('rdfs',   'http://www.w3.org/2000/01/rdf-schema#')
('xsd',    'http://www.w3.org/2001/XMLSchema#')

You can override any of these defaults by specifying the namespace_id value and a different namespace_val value in the aliases attribute.

The filter attribute identifies any additional selection criteria. If this attribute is not null, it should be a string in the form of a WHERE clause without the WHERE keyword. For example: '(h >= 6)' to limit the result to cases where the height of the grandfather's grandchild is 6 or greater (using the example of triple patterns earlier in this section).


Note:

Instead of using the filter attribute, you are encourage to use the FILTER keyword inside your query pattern whenever possible (as explained in Section 1.6.2). Using the FILTER keyword is likely to give better performance because of internal optimizations. The filter argument, however, can be useful if you require SQL constructs that cannot be expressed with the FILTER keyword.

The index_status attribute lets you query semantic data even when the relevant entailment does not have a valid status. (If you are querying a virtual model, this attribute refers to the entailment associated with the virtual model.) If this attribute is null, the query returns an error if the entailment does not have a valid status. If this attribute is not null, it must be the string INCOMPLETE or INVALID. For an explanation of query behavior with different index_status values, see Section 1.6.1.

The options attribute identifies options that can affect the results of queries. Options are expressed as keyword-value pairs. The following options are supported:

  • ALL_BGP_HASH and ALL_BGP_NL are global query optimizer hints that specify that all inter-BGP joins (for example. the join between the root BGP and an OPTIONAL BGP) should use the specified join type. (BGP stands for basic graph pattern. From the W3C SPARQL Query Language for RDF Recommendation: "SPARQL graph pattern matching is defined in terms of combining the results from matching basic graph patterns. A sequence of triple patterns interrupted by a filter comprises a single basic graph pattern. Any graph pattern terminates a basic graph pattern."

    Example 1-14 shows the ALL_BGP_HASH option used in a SEM_MATCH query.

  • ALL_LINK_HASH and ALL_LINK_NL are global query optimizer hints that specify the join type for all LINK$ joins (that is, all joins between triple patterns within a BGP). If ALL_LINK_HASH or ALL_LINK_NL is used with a HINT0 hint, the HINT0 hint overrides the ALL_ORDERED hint.

  • ALL_ORDERED is a global query optimizer hint that specifies that the triple patterns in each BGP in the query should be evaluated in order. If ALL_ORDERED is used with a HINT0 hint, the HINT0 hint overrides the ALL_ORDERED hint.

    Example 1-14 shows the ALL_ORDERED option used in a SEM_MATCH query.

  • ALLOW_DUP=T generates an underlying SQL statement that performs a "union all" instead of a union of the semantic models and inferred data (if applicable). This option may introduce more rows (duplicate triples) in the result set, and you may need to adjust the application logic accordingly. If you do not specify this option, duplicate triples are automatically removed across all the models and inferred data to maintain the set semantics of merged RDF graphs; however, removing duplicate triples increases query processing time. In general, specifying 'ALLOW_DUP=T' improves performance significantly when multiple semantic models are involved in a SEM_MATCH query.

    If you are querying a virtual model, specifying ALLOW_DUP=T causes the SEMV_vm_name view to be queried; otherwise, the SEMU_vm_name view is queried.

  • FINAL_VALUE_HASH and FINAL_VALUE_NL are global query optimizer hints that specify the join method that should be used to obtain the lexical values for any query variables that are not used in a FILTER clause.

  • GRAPH_MATCH_UNNAMED=T allows unnamed triples (null G_ID) to be matched inside GRAPH clauses. That is, two triples will satisfy the graph join condition if their graphs are equal or if one or both of the graphs are null. This option may be useful when your dataset includes unnamed TBOX triples or unnamed entailed triples.

  • HINT0={<hint-string>} (pronounced and written "hint" and the number zero) specifies one or more keywords with hints to influence the execution plan and results of queries. Conceptually, a graph pattern with n triple patterns and referring to m distinct variables results in an (n+m)-way join: n-way self-join of the target RDF model or models and optionally the corresponding entailment, and then m joins with RDF_VALUE$ for looking up the values for the m variables. A hint specification affects the join order and join type used for the query execution.

    The hint specification, <hint-string>, uses keywords, some of which have parameters consisting of a sequence or set of aliases, or references, for individual triple patterns and variables used in the query. Aliases for triple patterns are of the form ti where i refers to the 0-based ordinal numbers of triple patterns in the query. For example, the alias for the first triple pattern in a query is t0, the alias for the second one is t1, and so on. Aliases for the variables used in a query are simply the names of those variables. Thus, ?x will be used in the hint specification as the alias for a variable ?x used in the graph pattern.

    Hints used for influencing query execution plans include LEADING(<sequence of aliases>), USE_NL(<set of aliases>), USE_HASH(<set of aliases>), and INDEX(<alias> <index_name>). These hints have the same format and basic meaning as hints in SQL statements, which are explained in Oracle Database SQL Language Reference.

    Example 1-10 shows the HINT0 option used in a SEM_MATCH query.

  • INF_ONLY=T queries only the entailed graph for the specified models and rulebases.

  • PLUS_RDFT=T can be used with SPARQL SELECT syntax (see Section 1.6.3, "Graph Patterns: Support for SPARQL SELECT Syntax") to additionally return a var$RDFTERM CLOB column for each projected query variable. The value for this column is equivalent to the result of SEM_APIS.COMPOSE_RDF_TERM(var, var$RDFVTYP, var$RDFLTYP, var$RDFLANG, var$RDFCLOB). When using this option, the return columns for each variable var will be var, var$RDFVID, var$_PREFIX, var$_SUFFIX, var$RDFVTYP, var$RDFCLOB, var$RDFLTYP, var$RDFLANG, and var$RDFTERM.

  • STRICT_DEFAULT=T restricts the default graph to unnamed triples when no dataset information is specified.

The graphs attribute specifies the set of named graphs from which to construct the default graph for a SEM_MACH query. Its data type is SEM_GRAPHS, which has the following definition: TABLE OF VARCHAR2(4000). The default value for this attribute is NULL. When graphs is NULL, the "union all" of all graphs in the set of query models is used as the default graph.

The named_graphs attribute specifies the set of named graphs that can be matched by a GRAPH clause. Its data type is SEM_GRAPHS, which has the following definition: TABLE OF VARCHAR2(4000). The default value for this attribute is NULL. When named_graphs is NULL, all named graphs in the set of query models can be matched by a GRAPH clause.

The SEM_MATCH table function returns an object of type ANYDATASET, with elements that depend on the input variables. In the following explanations, var represents the name of a variable used in the query. For each variable var, the result elements have the following attributes: var, var$RDFVID, var$_PREFIX, var$_SUFFIX, var$RDFVTYP, var$RDFCLOB, var$RDFLTYP, and var$RDFLANG.

In such cases, var has the lexical value bound to the variable, var$RDFVID has the VALUE_ID of the value bound to the variable, var$_PREFIX and var$_SUFFIX are the prefix and suffix of the value bound to the variable, var$RDFVTYP indicates the type of value bound to the variable (URI, LIT [literal], or BLN [blank node]), var$RDFCLOB has the lexical value bound to the variable if the value is a long literal, var$RDFLTYP indicates the type of literal bound if a literal is bound, and var$RDFLANG has the language tag of the bound literal if a literal with language tag is bound. var$RDFCLOB is of type CLOB, while all other attributes are of type VARCHAR2.

For a literal value or a blank node, its prefix is the value itself and its suffix is null. For a URI value, its prefix is the left portion of the value up to and including the rightmost occurrence of any of the three characters / (slash), # (pound), or : (colon), and its suffix is the remaining portion of the value to the right. For example, the prefix and suffix for the URI value http://www.example.org/family/grandParentOf are http://www.example.org/family/ and grandParentOf, respectively.

Along with columns for variable values, a SEM_MATCH query that uses SPARQL SELECT syntax returns one additional NUMBER column, SEM$ROWNUM, which can be used to ensure the correct result ordering for queries that involve a SPARQL ORDER BY clause.

Example 1-9 selects all grandfathers (grandparents who are male) and their grandchildren from the family model, using inferencing from both the RDFS and family_rb rulebases. (This example is an excerpt from Example 1-43 in Section 1.11.2.)

Example 1-9 SEM_MATCH Table Function

SELECT x, y
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male}',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

Example 1-10 is functionally the same as Example 1-9, but it adds the HINT0 option.

Example 1-10 HINT0 Option with SEM_MATCH Table Function

SELECT x, y
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male}',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_Aliases(SEM_ALIAS('','http://www.example.org/family/')),
    null,
    null,
    'HINT0={LEADING(t0 t1) USE_NL(?x ?y) GET_CANON_VALUE(?x ?y)}'));

Example 1-11 uses the Pathway/Genome BioPax ontology to get all chemical compound types that belong to both Proteins and Complexes:

Example 1-11 SEM_MATCH Table Function

SELECT t.r 
  FROM TABLE (SEM_MATCH ( 
      '{?r rdfs:subClassOf :Proteins .  
        ?r rdfs:subClassOf :Complexes}', 
      SEM_Models ('BioPax'), 
      SEM_Rulebases ('rdfs'), 
      SEM_Aliases (SEM_ALIAS('', 'http://www.biopax.org/release1/biopax-release1.owl')),
      NULL)) t;

As shown in Example 1-11, the search pattern for the SEM_MATCH table function is specified using SPARQL-like syntax where the variable starts with the question-mark character (?). In this example, the variable ?r must match to the same term, and thus is must be a subclass of both Proteins and Complexes.

To use the SEM_RELATED operator to query an OWL ontology, see Section 2.3.

When you are querying multiple models or querying one or more models and the corresponding entailment, consider using virtual models (explained in Section 1.3.8) because of the potential performance benefits.

1.6.1 Performing Queries with Incomplete or Invalid Entailments

You can query semantic data even when the relevant entailment does not have a valid status if you specify the string value INCOMPLETE or INVALID for the index_status attribute of the SEM_MATCH table function. (The entailment status is stored in the STATUS column of the MDSYS.SEM_RULES_INDEX_INFO view, which is described in Section 1.3.7. The SEM_MATCH table function is described in Section 1.6.)

The index_status attribute value affects the query behavior as follows:

  • If the entailment has a valid status, the query behavior is not affected by the value of the index_status attribute.

  • If you provide no value or specify a null value for index_status, the query returns an error if the entailment does not have a valid status.

  • If you specify the string INCOMPLETE for the index_status attribute, the query is performed if the status of the entailment is incomplete or valid.

  • If you specify the string INVALID for the index_status attribute, the query is performed regardless of the actual status of the entailment (invalid, incomplete, or valid).

However, the following considerations apply if the status of the entailment is incomplete or invalid:

  • If the status is incomplete, the content of an entailment may be approximate, because some triples that are inferable (due to the recent insertions into the underlying models) may not actually be present in the entailment, and therefore results returned by the query may be inaccurate.

  • If the status is invalid, the content of the entailment may be approximate, because some triples that are no longer inferable (due to recent modifications to the underlying models or rulebases, or both) may still be present in the entailment, and this may affect the accuracy of the result returned by the query. In addition to possible presence of triples that are no longer inferable, some inferable rows may not actually be present in the entailment.

1.6.2 Graph Patterns: Support for Curly Brace Syntax, and OPTIONAL, FILTER, UNION, and GRAPH Keywords

The SEM_MATCH table function accepts the syntax for the graph pattern in which a sequence of triple patterns is enclosed within curly braces. The period is usually required as a separator unless followed by the OPTIONAL, FILTER, UNION, or GRAPH keyword. With this syntax, you can do any combination of the following:

  • Use the OPTIONAL construct to retrieve results even in the case of a partial match

  • Use the FILTER construct to specify a filter expression in the graph pattern to restrict the solutions to a query

  • Use the UNION construct to match one of multiple alternative graph patterns

  • Use the GRAPH construct (explained in Section 1.6.2.1) to scope graph pattern matching to a set of named graphs

Example 1-12 is functionally the same as Example 1-9, but it uses the syntax with curly braces and a period to express a graph pattern in the SEM_MATCH table function.

Example 1-12 Curly Brace Syntax

SELECT x, y
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male}',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

Example 1-13 uses the OPTIONAL construct to modify Example 1-12, so that it also returns, for each grandfather, the names of the games that he plays or null if he does not play any games.

Example 1-13 Curly Brace Syntax and OPTIONAL Construct

SELECT x, y, game
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male . 
      OPTIONAL{?x :plays ?game} 
     }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null,
    null,
    'HINT0={LEADING(t0 t1) USE_NL(?x ?y)}'));

When multiple triple patterns are present in an OPTIONAL graph pattern, values for optional variables are returned only if a match is found for each triple pattern in the OPTIONAL graph pattern. Example 1-14 modifies Example 1-13 so that it returns, for each grandfather, the names of the games both he and his grandchildren play, or null if he and his grandchildren have no such games in common. It also uses global query optimizer hints to specify that triple patterns should be evaluated in order within each BGP and that a hash join should be used to join the root BGP with the OPTIONAL BGP.

Example 1-14 Curly Brace Syntax and Multi-Pattern OPTIONAL Construct

SELECT x, y, game
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male . 
      OPTIONAL{?x :plays ?game . ?y :plays ?game} 
     }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null,
    'ALL_ORDERED ALL_BGP_HASH'));

A single query can contain multiple OPTIONAL graph patterns, which can be nested or parallel. Example 1-15 modifies Example 1-14 with a nested OPTIONAL graph pattern. This example returns, for each grandfather, (1) the games he plays or null if he plays no games and (2) if he plays games, the ages of his grandchildren that play the same game, or null if he has no games in common with his grandchildren. Note that in Example 1-15 a value is returned for ?game even if the nested OPTIONAL graph pattern ?y :plays ?game . ?y :age ?age is not matched.

Example 1-15 Curly Brace Syntax and Nested OPTIONAL Construct

SELECT x, y, game, age
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male . 
      OPTIONAL{?x :plays ?game 
                          OPTIONAL {?y :plays ?game . ?y :age ?age} } 
     }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

Example 1-16 modifies Example 1-14 with a parallel OPTIONAL graph pattern. This example returns, for each grandfather, (1) the games he plays or null if he plays no games and (2) his email address or null if he has no email address. Note that, unlike nested OPTIONAL graph patterns, parallel OPTIONAL graph patterns are treated independently. That is, if an email address is found, it will be returned regardless of whether or not a game was found; and if a game was found, it will be returned regardless of whether an email address was found.

Example 1-16 Curly Brace Syntax and Parallel OPTIONAL Construct

SELECT x, y, game, email
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male . 
      OPTIONAL{?x :plays ?game}
      OPTIONAL{?x :email ?email} 
     }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

Example 1-17 uses the FILTER construct to modify Example 1-12, so that it returns grandchildren information for only those grandfathers who are residents of either NY or CA.

Example 1-17 Curly Brace Syntax and FILTER Construct

SELECT x, y
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male . ?x :residentOf ?z
       FILTER (?z = "NY"  || ?z = "CA")}',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

In addition to arithmetic operators (+, -, *, /), boolean operators and logical connectives (||, &&, !), and comparison operators (<, >, <=, >=, =, !=), several built-in functions are available for use in FILTER clauses. Table 1-12 lists built-in functions that you can use in the FILTER clause. In the Description column of Table 1-12, x, y, and z are arguments of the appropriate types.

Table 1-12 Built-in Functions Available for FILTER Clause

FunctionDescription

BOUND(variable)

BOUND(x) returns true if x is bound (that is, non-null) in the result, false otherwise.

DATATYPE(literal)

DATATYPE(x) returns a URI representing the datatype of x.

isBLANK(RDF term)

isBLANK(x) returns true if x is a blank node, false otherwise.

isIRI(RDF term)

isIRI(x) returns true if x is an IRI, false otherwise.

isLITERAL(RDF term)

isLiteral(x) returns true if x is a literal, false otherwise.

isURI(RDF term)

isURI(x) returns true if x is a URI, false otherwise.

LANG(literal)

LANG(x) returns a plain literal serializing the language tag of x.

langMATCHES(literal, literal)

langMATCHES(x, y) returns true if language tag x matches language range y, false otherwise.

REGEX(string, pattern)

REGEX(x,y) returns true if x matches the regular expression y, false otherwise. For more information about the regular expressions supported, see the Oracle Regular Expression Support appendix in Oracle Database SQL Language Reference.

REGEX(string, pattern, flags)

REGEX(x,y,z) returns true if x matches the regular expression y using the options given in z, false otherwise. Available options: 's' – dot all mode ('.' matches any character including the newline character); 'm' – multiline mode ('^' matches the beginning of any line and '$' matches the end of any line); 'i' – case insensitive mode; 'x' – remove whitespace characters from the regular expression before matching.

sameTERM(RDF term, RDF term)

sameTERM(x, y) returns true if x and y are the same RDF term, false otherwise.

STR(RDF term)

STR(x) returns a plain literal of the string representation of x (that is, what would be stored in the VALUE_NAME column of MDSYS.RDF_VALUE$ enclosed within double quotes).


See also the descriptions of the built-in functions defined in the SPARQL query language specification (http://www.w3.org/TR/rdf-sparql-query/), to better understand the built-in functions available in SEM_MATCH.

Example 1-18 uses the REGEX built-in function to select all grandfathers that have an Oracle email address. Note that backslash (\) characters in the regular expression pattern must be escaped in the query string; for example, \\. produces the following pattern: \.

Example 1-18 Curly Brace Syntax and FILTER with REGEX and STR Built-In Constructs

SELECT x, y, z
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male . ?x :email ?z
       FILTER (REGEX(STR(?z), "@oracle\\.com$"))}',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

Example 1-19 uses the UNION construct to modify Example 1-17, so that grandfathers are returned only if they are residents of NY or CA or own property in NY or CA, or if both conditions are true (they reside in and own property in NY or CA).

Example 1-19 Curly Brace Syntax and UNION and FILTER Constructs

SELECT x, y
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male 
       {{?x :residentOf ?z} UNION {?x :ownsPropertyIn ?z}}
       FILTER (?z = "NY"  || ?z = "CA")}',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

If you use the syntax with curly braces to express a graph pattern:

  • The query always returns canonical lexical forms for the matching values for the variables.

  • Any hints specified in the options argument using HINT0={<hint-string>} (explained in Section 1.6), should be constructed only on the basis of the portion of the graph pattern inside the root BGP. For example, the only valid aliases for use in a hint specification for the query in Example 1-13 are t0, t1, ?x, and ?y. Inline query optimizer hints can be used to influence other parts of the graph pattern (see Section 1.6.4).

  • The FILTER construct is not supported for variables bound to long literals.

1.6.2.1 GRAPH Keyword Support

A SEM_MATCH query is executed against an RDF Dataset. An RDF Dataset is a collection of graphs that includes one unnamed graph, known as the default graph, and one or more named graphs, which are identified by a URI. Graph patterns that appear inside a GRAPH clause are matched against the set of named graphs, and graph patterns that do not appear inside a graph clause are matched against the default graph. The graphs and named_graphs SEM_MATCH parameters are used to construct the default graph and set of named graphs for a given SEM_MATCH query. A summary of possible dataset configurations is shown in Table 1-13.

Table 1-13 SEM_MATCH graphs and named_graphs Values, and Resulting Dataset Configurations

Parameter ValuesDefault GraphSet of Named Graphs

graphs: NULL

named_graphs: NULL

Union All of all unnamed triples and all named graph triples. (But if the options parameter contains STRICT_DEFAULT=T, only unnamed triples are included in the default graph.)

All named graphs

graphs: NULL

named_graphs: {g1,…, gn}

Empty set

{g1,…, gn}

graphs: {g1,…, gm}

named_graphs: NULL

Union All of {g1,…, gm}

Empty set

graphs: {g1,…, gm}

named_graphs: {gn,…, gz}

Union All of {g1,…, gm}

{gn,…, gz}


Example 1-20 uses the GRAPH construct to scope graph pattern matching to a specific named graph. This example finds the names and email addresses of all people in the <http://www.example.org/family/Smith> named graph.

Example 1-20 Named Graph Construct

SELECT name, email
  FROM TABLE(SEM_MATCH(
    '{GRAPH :Smith {
       ?x :name ?name . ?x :email ?email } }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

In addition to URIs, variables can appear after the GRAPH keyword. Example 1-21 uses a variable, ?g, with the GRAPH keyword, and uses the named_graphs parameter to restrict the possible values of ?g to the <http://www.example.org/family/Smith> and <http://www.example.org/family/Jones> named graphs. Note that aliases specified in SEM_ALIASES argument can be used in the graphs and named_graphs parameters.

Example 1-21 Using the named_graphs Parameter

SELECT name, email
  FROM TABLE(SEM_MATCH(
    '{GRAPH ?g {
       ?x :name ?name . ?x :email ?email } }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null,null,null,null,
    SEM_GRAPHS('<http://www.example.org/family/Smith>',
               ':Jones')));

Example 1-22 uses the default graph to query the union of the <http://www.example.org/family/Smith> and <http://www.example.org/family/Jones> named graphs.

Example 1-22 Using the graphs Parameter

FROM TABLE(SEM_MATCH(
    '{?x :name ?name . ?x :email ?email }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null,null,null,
    SEM_GRAPHS('<http://www.example.org/family/Smith>', 
               ':Jones'),
    null));

See also the W3C SPARQL specification for more information on RDF Datasets and the GRAPH construct, specifically: http://www.w3.org/TR/rdf-sparql-query/#rdfDataset

1.6.3 Graph Patterns: Support for SPARQL SELECT Syntax

In addition to curly-brace graph patterns, SEM_MATCH allows fully-specified SPARQL SELECT queries in the query parameter. When using the SPARQL SELECT syntax option, SEM_MATCH supports the following query constructs: PREFIX, SELECT, SELECT DISTINCT, FROM, FROM NAMED, WHERE, ORDER BY, LIMIT, and OFFSET. Each SPARQL SELECT syntax query must include a SELECT clause and a graph pattern.

A key difference between curly-brace and SPARQL SELECT syntax when using SEM_MATCH is that only variables appearing in the SPARQL SELECT clause are returned from SEM_MATCH when using SPARQL SELECT syntax.

One additional column, SEM$ROWNUM, is returned from SEM_MATCH when using SPARQL SELECT syntax. This NUMBER column can be used to order the results of a SEM_MATCH query so that the result order matches the ordering specified by a SPARQL ORDER BY clause.

Example 1-23 uses the following SPARQL constructs:

  • SPARQL PREFIX clause to specify an abbreviation for the http://www.example.org/family/ and http://xmlns.com/foaf/0.1/ namespaces

  • SPARQL SELECT clause to specify the set of variables to project out of the query

  • SPARQL WHERE clause to specify the query graph pattern

Example 1-23 SPARQL PREFIX, SELECT, and WHERE Clauses

SELECT y, name
  FROM TABLE(SEM_MATCH(
    'PREFIX : <http://www.example.org/family/> 
     PREFIX foaf: <http://xmlns.com/foaf/0.1/>
     SELECT ?y ?name
     WHERE
     {?x :grandParentOf ?y . 
      ?x foaf:name ?name }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    null, null));

Example 1-23 returns the following columns: y, y$RDFVID, y$_PREFIX, y$_SUFFIX, y$RDFVTYP, y$RDFCLOB, y$RDFLTYP, y$RDFLANG, name, name$RDFVID, name$_PREFIX, name$_SUFFIX, name$RDFVTYP, name$RDFCLOB, name$RDFLTYP, name$RDFLANG, and SEM$ROWNUM.

The SPARQL SELECT clause specifies either (A) a sequence of variables and/or expressions, or (B) *, which projects all variables that appear in a specified triple pattern. Example 1-24 uses the SPARQL SELECT clause to select all variables that appear in a specified triple pattern.

Example 1-24 SPARQL SELECT * (All Variables in Triple Pattern)

SELECT x, y, name
  FROM TABLE(SEM_MATCH(
    'PREFIX : <http://www.example.org/family/> 
     PREFIX foaf: <http://xmlns.com/foaf/0.1/>
     SELECT *
     WHERE
     {?x :grandParentOf ?y . 
      ?x foaf:name ?name }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    null, null));

The DISTINCT keyword can be used after SELECT to remove duplicate result rows. Example 1-25 uses SELECT DISTINCT to select only the distinct names.

Example 1-25 SPARQL SELECT DISTINCT

SELECT name
  FROM TABLE(SEM_MATCH(
    'PREFIX : <http://www.example.org/family/> 
     PREFIX foaf: <http://xmlns.com/foaf/0.1/>
     SELECT DISTINCT ?name
     WHERE
     {?x :grandParentOf ?y . 
      ?x foaf:name ?name }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    null, null));

SPARQL FROM and FROM NAMED are used to specify the RDF dataset for a query. FROM clauses are used to specify the set of graphs that make up the default graph, and FROM NAMED clauses are used to specify the set of graphs that make up the set of named graphs. Example 1-26 uses FROM and FROM NAMED to select email addresses and friend of relationships from the union of the <http://www.friends.com/friends> and <http://www.contacts.com/contacts> graphs and grandparent information from the <http://www.example.org/family/Smith> and <http://www.example.org/family/Jones> graphs.

Example 1-26 RDF Dataset Specification Using FROM and FROM NAMED

SELECT x, y, z, email
  FROM TABLE(SEM_MATCH(
    'PREFIX : <http://www.example.org/family/> 
     PREFIX foaf: <http://xmlns.com/foaf/0.1/>
     PREFIX friends: <http://www.friends.com/>
     PREFIX contacts: <http://www.contacts.com/>
     SELECT *
     FROM friends:friends
     FROM contacts:contacts
     FROM NAMED :Smith
     FROM NAMED :Jones
     WHERE
     {?x foaf:frendOf ?y .
      ?x :email ?email .
      GRAPH ?g {
        ?x :grandParentOf ?z }
     }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    null, null));

The SPARQL ORDER BY clause can be used to order the results of SEM_MATCH queries. This clause specifies a sequence of comparators used to order the results of a given query. A comparator consists of an expression composed of variables, RDF terms, arithmetic operators (+, -, *, /), boolean operators and logical connectives (||, &&, !), comparison operators (<, >, <=, >=, =, !=), and any functions available for use in FILTER expressions.

In a SPARQL ORDER BY clause:

  • Single variable ordering conditions do not require enclosing parenthesis, but parentheses are required for more complex ordering conditions.

  • An optional ASC() or DESC() order modifier can be used to indicate the desired order (ascending or descending, respectively). Ascending is the default order.

  • When using SPARQL ORDER BY in SEM_MATCH, the containing SQL query should be ordered by SEM$ROWNUM to ensure that the desired ordering is maintained through any enclosing SQL blocks.

Example 1-27 uses a SPARQL ORDER BY clause to select all cameras, and it specifies ordering by descending type and ascending total price (price * (1 - discount) * (1 + tax)).

Example 1-27 SPARQL ORDER BY

SELECT *
  FROM TABLE(SEM_MATCH(
    'PREFIX : <http://www.example.org/electronics/> 
     SELECT * 
     WHERE
      {?x :price ?p .
       ?x :discount ?d .
       ?x :tax ?t .
       ?x :cameraType ?cType .
      }
     ORDER BY DESC(?cType) ASC(?p * (1-?d) * (1+?t))',
    SEM_Models('electronics'),
    SEM_Rulebases('RDFS'), 
    null, null))
ORDER BY SEM$ROWNUM;

SPARQL LIMIT and SPARQL OFFSET can be used to select different subsets of the query solutions. Example 1-28 uses SPARQL LIMIT to select the five cheapest cameras, and Example 1-29 uses SPARQL LIMIT and OFFSET to select the fifth through tenth cheapest cameras.

Example 1-28 SPARQL LIMIT

SELECT *
  FROM TABLE(SEM_MATCH(
    'PREFIX : <http://www.example.org/electronics/> 
     SELECT ?x ?cType ?p
     WHERE
      {?x :price ?p .
       ?x :cameraType ?cType .
      }
     ORDER BY ASC(?p)
     LIMIT 5',
    SEM_Models('electronics'),
    SEM_Rulebases('RDFS'), 
    null, null))
ORDER BY SEM$ROWNUM;

Example 1-29 SPARQL OFFSET

SELECT *
  FROM TABLE(SEM_MATCH(
    'PREFIX : <http://www.example.org/electronics/> 
     SELECT ?x ?cType ?p
     WHERE
      {?x :price ?p .
       ?x :cameraType ?cType .
      }
     ORDER BY ASC(?p)
     LIMIT 5
     OFFSET 5',
    SEM_Models('electronics'),
    SEM_Rulebases('RDFS'), 
    null, null))
ORDER BY SEM$ROWNUM;

See also the W3C SPARQL specification for more information on SPARQL PREFIX, SELECT, SELECT DISTINCT, FROM, FROM NAMED, WHERE, ORDER BY, LIMIT, and OFFSET constructs, specifically: http://www.w3.org/TR/rdf-sparql-query/

1.6.4 Inline Query Optimizer Hints

In SEM_MATCH, the SPARQL comment construct has been overloaded to allow inline HINT0 query optimizer hints. In SPARQL, the hash (#) character indicates that the remainder of the line is a comment. To associate an inline hint with a particular BGP, place a HINT0 hint string inside a SPARQL comment and insert the comment between the opening curly bracket ({) and the first triple pattern in the BGP. Inline hints enable you to influence the execution plan for each BGP in a query. Example 1-30 shows a query with inline query optimizer hints.

Example 1-30 Inline Query Optimizer Hints

SELECT x, y, hp, cp
  FROM TABLE(SEM_MATCH(
    '{ # HINT0={ LEADING(t0) USE_NL(?x ?y ?bd) }
      ?x :grandParentOf ?y . ?x rdf:type :Male . ?x :birthDate ?bd
      OPTIONAL { # HINT0={ LEADING(t0 t1) BGP_JOIN(USE_HASH) }
                 ?x :homepage ?hp . ?x :cellPhoneNum ?cp }
     }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));

The BGP_JOIN hint influences inter-BGP joins and has the following syntax: BGP_JOIN(<join_type>), where <join_type> is USE_HASH or USE_NL. Example 1-30 uses the BGP_JOIN(USE_HASH) hint to specify that a hash join should be used when joining the OPTIONAL BGP with its parent BGP.

Inline optimizer hints override any hints passed to SEM_MATCH through the options argument. For example, a global ALL_ORDERED hint applies to each BGP that does not specify an inline optimizer hint, but those BGPs with an inline hint use the inline hint instead of the ALL_ORDERED hint.

1.6.5 Full-Text Search

The Oracle-specific orardf:textContains SPARQL FILTER function uses full-text indexes on the MDSYS.RDF_VALUE$ table. This function has the following syntax (where orardf is a built-in prefix that expands to <http://xmlns.oracle.com/rdf/>):

orardf:textContains(variable, pattern)

The first argument to orardf:textContains must be a local variable (that is, a variable present in the BGP that contains the orardf:textContains filter), and the second argument must be a constant plain literal.

For example, orardf:textContains(x, y) returns true if x matches the expression y, where y is a valid expression for the Oracle Text SQL operator CONTAINS. For more information about such expressions, see Oracle Text Reference.

Before using orardf:textContains, you must create an Oracle Text index for the RDF network. To create such an index, invoke the SEM_APIS.ADD_DATATYPE_INDEX procedure as follows:

EXECUTE SEM_APIS.ADD_DATATYPE_INDEX('http://xmlns.oracle.com/rdf/text');

When performing large bulk loads into a semantic network with a text index, the overall load time may be faster if you drop the text index, perform the bulk load, and then re-create the text index. See Section 1.9 for more information about data type indexing.

After creating a text index, you can use the orardf:textContains FILTER function in SEM_MATCH queries. Example 1-31 uses orardf:textContains to find all grandfathers whose names start with the letter A or B.

Example 1-31 Full-Text Search

SELECT x, y, hp, cp
  FROM TABLE(SEM_MATCH(
    '{ ?x :grandParentOf ?y . ?x rdf:type :Male . ?x :name ?n 
       FILTER (orardf:textContains(?n, " A% | B% ")) }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
          null));

1.6.6 Spatial Support


Note:

Before using spatial support in Oracle Database Semantic Technologies, be sure that Oracle Spatial Patch 11828358: ORA-7445 ON SELECT WITH SDO_RELATE IN WHERE CLAUSE has been applied.

Oracle Database Semantic Technologies supports storage of spatial geometry data encoded as orageo:WKTLiteral typed literals, and it provides a set of query functions for spatial operations. (orageo is a built-in prefix that expands to: <http://xmlns.oracle.com/rdf/geo/>)

This section covers the following topics:

1.6.6.1 Representing Spatial Data in RDF

Spatial geometries can be represented in RDF as orageo:WKTLiteral typed literals. Example 1-32 shows the orageo:WKTLiteral encoding for a simple point geometry.

Example 1-32 Spatial Point Geometry Represented as orageo:WKTLiteral

"Point(-83.4 34.3)"^^<http://xmlns.oracle.com/rdf/geo/WKTLiteral>

orageo:WKTLiteral encodings consist of an optional spatial reference system URI, followed by a Well-Known Text (WKT) string that encodes a geometry value. The spatial reference system URI and the WKT string should be separated by a whitespace character.

Spatial reference system URIs have the following form <http://xmlns.oracle.com/rdf/geo/srid/{srid}>, where {srid} is a valid spatial reference system id from Oracle Spatial. If an orageo:WKTLiteral value does not include a spatial reference system URI, then the default spatial reference system, WGS84 Longitude-Latitude (URI <http://xmlns.oracle.com/rdf/geo/srid/8307>), is used. The same default spatial reference system is used when orageo:WKTLiteral values are encountered in a query string.

Several geometry types can be represented as orageo:WKTLiteral values, including point, linestring, polygon, polyhedral surface, triangle, TIN, multipoint, multi-linestring, multipolygon, and geometry collection. Up to 500,000 vertices per geometry are supported for two-dimensional geometries.

Example 1-33 shows some RDF spatial data (in N-triple format) encoded using orageo:WKTLiteral values. In this example, the first two geometries (in lot1) use the default coordinate system (SRID 8307), but the other two geometries (in lot2) specify SRID 8265.

Example 1-33 Spatial Data Encoded Using orageo:WKTLiteral Values

# spatial data for lot1 using the default WGS84 Longitude-Latitude spatial reference system
<urn:lot1> <urn:hasExactGeometry> "Polygon((-83.6 34.1, -83.6 34.5, -83.2 34.5, -83.2 34.1, -83.6 34.1))"^^<http://xmlns.oracle.com/rdf/geo/WKTLiteral> .
<urn:lot1> <urn:hasPointGeometry> "Point(-83.4 34.3)"^^<http://xmlns.oracle.com/rdf/geo/WKTLiteral> .
# spatial data for lot2 using the NAD83 Longitude-Latitude spatial reference system
<urn:lot2> <urn:hasExactGeometry> "<http://xmlns.oracle.com/rdf/geo/srid/8265> Polygon((-83.6  34.1, -83.6 34.3, -83.4 34.3, -83.4 34.1, -83.6 34.1))"^^<http://xmlns.oracle.com/rdf/geo/WKTLiteral> .
<urn:lot2> <urn:hasPointGeometry> "<http://xmlns.oracle.com/rdf/geo/srid/8265> Point(-83.5 34.2)"^^<http://xmlns.oracle.com/rdf/geo/WKTLiteral> .

For more information, see the chapter about coordinate systems (spatial reference systems) in Oracle Spatial Developer's Guide. See also the material about the WKT geometry representation in the Open Geospatial Consortium (OGC) Simple Features document: http://portal.opengeospatial.org/files/?artifact_id=25355

1.6.6.2 Indexing Spatial Data

Before you can use any of the SPARQL extension functions (introduced in Section 1.6.6.3) to query spatial data, you must create a spatial index on the RDF network by calling the SEM_APIS.ADD_DATATYPE_INDEX procedure.

When you create the spatial index, you must specify the following information:

  • SRID - The ID for the spatial reference system in which to create the spatial index. Any valid spatial reference system ID from Oracle Spatial can be used as an SRID value.

  • TOLERANCE – The tolerance value for the spatial index. Tolerance is a positive number indicating how close together two points must be to be considered the same point. The units for this value are determined by the default units for the SRID used (for example, meters for WGS84 Long-Lat). Tolerance is explained in detail in Oracle Spatial Developer's Guide.

  • DIMENSIONS - A text string encoding dimension information for the spatial index. Each dimension is represented by a sequence of three comma-separated values: name, minimum value, and maximum value. Each dimension is enclosed in parentheses, and the set of dimensions is enclosed by an outer parenthesis.

Example 1-34 adds a spatial data type index on the RDF network, specifying the WGS84 Longitude-Latitude spatial reference system, a tolerance value of 10 meters, and the recommended dimensions for the indexing of spatial data that uses this coordinate system. Note that the TOLERANCE, SRID, and DIMENSIONS keywords are case sensitive.

Example 1-34 Adding a Spatial Data Type Index on RDF Data

EXECUTE sem_apis.add_datatype_index('http://xmlns.oracle.com/rdf/geo/WKTLiteral',  options=>'TOLERANCE=10 SRID=8307 DIMENSIONS=((LONGITUDE,-180,180) (LATITUDE,-90,90))');

No more than one spatial data type index is supported for an RDF network. orageo:WKTLiteral values stored in the RDF network are automatically normalized to the spatial reference system used for the index, so a single spatial index can simultaneously support orageo:WKTLiteral values from different spatial reference systems. This coordinate transformation is done transparently for indexing and spatial computations. When orageo:WKTLiteral values are returned from a SEM_MATCH query, the original, untransformed geometry is returned.

For more information about spatial indexing, see the chapter about indexing and querying spatial data in Oracle Spatial Developer's Guide.

1.6.6.3 Querying Spatial Data

Several SPARQL extension functions are available for performing spatial queries in SEM_MATCH. For example, for spatial RDF data, you can find the area and perimeter (length) of a geometry, the distance between two geometries, and the centroid and the minimum bounding rectangle (MBR) of a geometry, and you can check various topological relationships between geometries.

Appendix B contains reference and usage information about the following functions:

1.6.7 Best Practices for Query Performance

This section describes some recommended practices for using the SEM_MATCH table function to query semantic data. It includes the following subsections:

1.6.7.1 FILTER Constructs Involving xsd:dateTime, xsd:date, and xsd:time

By default, SEM_MATCH complies with the XML Schema standard for comparison of xsd:date, xsd:time, and xsd:dateTime values. According to this standard, when comparing two calendar values c1 and c2 where c1 has an explicitly specified time zone and c2 does not have a specified time zone, c2 is converted into the interval [c2-14:00, c2+14:00]. If c2-14:00 <= c1 <= c2+14:00, then the comparison is undefined and will always evaluate to false. If c1 is outside this interval, then the comparison is defined.

However, the extra logic required to evaluate such comparisons (value with a time zone and value without a time zone) can significantly slow down queries with FILTER constructs that involve calendar values. For improved query performance, you can disable this extra logic by specifying FAST_DATE_FILTER=T in the options parameter of the SEM_MATCH table function. When FAST_DATE_FILTER=T is specified, all calendar values without time zones are assumed to be in Greenwich Mean Time (GMT).

Note that using FAST_DATE_FILTER=T does not affect query correctness when either (1) all calendar values in the data set have a time zone or (2) all calendar values in the data set do not have a time zone.

1.6.7.2 Function-Based Indexes for FILTER Constructs Involving Typed Literals

The evaluation of SEM_MATCH queries involving the FILTER construct often requires executing one or more SQL functions against the RDF_VALUE$ table. For example, the filter (?x < "1929-11-16Z"^^xsd:date) invokes the SEM_APIS.GETV$DATETZVAL function.

Function-based indexes can be used to improve the performance of queries that contain a filter condition involving a typed literal. For example, an xsd:date function-based index may speed up evaluation of the filter (?x < "1929-11-16Z"^^xsd:date).

Convenient interfaces are provided for creating, altering, and dropping these function-based indexes. For more information, see Section 1.9, "Using Data Type Indexes".

Note, however, that the existence of these function-based indexes on the MDSYS.RDF_VALUE$ table can significantly slow down bulk load operations. In many cases it may be faster to drop the indexes, perform the bulk load, and then re-create the indexes, as opposed to doing the bulk load with the indexes in place.

1.6.7.3 FILTER Constructs Involving Relational Expressions

The following recommendations apply to FILTER constructs involving relational expressions:

  • The sameTerm built-in function is more efficient than using = or != when comparing two variables in a FILTER clause, so (for example) use sameTerm(?a, ?b) instead of (?a = ?b) and use (!sameTerm(?a, ?b)) instead of (?a != ?b) whenever possible.

  • When comparing values in FILTER expressions, you may get better performance by reducing the use of negation. For example, it is more efficient to evaluate (?x <= "10"^^xsd:int) than it is to evaluate the expression (!(?x > "10"^^xsd:int)).

1.6.7.4 Optimizer Statistics and Dynamic Sampling

Having sufficient statistics for the query optimizer is critical for good query performance. In general, you should ensure that you have gathered basic statistics for the semantic network using the SEM_PERF.GATHER_STATS procedure (described in Chapter 11).

Due to the inherent flexibility of the RDF data model, static information may not produce optimal execution plans for SEM_MATCH queries. Dynamic sampling can often produce much better query execution plans. Dynamic sampling levels can be set at the session or system level using the optimizer_dynamic_sampling parameter, and at the individual query level using the dynamic_sampling(level) SQL query hint. In general, it is good to experiment with dynamic sampling levels between 3 and 6. For information about estimating statistics with dynamic sampling, see Oracle Database Performance Tuning Guide.

Example 1-35 uses a SQL hint for a dynamic sampling level of 6.

Example 1-35 SQL Hint for Dynamic Sampling

SELECT /*+ DYNAMIC_SAMPLING(6) */ x, y
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . 
      ?x rdf:type :Male . 
      ?x :birthDate ?bd }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null, null, '' ));

1.6.7.5 Multi-Partition Queries

The following recommendations apply to the use of multiple semantic models, semantic models plus entailments, and virtual models:

  • If you execute SEM_MATCH queries against multiple semantic models or against semantic models plus entailments, you can probably improve query performance if you create a virtual model (see Section 1.3.8) that contains all the models and entailments you are querying and then query this single virtual model.

  • Use the ALLOW_DUP=T query option. If you do not use this option, then an expensive (in terms of processing) duplicate-elimination step is required during query processing, in order to maintain set semantics for RDF data. However, if you use this option, the duplicate-elimination step is not performed, and this results in significant performance gains.

1.7 Loading and Exporting Semantic Data

To load semantic data into a model, use one or more of the following options:

  • Bulk load or append data into the semantic data store from a staging table, with each row containing the three components -- subject, predicate, and object -- of an RDF triple and optionally a named graph. This is explained in Section 1.7.1.

    This is the fastest option for loading large amounts of data; however, it cannot handle triples containing object values with more than 4000 bytes.

  • Batch load using a Java client interface to load or append data from an N-Triple format file into the semantic data store (see Section 1.7.2).

    This option is slower than bulk loading, but it handles triples containing object values with more than 4000 bytes. However, this option does not handle named graphs.

  • Load into the application table using SQL INSERT statements that call the SDO_RDF_TRIPLE_S constructor, which results in the corresponding RDF triple, possibly including a graph name, to be inserted into the semantic data store, as explained in Section 1.7.3.

    This option is convenient for loading small amounts of data. However, this option does not handle named graphs.

To export semantic data, see Section 1.7.4.

1.7.1 Bulk Loading Semantic Data Using a Staging Table

You can load semantic data (and optionally associated non-semantic data) in bulk using a staging table. Call the SEM_APIS.LOAD_INTO_STAGING_TABLE procedure (described in Chapter 9) to load the data, and you can have during the load operation to check for syntax correctness. Then, you can call the SEM_APIS.BULK_LOAD_FROM_STAGING_TABLE procedure to load the data into the semantic store from the staging table. (If the data was not parsed during the load operation into the staging table, you must specify the PARSE keyword in the flags parameter when you call the SEM_APIS.BULK_LOAD_FROM_STAGING_TABLE procedure.)

The following example shows the format for the staging table, including all required columns and the required names for these columns, plus the optional RDF$STC_graph column which must be included if one or more of the RDF triples to be loaded include a graph name:

CREATE TABLE stage_table (
                     RDF$STC_sub varchar2(4000) not null,
                     RDF$STC_pred varchar2(4000) not null,
                     RDF$STC_obj varchar2(4000) not null,
                     RDF$STC_graph varchar2(4000)
);

If you also want to load non-semantic data, specify additional columns for the non-semantic data in the CREATE TABLE statement. The non-semantic column names must be different from the names of the required columns. The following example creates the staging table with two additional columns (SOURCE and ID) for non-semantic attributes.

CREATE TABLE stage_table_with_extra_cols (
                     source VARCHAR2(4000),
                     id NUMBER,
                     RDF$STC_sub varchar2(4000) not null,
                     RDF$STC_pred varchar2(4000) not null,
                     RDF$STC_obj varchar2(4000) not null,
                     RDF$STC_graph varchar2(4000)
);

Note:

For either form of the CREATE TABLE statement, you may want to add the COMPRESS clause to use table compression, which will reduce the disk space requirements and may improve bulk-load performance.

Both the invoker and the MDSYS user must have the following privileges: SELECT privilege on the staging table, and INSERT privilege on the application table.

See also the following:

1.7.1.1 Loading the Staging Table

You can load semantic data into the staging table, as a preparation for loading it into the semantic store, in several ways. Some of the common ways are the following:

  • Using Oracle SQL*Loader to load the staging table, as described in Section 1.7.1.1.1

  • Using an external table to load the staging table, as described in Section 1.7.1.1.2

1.7.1.1.1 Loading N-Triple Format Data into a Staging Table Using SQL*Loader

You can use the SQL*Loader utility to parse and load semantic data into a staging table. If you installed the demo files from the Oracle Database Examples media (see Oracle Database Examples Installation Guide), a sample control file is available at $ORACLE_HOME/md/demo/network/rdf_demos/bulkload.ctl. You can modify and use this file if the input data is in N-Triple format.

Objects longer than 4000 bytes cannot be loaded. If you use the sample SQL*Loader control file, triples (rows) containing such long values will be automatically rejected and stored in a SQL*Loader "bad" file. However, you can load these rejected rows by inserting them into the application table using SQL INSERT statements (see Section 1.7.3).

1.7.1.1.2 Loading N-Quad Format Data into a Staging Table Using an External Table

You can use an Oracle external table to load N-Quad format data (extended triple having four components) into a staging table, as follows:

  1. Call the SEM_APIS.CREATE_SOURCE_EXTERNAL_TABLE procedure to create an external table, and then use the SQL STATEMENT ALTER TABLE to alter the external table to include the relevant input file name or names. You must have READ and WRITE privileges for the directory object associated with folder containing the input file or files.

  2. After you create the external table, grant the MDSYS user SELECT and INSERT privileges on the table.

  3. Call the SEM_APIS.LOAD_INTO_STAGING_TABLE procedure to populate the staging table.

  4. After the loading is finished, issue a COMMIT statement to complete the transaction.

Rows where the objects and graph URIs (combined) are longer than 4000 bytes will be rejected and stored in a "bad" file. However, you can load these rejected rows by inserting them into the application table using SQL INSERT statements (see Section 1.7.3).

Example 1-36 shows the use of an external table to load a staging table.

Example 1-36 Using an External Table to Load a Staging Table

-- Create a source external table (note: table names are case sensitive)
BEGIN
  sem_apis.create_source_external_table(
    source_table    => 'stage_table_source'
   ,def_directory   => 'DATA_DIR'
   ,bad_file        => 'CLOBrows.bad'
   );
END;
/
grant SELECT on "stage_table_source" to MDSYS;
 
-- Use ALTER TABLE to target the appropriate file(s)
alter table "stage_table_source" location ('demo_datafile.nt');
 
-- Load the staging table (note: table names are case sensitive)
BEGIN
  sem_apis.load_into_staging_table(
    staging_table => 'STAGE_TABLE'
   ,source_table  => 'stage_table_source'
   ,input_format  => 'N-QUAD');
END;
/

1.7.1.2 Recording Event Traces During Bulk Loading

If a table named RDF$ET_TAB exists in the invoker's schema and if the MDSYS user has been granted the INSERT and UPDATE privileges on this table, event traces for some of the tasks performed during executions of the SEM_APIS.BULK_LOAD_FROM_STAGING_TABLE procedure will be added to the table. You may find the content of this table useful if you ever need to report any problems in bulk load. The RDF$ET_TAB table must be created as follows:

CREATE TABLE RDF$ET_TAB (
  proc_sid VARCHAR2(30), 
  proc_sig VARCHAR2(200),
  event_name varchar2(200),
  start_time timestamp,
  end_time timestamp,
  start_comment varchar2(1000) DEFAULT NULL,
  end_comment varchar2(1000) DEFAULT NULL
);
GRANT INSERT, UPDATE on RDF$ET_TAB to MDSYS;

1.7.2 Batch Loading N-Triple Format Semantic Data Using the Java API


Note:

The Java class oracle.spatial.rdf.client.BatchLoader described in this section has been deprecated, and it does not support loading of N-Quad data.

You are instead encouraged to use the bulk loading capabilities of the Jena Adapter, as described in Section 7.11, "Bulk Loading Using the Jena Adapter".


You can perform a batch (bulk) load operation for N-Triple format semantic data using the Java class oracle.spatial.rdf.client.BatchLoader, which is packaged in <ORACLE_HOME>/md/jlib/sdordf.jar. Before performing a batch load operation, ensure that the following are true:

  • The semantic data is in N-Triple format. (Several tools are available for converting RDF/XML to N-Triple format; see the Oracle Technology Network or perform a Web search for information about RDF/XML to N-Triple conversion.)

  • Oracle Database Release 11, with Oracle Spatial, is installed, and partitioning is enabled.

  • A semantic technologies network, an application table, and its corresponding semantic model have been created in the database.

  • The CLASSPATH definition includes ojdbc6.jar.

  • You are using JDK version 1.5 or later. (You can use the Java version packaged under <ORACLE_HOME>/jdk/bin.)

To run the oracle.spatial.rdf.client.BatchLoader class, use a command (on a single command line) in the following general form (replacing the sample example database connection information with your own connection information).

  • Linux systems:

    java -Ddb.user=scott -Ddb.password=password -Ddb.host=127.0.0.1 -Ddb.port=1522 -Ddb.sid=orcl -classpath ${ORACLE_HOME}/md/jlib/sdordf.jar:${ORACLE_HOME}/jdbc/lib/ojdbc6.jar oracle.spatial.rdf.client.BatchLoader <N-TripleFile> <tablename> <tablespaceName> <modelName>
    
  • Windows systems:

    java -Ddb.user=scott -Ddb.password=password -Ddb.host=127.0.0.1 -Ddb.port=1522 -Ddb.sid=orcl -classpath %ORACLE_HOME%\md\jlib\sdordf.jar;%ORACLE_HOME%\jdbc\lib\ojdbc6.jar oracle.spatial.rdf.client.BatchLoader <N-TripleFile> <tablename> <tablespaceName> <modelName>
    

The values for -Ddb.user and -Ddb.password must correspond either to the owner of the model <modelName> or to a DBA user.

By default, BatchLoader assumes there are at least two columns, a column named ID of type NUMBER and a column named TRIPLE of type SDO_RDF_TRIPLE_S, in the user's application table. However, you can override the default names by using the JVM properties -DidColumn=<idColumnName> and -DtripleColumn=<tripleColumnName>. Note that the ID column is not required; and to prevent BatchLoader from generating a sequence-like identifier in the ID column for each triple inserted, specify the JVM property -DjustTriple=true.

If the application table is not empty and if you want the batch loading to be done in append mode, specify an additional JVM property: -Dappend=true. Moreover, in append mode you might want to choose a different starting value for ID column in user's application table, and to accomplish this you can add the JVM property -DstartID=<startingIntegerValue> to the command line. By default, the ID column starts at 1 and is increased sequentially as new triples are inserted into the application table.

To skip the first n triples in <N-TripleFile>, add the JVM property -Dskip=<numberOfTriplesSkipped> to the command line.

To load an N-Triple file with a character set different from the default, specify the JVM property -Dcharset=<charsetName>. For example, -Dcharset="UTF-8" will recognize UTF-8 encoding. However, for UTF-8 characters to be stored properly in the N-Triple file, the Oracle database must be configured to use a corresponding universal character set, such as AL32UTF8.

The BatchLoader class supports loading an N-Triple file in compressed format. If the <N-TripleFile> has a file extension of .zip or .jar, the file will be uncompressed and loaded at the same time.

1.7.3 Loading Semantic Data Using INSERT Statements

To load semantic data using INSERT statements, the data should be encoded using < > (angle brackets) for URIs, _: (underscore colon) for blank nodes, and " " (quotation marks) for literals. Spaces are not allowed in URIs or blank nodes. Use the SDO_RDF_TRIPLE_S constructor to insert the data, as described in Section 1.5.1. You must have INSERT privilege on the application table.


Note:

If URIs are not encoded with < > and literals with " ", statements will still be processed. However, the statements will take longer to load, since they will have to be further processed to determine their VALUE_TYPE values.

The following example includes statements with URIs, a blank node, a literal, a literal with a language tag, and a typed literal:

INSERT INTO nsu_data VALUES (SDO_RDF_TRIPLE_S('nsu', '<http://nature.example.com/nsu/rss.rdf>',
  '<http://purl.org/rss/1.0/title>', '"Nature''s Science Update"'));
INSERT INTO nsu_data VALUES (SDO_RDF_TRIPLE_S('nsu', '_:BNSEQN1001A',
  '<http://www.w3.org/1999/02/22-rdf-syntax-ns#type>', 
  '<http://www.w3.org/1999/02/22-rdf-syntax-ns#Seq>'));
INSERT INTO nsu_data VALUES (SDO_RDF_TRIPLE_S('nsu',
  '<http://nature.example.com/cgi-taf/dynapage.taf?file=/nature/journal/v428/n6978/index.html>',
  '<http://purl.org/dc/elements/1.1/language>', '"English"@en-GB'));
INSERT INTO nature VALUES (SDO_RDF_TRIPLE_S('nsu', '<http://dx.doi.org/10.1038/428004b>',
  '<http://purl.org/dc/elements/1.1/date>', '"2004-03-04"^^xsd:date'));

To convert semantic XML data to INSERT statements, you can edit the sample rss2insert.xsl XSLT file to convert all the features in the semantic data XML file. The blank node constructor is used to insert statements with blank nodes. After editing the XSLT, download the Xalan XSLT processor (http://xml.apache.org/xalan) and follow the installation instructions. To convert a semantic data XML file to INSERT statements using your edited version of the rss2insert.xsl file, use a command in the following format:

java org.apache.xalan.xslt.Process –in input.rdf -xsl rss2insert.xsl –out output.nt

1.7.3.1 Loading Data into Named Graphs Using INSERT Statements

To load an RDF triple with a non-null graph name using an INSERT statement, you must append the graph name, enclosed within angle brackets (< >), after the model name and colon (:) separator character, as shown in the following example:

INSERT INTO articles_rdf_data VALUES (99,
  SDO_RDF_TRIPLE_S ('articles:<http://examples.com/ns#Graph1>',
    '<http://nature.example.com/Article101>',
    '<http://purl.org/dc/elements/1.1/creator>',
    '"John Smith"'));

1.7.4 Exporting Semantic Data

This section contains the following topics related to exporting semantic data, that is, retrieving semantic data from Oracle Database:

1.7.4.1 Retrieving Semantic Data from an Application Table

Semantic data can be retrieved from an application table using the member functions of SDO_RDF_TRIPLE_S, as shown in Example 1-37.

Example 1-37 Retrieving Semantic Data from an Application Table

--
-- Retrieves model-graph, subject, predicate, and object
--
SQL> SELECT a.triple.GET_MODEL() AS model_graph, a.triple.GET_SUBJECT() AS sub, a.triple.GET_PROPERTY() pred, a.triple.GET_OBJECT() obj FROM articles_rdf_data a where id in (2,99);
 
MODEL_GRAPH
------------------------------------------------------------
SUB
------------------------------------------------------------
PRED
------------------------------------------------------------
OBJ
------------------------------------------------------------
ARTICLES
<http://nature.example.com/Article1>
<http://purl.org/dc/elements/1.1/creator>
"Jane Smith"
 
ARTICLES:<http://examples.com/ns#Graph1>
<http://nature.example.com/Article101>
<http://purl.org/dc/elements/1.1/creator>
"John Smith"
 
--
-- Retrieves graph, subject, predicate, and object
--
SQL> select (case sep_pos when 0 then NULL else substr(model_graph,sep_pos+1) end) graph, sub, pred, obj from (SELECT instr(a.triple.GET_MODEL(),':') AS sep_pos, a.triple.GET_MODEL() AS model_graph, a.triple.GET_SUBJECT() AS sub, a.triple.GET_PROPERTY() pred, a.triple.GET_OBJECT() obj FROM articles_rdf_data a where id in (2,99));
 
GRAPH
--------------------------------------------------------------------------------
SUB
------------------------------------------------------------
PRED
------------------------------------------------------------
OBJ
------------------------------------------------------------
 
<http://nature.example.com/Article1>
<http://purl.org/dc/elements/1.1/creator>
"Jane Smith"
 
<http://examples.com/ns#Graph1>
<http://nature.example.com/Article101>
<http://purl.org/dc/elements/1.1/creator>
"John Smith"

1.7.4.2 Retrieving Semantic Data from an RDF Model

Semantic data can be retrieved from an RDF model using the SEM_MATCH table function (described in Section 1.6), as shown in Example 1-38.

Example 1-38 Retrieving Semantic Data from an RDF Model

--
-- Retrieves graph, subject, predicate, and object
--
SQL> select to_char(g$rdfterm) graph, to_char(x$rdfterm) sub, to_char(p$rdfterm) pred, y$rdfterm obj from table(sem_match('Select ?g ?x ?p ?y FROM NAMED <http://examples.com/ns#Graph1> {GRAPH ?g {?x ?p ?y}}',sem_models('articles'),null,null,null,null,' GRAPH_MATCH_UNNAMED=T PLUS_RDFT=T '));
 
GRAPH
------------------------------------------------------------
SUB
------------------------------------------------------------
PRED
------------------------------------------------------------
OBJ
---------------------------------------------------------------------------
<http://examples.com/ns#Graph1>
_:m99g3C687474703A2F2F6578616D706C65732E636F6D2F6E73234772617068313Egmb2
<http://purl.org/dc/elements/1.1/creator>
_:m99g3C687474703A2F2F6578616D706C65732E636F6D2F6E73234772617068313Egmb1
 
<http://examples.com/ns#Graph1>
<http://nature.example.com/Article102>
<http://purl.org/dc/elements/1.1/creator>
_:m99g3C687474703A2F2F6578616D706C65732E636F6D2F6E73234772617068313Egmb1
 
<http://examples.com/ns#Graph1>
<http://nature.example.com/Article101>
<http://purl.org/dc/elements/1.1/creator>
"John Smith"
 
<http://nature.example.com/Article1>
<http://purl.org/dc/elements/1.1/creator>
"Jane Smith"

1.7.4.3 Removing Model and Graph Information from Retrieved Blank Node Identifiers

Blank node identifiers retrieved during the retrieval of semantic data) can be trimmed to remove the occurrence of model and graph information using the transformations shown in the code excerpt in Example 1-39, which are applicable to a VARCHAR2 (for example, subject component) and a CLOB (for example, object component), respectively.

Example 1-39 Retrieving Semantic Data from an Application Table

--
-- Transformation on column “sub VARCHAR2” 
-- holding blank node identifier values
--
Select (case substr(sub,1,2) when '_:' then '_:' || substr(sub,instr(sub,'m',1,2)+1) else sub end) from …
--
-- Transformation on column “obj CLOB” 
-- holding blank node identifier values
--
Select (case dbms_lob.substr(obj,2,1) when '_:' then to_clob('_:' || substr(obj,instr(obj,'m',1,2)+1)) else obj end) from …

Example 1-40 shows the results obtained after using these two transformations in Example 1-39 on the sub and obj columns, respectively, using the semantic data retrieval query described in Section 1.7.4.2.

Example 1-40 Results from Applying Transformations from Example 1-39

--
-- Results obtained by applying transformations on the sub and pred cols
-- 
SQL> select (case substr(sub,1,2) when '_:' then '_:' || substr(sub,instr(sub,'m',1,2)+1) else sub end) sub, pred, (case dbms_lob.substr(obj,2,1) when '_:' then to_clob('_:' || substr(obj,instr(obj,'m',1,2)+1)) else obj end) obj from (select to_char(g$rdfterm) graph, to_char(x$rdfterm) sub, to_char(p$rdfterm) pred, y$rdfterm obj from table(sem_match('Select ?g ?x ?p ?y FROM NAMED <http://examples.com/ns#Graph1> {GRAPH ?g {?x ?p ?y}}',sem_models('articles'),null,null,null,null,' GRAPH_MATCH_UNNAMED=T PLUS_RDFT=T ')));
 
SUB
------------------------------------------------------------
PRED
------------------------------------------------------------
OBJ
---------------------------------------------------------------------------
_:b2
<http://purl.org/dc/elements/1.1/creator>
_:b1
 
<http://nature.example.com/Article102>
<http://purl.org/dc/elements/1.1/creator>
_:b1

1.8 Using Semantic Network Indexes

Semantic network indexes are nonunique B-tree indexes that you can add, alter, and drop for use with models and entailments in a semantic network. You can use such indexes to tune the performance of SEM_MATCH queries on the models and entailments in the network. As with any indexes, semantic network indexes enable index-based access that suits your query workload. This can lead to substantial performance benefits, such as in the following example scenarios:

  • If your graph pattern is '{<John> ?p <Mary>}', you may want to have a usable 'CSP' or 'SCP' index for the target model or models and on the corresponding entailment, if used in the query.

  • If your graph pattern is '{?x <talksTo> ?y . ?z ?p ?y}', you may want to have a usable semantic network index on the relevant model or models and entailment, with C as the leading key (for example, 'C' or 'CPS').

However, using semantic network indexes can affect overall performance by increasing the time required for DML, load, and inference operations.

You can create and manage semantic network indexes using the following subprograms:

All of these subprograms have an index_code parameter, which can contain any sequence of the following letters (without repetition): P, C, S, G, M. These letters used in the index_code correspond to the following columns in the SEMM_* and SEMI_* views: P_VALUE_ID, CANON_END_NODE_ID, START_NODE_ID, G_ID, and MODEL_ID.

The SEM_APIS.ADD_SEM_INDEX procedure creates a semantic network index that results in creation of a nonunique B-tree index in UNUSABLE status for each of the existing models and entailments. The name of the index is RDF_LNK_<index_code>_IDX and the index is owned by MDSYS. This operation is allowed only if the invoker has DBA role. The following example shows creation of the PSCGM index with the following key: <P_VALUE_ID, START_NODE_ID, CANON_END_NODE_ID, G_ID, MODEL_ID>.

EXECUTE SEM_APIS.ADD_SEM_INDEX('PSCGM');

After you create a semantic network index, each of the corresponding nonunique B-tree indexes is in the UNUSABLE status, because making it usable can cause significant time and resources to be used, and because subsequent index maintenance operations might involve performance costs that you do not want to incur. You can make a semantic network index usable or unusable for specific models or entailments that you own by calling the SEM_APIS.ALTER_SEM_INDEX_ON_MODEL and SEM_APIS.ALTER_SEM_INDEX_ON_ENTAILMENT procedures and specifying 'REBUILD' or 'UNUSABLE' as the command parameter. Thus, you can experiment by making different semantic network indexes usable and unusable, and checking for any differences in performance. For example, the following statement makes the PSCGM index usable for the FAMILY model:

EXECUTE SEM_APIS.ALTER_SEM_INDEX_ON_MODEL('FAMILY','PSCGM','REBUILD');

Also note the following:

  • Independent of any semantic network indexes that you create, when a semantic network is created, one of the indexes that is automatically created is an index that you can manage by referring to the index_code as 'PSCGM' when you call the subprograms mentioned in this section.

  • When you create a new model or a new entailment, a new nonunique B-tree index is created for each of the semantic network indexes, and each such B-tree index is in the USABLE status.

  • Including the MODEL_ID column in a semantic network index key (by including 'M' in the index_code value) may improve query performance. This is particularly relevant when virtual models are used.

1.8.1 MDSYS.SEM_NETWORK_INDEX_INFO View

Information about all network indexes on models and entailments is maintained in the MDSYS.SEM_NETWORK_INDEX_INFO view, which includes (a partial list) the columns shown in Table 1-14 and one row for each network index.

Table 1-14 MDSYS.SEM_NETWORK_INDEX_INFO View Columns (Partial List)

Column NameData TypeDescription

NAME

VARCHAR2(30)

Name of the RDF model or entailment

TYPE

VARCHAR2(10)

Type of object on which the index is built: MODEL, ENTAILMENT, or NETWORK

ID

NUMBER

ID number for the model or entailment, or zero (0) for an index on the network

INDEX_CODE

VARCHAR2(25)

Code for the index (for example, PSCGM).

INDEX_NAME

VARCHAR2(30)

Name of the index (for example, RDF_LNK_PSCGM_IDX)

LAST_REFRESH

TIMESTAMP(6) WITH TIME ZONE

Timestamp for the last time this content was refreshed


In addition to the columns listed in Table 1-14, the MDSYS.SEM_NETWORK_INDEX_INFO view contains columns from the ALL_INDEXES and ALL_IND_PARTITIONS views (both described in Oracle Database Reference), including:

  • From the ALL_INDEXES view: UNIQUENESS, COMPRESSION, PREFIX_LENGTH

  • From the ALL_IND_PARTITIONS view: STATUS, TABLESPACE_NAME, BLEVEL, LEAF_BLOCKS, NUM_ROWS, DISTINCT_KEYS, AVG_LEAF_BLOCKS_PER_KEY, AVG_DATA_BLOCKS_PER_KEY, CLUSTERING_FACTOR, SAMPLE_SIZE, LAST_ANALYZED

Note that the information in the MDSYS.SEM_NETWORK_INDEX_INFO view may sometimes be stale. You can refresh this information by using the SEM_APIS.REFRESH_SEM_NETWORK_INDEX_INFO procedure.

1.9 Using Data Type Indexes

Data type indexes are indexes on the values of typed literals stored in a semantic network. These indexes may significantly improve the performance of SEM_MATCH queries involving certain types of FILTER expressions. For example, a data type index on xsd:dateTime literals may speed up evaluation of the filter (?x < "1929-11-16T13:45:00Z"^^xsd:dateTime). Indexes can be created for several data types, which are listed in Table 1-15.

Table 1-15 Data Types for Data Type Indexing

Data Type URIOracle TypeIndex Type

http://www.w3.org/2001/XMLSchema#decimal

NUMBER

Non-unique B-tree (creates a single index for all xsd numeric types, including xsd:float, xsd:double, and xsd:decimal and all of its subtypes)

http://www.w3.org/2001/XMLSchema#string

VARCHAR2

Non-unique B-tree (creates a single index for xsd:string typed literals and plain literals)

http://www.w3.org/2001/XMLSchema#time

TIMESTAMP WITH TIMEZONE

Non-unique B-tree

http://www.w3.org/2001/XMLSchema#date

TIMESTAMP WITH TIMEZONE

Non-unique B-tree

http://www.w3.org/2001/XMLSchema#dateTime

TIMESTAMP WITH TIMEZONE

Non-unique B-tree

http://xmlns.oracle.com/rdf/text

(Not applicable)

CTXSYS.CONTEXT

http://xmlns.oracle.com/rdf/geo/WKTLiteral

SDO_GEOMETRY

MDSYS.SPATIAL_INDEX


The suitability of data type indexes depends on your query workload. Data type indexes on xsd data types can be used for filters that compare a variable with a constant value, and are particularly useful when queries have an unselective graph pattern with a very selective filter condition. Appropriate data type indexes are required for queries with spatial or text filters.

While data type indexes improve query performance, overhead from incremental index maintenance can degrade the performance of DML and bulk load operations on the semantic network. For bulk load operations, it may often be faster to drop data type indexes, perform the bulk load, and then re-create the data type indexes.

You can add, alter, and drop data type indexes using the following procedures, which are described in Chapter 9:

Information about existing data type indexes is maintained in the MDSYS.SEM_DTYPE_INDEX_INFO view, which has the columns shown in Table 1-16 and one row for each data type index.

Table 1-16 MDSYS.SEM_DTYPE_INDEX_INFO View Columns

Column NameData TypeDescription

DATATYPE

VARCHAR2(51)

Data type URI

INDEX_NAME

VARCHAR2(30)

Name of the index

STATUS

VARCHAR2(8)

Status of the index: USABLE or UNUSABLE

TABLESPACE_NAME

VARCHAR2(30)

Tablespace for the index


You can use the HINT0 hint to ensure that data type indexes are used during query evaluation, as shown in Example 1-41, which finds all grandfathers who were born before November 16, 1929.

Example 1-41 Using HINT0 to Ensure Use of Data Type Index

SELECT x, y
  FROM TABLE(SEM_MATCH(
    '{?x :grandParentOf ?y . ?x rdf:type :Male . ?x :birthDate ?bd
       FILTER (?bd <= "1929-11-15T23:59:59Z"^^xsd:dateTime) }',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null, null, 
    'HINT0={ LEADING(?bd) INDEX(?bd rdf_v$dateTime_idx) } 
             FAST_DATE_FILTER=T' ));

1.10 Quick Start for Using Semantic Data

To work with semantic data in an Oracle database, follow these general steps:

  1. Create a tablespace for the system tables. You must be connected as a user with appropriate privileges to create the tablespace. The following example creates a tablespace named RDF_TBLSPACE:

    CREATE TABLESPACE rdf_tblspace
     DATAFILE '/oradata/orcl/rdf_tblspace.dat' SIZE 1024M REUSE
     AUTOEXTEND ON NEXT 256M MAXSIZE UNLIMITED
     SEGMENT SPACE MANAGEMENT AUTO;
    
  2. Create a semantic data network.

    Creating a semantic data network adds semantic data support to an Oracle database. You must create a semantic data network as a user with DBA privileges, specifying a valid tablespace with adequate space. Create the network only once for an Oracle database.

    The following example creates a semantic data network using a tablespace named RDF_TBLSPACE (which must already exist):

    EXECUTE SEM_APIS.CREATE_SEM_NETWORK('rdf_tblspace');
    
  3. Connect as the database user under whose schema you will store your semantic data; do not perform the following steps while connected as SYS, SYSTEM, or MDSYS.

  4. Create a table to store references to the semantic data. (You do not need to be connected as a user with DBA privileges for this step and the remaining steps.)

    This table must contain a column of type SDO_RDF_TRIPLE_S, which will contain references to all data associated with a single model.

    The following example creates a table named ARTICLES_RDF_DATA:

    CREATE TABLE articles_rdf_data (id NUMBER, triple SDO_RDF_TRIPLE_S);
    
  5. Create a model.

    When you create a model, you specify the model name, the table to hold references to semantic data for the model, and the column of type SDO_RDF_TRIPLE_S in that table.

    The following command creates a model named ARTICLES, which will use the table created in the preceding step.

    EXECUTE SEM_APIS.CREATE_SEM_MODEL('articles', 'articles_rdf_data', 'triple');
    
  6. Where possible, create Oracle database indexes on conditions that will be specified in the WHERE clause of SELECT statements, to provide better performance for direct queries against the application table's SDO_RDF_TRIPLE_S column. (These indexes are not relevant if the SEM_MATCH table function is being used.) The following example creates such indexes:

    -- Create indexes on the subjects, properties, and objects
    -- in the ARTICLES_RDF_DATA table.
    CREATE INDEX articles_sub_idx ON articles_rdf_data (triple.GET_SUBJECT());
    CREATE INDEX articles_prop_idx ON articles_rdf_data (triple.GET_PROPERTY());
    CREATE INDEX articles_obj_idx ON articles_rdf_data (TO_CHAR(triple.GET_OBJECT()));
    

After you create the model, you can insert triples into the table, as shown in the examples in Section 1.11.

1.11 Semantic Data Examples (PL/SQL and Java)

This section contains the following PL/SQL examples:

In addition to the examples in this guide, see the sample code at http://www.oracle.com/technetwork/indexes/samplecode/semantic-sample-522114.html.

1.11.1 Example: Journal Article Information

This section presents a simplified PL/SQL example of model for statements about journal articles. Example 1-42 contains descriptive comments, refer to concepts that are explained in this chapter, and uses functions and procedures documented in Chapter 9.

Example 1-42 Using a Model for Journal Article Information

-- Basic steps:
-- After you have connected as a privileged user and called 
-- SEM_APIS.CREATE_SEM_NETWORK to add RDF support,
-- connect as a regular database user and do the following.
-- 1. For each desired model, create a table to hold its data.
-- 2. For each model, create a model (SEM_APIS.CREATE_RDF_MODEL).
-- 3. For each table to hold semantic data, insert data into the table.
-- 4. Use various subprograms and consructors.
 
-- Create the table to hold data for the model.
CREATE TABLE articles_rdf_data (id NUMBER, triple SDO_RDF_TRIPLE_S);
 
-- Create the model.
EXECUTE SEM_APIS.CREATE_RDF_MODEL('articles', 'articles_rdf_data', 'triple');
 
-- Information to be stored about some fictitious articles:
-- Article1, titled "All about XYZ" and written by Jane Smith, refers 
--   to Article2 and Article3.
-- Article2, titled "A review of ABC" and written by Joe Bloggs, 
--   refers to Article3.
-- Seven SQL statements to store the information. In each statement:
-- Each article is referred to by its complete URI The URIs in
--   this example are fictitious.
-- Each property is referred to by the URL for its definition, as 
--   created by the Dublin Core Metadata Initiative.
 
-- Insert rows into the table.
 
-- Article1 has the title "All about XYZ".
INSERT INTO articles_rdf_data VALUES (1,
  SDO_RDF_TRIPLE_S ('articles','http://nature.example.com/Article1',
    'http://purl.org/dc/elements/1.1/title','All about XYZ'));
 
-- Article1 was created (written) by Jane Smith.
INSERT INTO articles_rdf_data VALUES (2,
  SDO_RDF_TRIPLE_S ('articles','http://nature.example.com/Article1',
    'http://purl.org/dc/elements/1.1/creator',
    'Jane Smith'));
 
-- Article1 references (refers to) Article2.
INSERT INTO articles_rdf_data VALUES (3,
  SDO_RDF_TRIPLE_S ('articles',
    'http://nature.example.com/Article1',
    'http://purl.org/dc/terms/references',
    'http://nature.example.com/Article2'));
 
-- Article1 references (refers to) Article3.
INSERT INTO articles_rdf_data VALUES (4,
  SDO_RDF_TRIPLE_S ('articles',
    'http://nature.example.com/Article1',
    'http://purl.org/dc/terms/references',
    'http://nature.example.com/Article3'));
 
-- Article2 has the title "A review of ABC".
INSERT INTO articles_rdf_data VALUES (5,
  SDO_RDF_TRIPLE_S ('articles',
    'http://nature.example.com/Article2',
    'http://purl.org/dc/elements/1.1/title',
    'A review of ABC'));
 
-- Article2 was created (written) by Joe Bloggs.
INSERT INTO articles_rdf_data VALUES (6,
  SDO_RDF_TRIPLE_S ('articles',
    'http://nature.example.com/Article2',
    'http://purl.org/dc/elements/1.1/creator',
    'Joe Bloggs'));
 
-- Article2 references (refers to) Article3.
INSERT INTO articles_rdf_data VALUES (7,
  SDO_RDF_TRIPLE_S ('articles',
    'http://nature.example.com/Article2',
    'http://purl.org/dc/terms/references',
    'http://nature.example.com/Article3'));
 
COMMIT;
 
-- Query semantic data.
 
SELECT SEM_APIS.GET_MODEL_ID('articles') AS model_id FROM DUAL;

SELECT SEM_APIS.GET_TRIPLE_ID(
  'articles',
  'http://nature.example.com/Article2',
  'http://purl.org/dc/terms/references',
  'http://nature.example.com/Article3') AS RDF_triple_id FROM DUAL;
 
SELECT SEM_APIS.IS_TRIPLE(
  'articles',
  'http://nature.example.com/Article2',
  'http://purl.org/dc/terms/references',
  'http://nature.example.com/Article3') AS is_triple FROM DUAL;
 
-- Use SDO_RDF_TRIPLE_S member functions in queries.
 
SELECT a.triple.GET_TRIPLE() AS triple 
  FROM articles_rdf_data a WHERE a.id = 1;
SELECT a.triple.GET_SUBJECT() AS subject 
  FROM articles_rdf_data a WHERE a.id = 1;
SELECT a.triple.GET_PROPERTY() AS property 
  FROM articles_rdf_data a WHERE a.id = 1;
SELECT a.triple.GET_OBJECT() AS object 
  FROM articles_rdf_data a WHERE a.id = 1;

1.11.2 Example: Family Information

This section presents a simplified PL/SQL example of a model for statements about family tree (genealogy) information. Example 1-42 contains descriptive comments, refer to concepts that are explained in this chapter, and uses functions and procedures documented in Chapter 9.

The family relationships in this example reflect the family tree shown in Figure 1-3. This figure also shows some of the information directly stated in the example: Cathy is the sister of Jack, Jack and Tom are male, and Cindy is female.

Figure 1-3 Family Tree for RDF Example

Description of Figure 1-3 follows
Description of "Figure 1-3 Family Tree for RDF Example"

Example 1-43 Using a Model for Family Information

-- Basic steps:
-- After you have connected as a privileged user and called 
-- SEM_APIS.CREATE_SEM_NETWORK to enable RDF support,
-- connect as a regular database user and do the following.
-- 1. For each desired model, create a table to hold its data.
-- 2. For each model, create a model (SEM_APIS.CREATE_RDF_MODEL).
-- 3. For each table to hold semantic data, insert data into the table.
-- 4. Use various subprograms and constructors.
 
-- Create the table to hold data for the model.
CREATE TABLE family_rdf_data (id NUMBER, triple SDO_RDF_TRIPLE_S);
 
-- Create the model.
execute SEM_APIS.create_rdf_model('family', 'family_rdf_data', 'triple');
 
-- Insert rows into the table. These express the following information:
-----------------
-- John and Janice have two children, Suzie and Matt.
-- Matt married Martha, and they have two children:
--   Tom (male, height 5.75) and Cindy (female, height 06.00).
-- Suzie married Sammy, and they have two children:
--   Cathy (height 5.8) and Jack (male, height 6).
 
-- Person is a class that has two subslasses: Male and Female.
-- parentOf is a property that has two subproperties: fatherOf and motherOf.
-- siblingOf is a property that has two subproperties: brotherOf and sisterOf.
-- The domain of the fatherOf and brotherOf properties is Male.
-- The domain of the motherOf and sisterOf properties is Female.
------------------------
 
-- John is the father of Suzie.
INSERT INTO family_rdf_data VALUES (1, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/John', 
'http://www.example.org/family/fatherOf', 
'http://www.example.org/family/Suzie'));
 
-- John is the father of Matt.
INSERT INTO family_rdf_data VALUES (2, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/John', 
'http://www.example.org/family/fatherOf', 
'http://www.example.org/family/Matt'));
 
-- Janice is the mother of Suzie.
INSERT INTO family_rdf_data VALUES (3, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Janice', 
'http://www.example.org/family/motherOf', 
'http://www.example.org/family/Suzie'));
 
-- Janice is the mother of Matt.
INSERT INTO family_rdf_data VALUES (4, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Janice', 
'http://www.example.org/family/motherOf', 
'http://www.example.org/family/Matt'));
 
-- Sammy is the father of Cathy.
INSERT INTO family_rdf_data VALUES (5, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Sammy', 
'http://www.example.org/family/fatherOf', 
'http://www.example.org/family/Cathy'));
 
-- Sammy is the father of Jack.
INSERT INTO family_rdf_data VALUES (6, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Sammy', 
'http://www.example.org/family/fatherOf', 
'http://www.example.org/family/Jack'));
 
-- Suzie is the mother of Cathy.
INSERT INTO family_rdf_data VALUES (7, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Suzie', 
'http://www.example.org/family/motherOf', 
'http://www.example.org/family/Cathy'));
 
-- Suzie is the mother of Jack.
INSERT INTO family_rdf_data VALUES (8, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Suzie', 
'http://www.example.org/family/motherOf', 
'http://www.example.org/family/Jack'));
 
-- Matt is the father of Tom.
INSERT INTO family_rdf_data VALUES (9, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Matt', 
'http://www.example.org/family/fatherOf', 
'http://www.example.org/family/Tom'));
 
-- Matt is the father of Cindy
INSERT INTO family_rdf_data VALUES (10, 
SD=zO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Matt', 
'http://www.example.org/family/fatherOf', 
'http://www.example.org/family/Cindy'));
 
-- Martha is the mother of Tom.
INSERT INTO family_rdf_data VALUES (11, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Martha', 
'http://www.example.org/family/motherOf', 
'http://www.example.org/family/Tom'));
 
-- Martha is the mother of Cindy. 
INSERT INTO family_rdf_data VALUES (12, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Martha', 
'http://www.example.org/family/motherOf', 
'http://www.example.org/family/Cindy'));
 
-- Cathy is the sister of Jack.
INSERT INTO family_rdf_data VALUES (13, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Cathy', 
'http://www.example.org/family/sisterOf', 
'http://www.example.org/family/Jack'));
 
-- Jack is male.
INSERT INTO family_rdf_data VALUES (14, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Jack', 
'http://www.w3.org/1999/02/22-rdf-syntax-ns#type',
'http://www.example.org/family/Male'));
 
-- Tom is male.
INSERT INTO family_rdf_data VALUES (15, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Tom', 
'http://www.w3.org/1999/02/22-rdf-syntax-ns#type',
'http://www.example.org/family/Male'));
 
-- Cindy is female.
INSERT INTO family_rdf_data VALUES (16, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Cindy', 
'http://www.w3.org/1999/02/22-rdf-syntax-ns#type',
'http://www.example.org/family/Female'));
 
-- Person is a class.
INSERT INTO family_rdf_data VALUES (17, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Person', 
'http://www.w3.org/1999/02/22-rdf-syntax-ns#type',
'http://www.w3.org/2000/01/rdf-schema#Class'));
 
-- Male is a subclass of Person.
INSERT INTO family_rdf_data VALUES (18, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Male', 
'http://www.w3.org/2000/01/rdf-schema#subClassOf',
'http://www.example.org/family/Person'));
 
-- Female is a subclass of Person. 
INSERT INTO family_rdf_data VALUES (19, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Female', 
'http://www.w3.org/2000/01/rdf-schema#subClassOf',
'http://www.example.org/family/Person'));
 
-- siblingOf is a property.
INSERT INTO family_rdf_data VALUES (20, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/siblingOf', 
'http://www.w3.org/1999/02/22-rdf-syntax-ns#type',
'http://www.w3.org/1999/02/22-rdf-syntax-ns#Property'));
 
-- parentOf is a property.
INSERT INTO family_rdf_data VALUES (21, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/parentOf', 
'http://www.w3.org/1999/02/22-rdf-syntax-ns#type',
'http://www.w3.org/1999/02/22-rdf-syntax-ns#Property'));
 
-- brotherOf is a subproperty of siblingOf.
INSERT INTO family_rdf_data VALUES (22, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/brotherOf', 
'http://www.w3.org/2000/01/rdf-schema#subPropertyOf',
'http://www.example.org/family/siblingOf'));
 
-- sisterOf is a subproperty of siblingOf.
INSERT INTO family_rdf_data VALUES (23, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/sisterOf', 
'http://www.w3.org/2000/01/rdf-schema#subPropertyOf',
'http://www.example.org/family/siblingOf'));
 
-- A brother is male.
INSERT INTO family_rdf_data VALUES (24, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/brotherOf', 
'http://www.w3.org/2000/01/rdf-schema#domain',
'http://www.example.org/family/Male'));
 
-- A sister is female.
INSERT INTO family_rdf_data VALUES (25, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/sisterOf', 
'http://www.w3.org/2000/01/rdf-schema#domain',
'http://www.example.org/family/Female'));
 
-- fatherOf is a subproperty of parentOf.
INSERT INTO family_rdf_data VALUES (26, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/fatherOf', 
'http://www.w3.org/2000/01/rdf-schema#subPropertyOf',
'http://www.example.org/family/parentOf'));
 
-- motherOf is a subproperty of parentOf.
INSERT INTO family_rdf_data VALUES (27, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/motherOf', 
'http://www.w3.org/2000/01/rdf-schema#subPropertyOf',
'http://www.example.org/family/parentOf'));
 
-- A father is male.
INSERT INTO family_rdf_data VALUES (28, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/fatherOf', 
'http://www.w3.org/2000/01/rdf-schema#domain',
'http://www.example.org/family/Male'));
 
-- A mother is female.
INSERT INTO family_rdf_data VALUES (29, 
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/motherOf', 
'http://www.w3.org/2000/01/rdf-schema#domain',
'http://www.example.org/family/Female'));
 
-- Use SET ESCAPE OFF to prevent the caret (^) from being
-- interpreted as an escape character. Two carets (^^) are
-- used to represent typed literals.
SET ESCAPE OFF;
 
-- Cathy's height is 5.8 (decimal).
INSERT INTO family_rdf_data VALUES (30,
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Cathy', 
'http://www.example.org/family/height',
'"5.8"^^xsd:decimal'));
 
-- Jack's height is 6 (integer).
INSERT INTO family_rdf_data VALUES (31,
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Jack', 
'http://www.example.org/family/height',
'"6"^^xsd:integer'));
 
-- Tom's height is 05.75 (decimal).
INSERT INTO family_rdf_data VALUES (32,
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Tom', 
'http://www.example.org/family/height',
'"05.75"^^xsd:decimal'));
 
-- Cindy's height is 06.00 (decimal).
INSERT INTO family_rdf_data VALUES (33,
SDO_RDF_TRIPLE_S('family', 
'http://www.example.org/family/Cindy', 
'http://www.example.org/family/height',
'"06.00"^^xsd:decimal'));
 
COMMIT;
 
-- RDFS inferencing in the family model
BEGIN
  SEM_APIS.CREATE_ENTAILMENT(
    'rdfs_rix_family',
    SEM_Models('family'),
    SEM_Rulebases('RDFS'));
END;
/
 
-- Select all males from the family model, without inferencing.
SELECT m
  FROM TABLE(SEM_MATCH(
    '(?m rdf:type :Male)',
    SEM_Models('family'),
    null,
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));
 
-- Select all males from the family model, with RDFS inferencing.
SELECT m
  FROM TABLE(SEM_MATCH(
    '(?m rdf:type :Male)',
    SEM_Models('family'),
    SDO_RDF_Rulebases('RDFS'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));
 
-- General inferencing in the family model
 
EXECUTE SEM_APIS.CREATE_RULEBASE('family_rb');
 
INSERT INTO mdsys.semr_family_rb VALUES(
  'grandparent_rule',
  '(?x :parentOf ?y) (?y :parentOf ?z)',
  NULL,
  '(?x :grandParentOf ?z)', 
  SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')));
 
COMMIT;
 
-- Because a new rulebase has been created, and it will be used in the
-- entailment, drop the preceding entailment and then re-create it.
EXECUTE SEM_APIS.DROP_ENTAILMENT ('rdfs_rix_family');
 
-- Re-create the entailment.
BEGIN
  SEM_APIS.CREATE_ENTAILMENT(
    'rdfs_rix_family',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'));
END;
/
 
-- Select all grandfathers and their grandchildren from the family model, 
-- without inferencing. (With no inferencing, no results are returned.)
SELECT x grandfather, y grandchild
  FROM TABLE(SEM_MATCH(
    '(?x :grandParentOf ?y) (?x rdf:type :Male)',
    SEM_Models('family'),
    null, 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));
 
-- Select all grandfathers and their grandchildren from the family model.
-- Use inferencing from both the RDFS and family_rb rulebases.
SELECT x grandfather, y grandchild
  FROM TABLE(SEM_MATCH(
    '(?x :grandParentOf ?y) (?x rdf:type :Male)',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')),
    null));
 
-- Set up to find grandfathers of tall (>= 6) grandchildren
-- from the family model, with RDFS inferencing and
-- inferencing using the "family_rb" rulebase.
 
UPDATE mdsys.semr_family_rb SET
  antecedents = '(?x :parentOf ?y) (?y :parentOf ?z) (?z :height ?h)',
  filter = '(h >= 6)',
  aliases = SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/'))
WHERE rule_name = 'GRANDPARENT_RULE';
 
-- Because the rulebase has been updated, drop the preceding entailment, 
-- and then re-create it.
EXECUTE SEM_APIS.DROP_ENTAILMENT ('rdfs_rix_family');
 
-- Re-create the entailment.
BEGIN
  SEM_APIS.CREATE_ENTAILMENT(
    'rdfs_rix_family',
    SEM_Models('family'),
    SEM_Rulebases('RDFS','family_rb'));
END;
/
 
-- Find the entailment that was just created (that is, the
-- one based on the specified model and rulebases).
SELECT SEM_APIS.LOOKUP_ENTAILMENT(SEM_MODELS('family'),
  SEM_RULEBASES('RDFS','family_rb')) AS lookup_entailment FROM DUAL;
 
-- Select grandfathers of tall (>= 6) grandchildren, and their
-- tall grandchildren.
SELECT x grandfather, y grandchild
  FROM TABLE(SEM_MATCH(
    '(?x :grandParentOf ?y) (?x rdf:type :Male)',
    SEM_Models('family'),
    SEM_RuleBases('RDFS','family_rb'), 
    SEM_ALIASES(SEM_ALIAS('','http://www.example.org/family/')), 
    null));

1.12 Software Naming Changes for Semantic Technologies

Because the support for semantic data has been expanded beyond the original focus on RDF, the names of many software objects (PL/SQL packages, functions and procedures, system tables and views, and so on) have been changed as of Oracle Database Release 11.1. In most cases, the change is to replace the string RDF with SEM. although in some cases it may be to replace SDO_RDF with SEM.

All valid code that used the pre-Release 11.1 names will continue to work; your existing applications will not be broken. However, it is suggested that you change old applications to use new object names, and you should use the new names for any new applications. This manual will document only the new names.

Table 1-17 lists the old and new names for some objects related to support for semantic technologies, in alphabetical order by old name.

Table 1-17 Semantic Technology Software Objects: Old and New Names

Old NameNew Name

RDF_ALIAS data type

SEM_ALIAS

RDF_MODEL$ view

SEM_MODEL$

RDF_RULEBASE_INFO view

SEM_RULEBASE_INFO

RDF_RULES_INDEX_DATASETS view

SEM_RULES_INDEX_DATASETS

RDF_RULES_INDEX_INFO view

SEM_RULES_INDEX_INFO

RDFI_rules-index-name view

SEMI_rules-index-name

RDFM_model-name view

SEMM_model-name

RDFR_rulebase-name view

SEMR_rulebase-name

SDO_RDF package

SEM_APIS

SDO_RDF_INFERENCE package

SEM_APIS

SDO_RDF_MATCH table function

SEM_MATCH

SDO_RDF_MODELS data type

SEM_MODELS

SDO_RDF_RULEBASES data type

SEM_RULEBASES


1.13 For More Information About Semantic Technologies

For more information about Oracle Database semantic technology support and related topics, you may find the following resources helpful:

PK.PK.AOEBPS/indexing_for_docs.htm Semantic Indexing for Documents

4 Semantic Indexing for Documents

Information extractors locate and extract meaningful information from unstructured documents. The ability to search for documents based on this extracted information is a significant improvement over the keyword-based searches supported by the full-text search engines.

Semantic indexing for documents introduces an index type that can make use of information extractors and annotators to semantically index documents stored in relational tables. Documents indexed semantically can be searched using SEM_CONTAINS operator within a standard SQL query. The search criteria for these documents are expressed using SPARQL query patterns that operate on the information extracted from the documents, as in the following example.

SELECT docId
FROM   Newsfeed
WHERE  SEM_CONTAINS (article, 
     ' { ?org    rdf:type            typ:Organization  . 
         ?org    pred:hasCategory    cat:BusinessFinance } ', ..) = 1

The key components that facilitate Semantic Indexing for documents in an Oracle Database include:

  • Extensible information extractor framework, which allows third-party information extractors to be plugged into the database

  • SEM_CONTAINS operator to identify documents of interest, based on their extracted information, using standard SQL queries

  • SEM_CONTAINS_SELECT ancillary operator to return relevant information about the documents identified using SEM_CONTAINS operator

  • SemContext index type to interact with the information extractor and manage the information extracted from a document set in an index structure and to facilitate semantically meaningful searches on the documents

The application program interface (API) for managing extractor policies and semantic indexes created for documents is provided in the SEM_RDFCTX PL/SQL package. Chapter 12 provides the reference information about the subprograms in SEM_RDFCTX package.

This chapter contains the following major sections:

4.1 Information Extractors for Semantically Indexing Documents

Information extractors process unstructured documents and extract meaningful information from them, often using natural-language processing engines with the aid of ontologies. The quality and the completeness of information extracted from a document vary from one extractor to another. Some extractors simply identify the entities (such as names of persons, organizations, and geographic locations from a document), while the others attempt to identify the relationships among the identified entities and additional description for those entities. You can search for a specific document from a large set when the information extracted from the documents is maintained as a semantic index.

You can use an information extractor to create a semantic index on the documents stored in a column of a relational table. An extensible framework allows any third-party information extractor that is accessible from the database to be plugged into the database. An object type created for an extractor encapsulates the extraction logic, and has methods to configure the extractor and receive information extracted from a given document in RDF/XML format.

An abstract type MDSYS.RDFCTX_EXTRACTOR defines the common interfaces to all information extractors. An implementation of this abstract type interacts with a specific information extractor to produce RDF/XML for a given document. An implementation for this type can access a third-party information extractor that either is available as a database application or is installed on the network (accessed using Web service callouts). Example 4-1 shows the definition of the RDFCTX_EXTRACTOR abstract type.

Example 4-1 RDFCTX_EXTRACTOR Abstract Type Definition

create or replace type rdfctx_extractor authid current_user as object (
  extr_type        VARCHAR2(32),
  member function  getDescription return VARCHAR2,
  member function  rdfReturnType return VARCHAR2,
  member function  getContext(attribute VARCHAR2) return VARCHAR2,
  member procedure startDriver,
  member function  extractRDF(document CLOB,
                              docId    VARCHAR2) return CLOB,
  member function  extractRdf(document CLOB,
                              docId    VARCHAR2,
                              params   VARCHAR2,
                              options  VARCHAR2 default NULL) return CLOB
  member function  batchExtractRdf(docCursor        SYS_REFCURSOR,
                              extracted_info_table  VARCHAR2,
                              params                VARCHAR2,
                              partition_name        VARCHAR2 default NULL,
                              docId                 VARCHAR2 default NULL,
                              preferences           SYS.XMLType default NULL,
                              options               VARCHAR2 default NULL)  
                              return CLOB,
  member procedure closeDriver
) not instantiable not final
/

A specific implementation of the RDFCTX_EXTRACTOR type sets an identifier for the extractor type in the extr_type attribute, and it returns a short description for the extractor type using getDescription method. All implementations of this abstract type return the extracted information as RDF triples. In the current release, the RDF triples are expected to be serialized using RDF/XML format, and therefore the rdfReturnType method should return 'RDF/XML'.

An extractor type implementation uses the extractRDF method to encapsulate the extraction logic, possibly by invoking external information extractor using proprietary interfaces, and returns the extracted information in RDF/XML format. When a third-party extractor uses some proprietary XML Schema to capture the extracted information, an XML style sheet can be used to generate an equivalent RDF/XML. The startDriver and closeDriver methods can perform any housekeeping operations pertaining to the information extractor. The optional params parameter allows the extractor to obtain additional information about the type of extraction needed (for example, the desired quality of extraction).

Optionally, an extractor type implementation may support a batch interface by providing an implementation of the batchExtractRdf member function. This function accepts a cursor through the input parameter docCursor and typically uses that cursor to retrieve each document, extract information from the document, and then insert the extracted information into (the specified partition identified by the partition_name partition of the extracted_info_table table. The preferences parameter is used to obtain the preferences value associated with the policy (as described in Section 4.8 and in the SEM_RDFCTX.CREATE_POLICY reference section).

The getContext member function accepts an attribute name and returns the value for that attribute. Currently this function is used only for extractors supporting the batch interface. The attribute names and corresponding possible return values are the following:

  • For the BATCH_SUPPORT attribute, the return values are 'YES' or 'NO' depending on whether the extractor supports the batch interface.

  • For the DBUSER attribute, the return value is the name of a database user that will connect to the database to retrieve rows from the cursor (identified by the docCursor parameter) and that will write to the table extracted_info_table.

This information is used for granting appropriate privileges to the table being indexed and the table extracted_info_table.

The startDriver and closeDriver methods can perform any housekeeping operations pertaining to the information extractor.

An extractor type for the General Architecture for Text Engineering (GATE) engine is defined as a subtype of the RDFCTX_EXTRACTOR type. The implementation of this extractor type sends the documents to a GATE engine over a TCP connection, receives annotations extracted by the engine in XML format, and converts this proprietary XML document to an RDF/XML document. For more information on configuring a GATE engine to work with Oracle Database, see Section 4.10. For an example of creating a new information extractor, see Section 4.11.

Information extractors that are deployed as Web services can be invoked from the database by extending the RDFCTX_WS_EXTRACTOR type, which is a subtype of the RDFCTX_EXTRACTOR type. The RDFCTX_WS_EXTRACTOR type encapsulates the Web service callouts in the extractRDF method; specific implementations for network-based extractors can reuse this implementation by setting relevant attribute values in the type constructor.

Thomson Reuters Calais is an example of a network-based information extractor that can be accessed using web-service callouts. The CALAIS_EXTRACTOR type, which is a subtype of the RDFCTX_WS_EXTRACTOR type, encapsulates the Calais extraction logic, and it can be used to semantically index the documents. The CALAIS_EXTRACTOR type must be configured for the database instance before it can be used to create semantic indexes, as explained in Section 4.9.

4.2 Extractor Policies

An extractor policy is a named dictionary entity that determines the characteristics of a semantic index that is created using the policy. Each extractor policy refers, directly or indirectly, to an instance of an extractor type. An extractor policy with a direct reference to an extractor type instance can be used to compose other extractor policies that include additional RDF models for ontologies.

The following example creates a basic extractor policy created using the GATE extractor type:

begin
  sem_rdfctx.create_policy (policy_name => 'SEM_EXTR',
                            extractor   => mdsys.gatenlp_extractor());
end;
/

The following example creates a dependent extractor policy that combines the metadata extracted by the policy in the preceding example with a user-defined RDF model named geo_ontology:

begin
  sem_rdfctx.create_policy (policy_name => 'SEM_EXTR_PLUS_GEOONT',
                            base_policy => 'SEM_EXTR',
                            user_models => SEM_MODELS ('geo_ontology'));
end;
/

You can use an extractor policy to create one or more semantic indexes on columns that store unstructured documents, as explained in Section 4.3.

4.3 Semantically Indexing Documents

Textual documents stored in a CLOB or VARCHAR2 column of a relational table can be indexed using the MDSYS.SEMCONTEXT index type, to facilitate semantically meaningful searches. The extractor policy specified at index creation determines the information extractor used to semantically index the documents. The extracted information, captured as a set of RDF triples for each document, is managed in the semantic data store. Each instance of the semantic index is associated with a system-generated RDF model, which maintains the RDF triples extracted from the corresponding documents.

The following example creates a semantic index named ArticleIndex on the textual documents in the ARTICLE column of the NEWSFEED table, using the extractor policy named SEM_EXTR:

CREATE INDEX ArticleIndex on Newsfeed (article)
   INDEXTYPE IS mdsys.SemContext PARAMETERS ('SEM_EXTR');

The RDF model created for an index is managed internally and it is not associated with an application table. The triples stored in such model are automatically maintained for any modifications (such as update, insert, or delete) made to the documents stored in the table column. Although a single RDF model is used to index all documents stored in a table column, the triples stored in the model maintain references to the documents from which they are extracted; therefore, all the triples extracted from a specific document form an individual graph within the RDF model. The documents that are semantically indexed can then be searched using a SPARQL query pattern that operates on the triples extracted from the documents.

When creating a semantic index for documents, you can use a basic extractor policy or a dependent policy, which may include one or more user-defined RDF models. When you create an index with a dependent extractor policy, the document search pattern specified using SPARQL could span the triples extracted from the documents as well as those defined in user-defined models.

You can create an index using multiple extractor policies, in which case the triples extracted by the corresponding extractors are maintained separately in distinct RDF models. A document search query using one such index can select the specific policy to be used for answering the query. For example, an extractor policy named CITY_EXTR can be created to extract the names of the cities from a given document, and this extractor policy can be used in combination with the SEM_EXTR policy to create a semantic index, as in the following example:

CREATE INDEX ArticleIndex on Newsfeed (article)
   INDEXTYPE IS mdsys.SemContext PARAMETERS ('SEM_EXTR CITY_EXTR');

The first extractor policy in the PARAMETERS list is considered to be the default policy if a query does not refer to a specific policy; however, you can change the default extractor policy for a semantic index by using the SEM_RDFCTX.SET_DEFAULT_POLICY procedure, as in the following example:

begin
  sem_rdfctx.set_default_policy (index_name => 'ArticleIndex',
                                 policy_name => 'CITY_EXTR');
end;
/

4.4 SEM_CONTAINS and Ancillary Operators

You can use the SEM_CONTAINS operator in a standard SQL statement to search for documents or document references that are stored in relational tables. This operator has the following syntax:

SEM_CONTAINS(
  column   VARCHAR2 / CLOB,
  sparql   VARCHAR2,
  policy   VARCHAR2,
  aliases  SEM_ALIASES,
  index_status  NUMBER,
  ancoper  NUMBER
 ) RETURN NUMBER;

The column and sparql attributes attribute are required. The other attributes are optional (that is, each can be a null value).

The column attribute identifies a VARCHAR2 or CLOB column in a relational table that stores the documents or references to documents that are semantically indexed. An index of type MDSYS.SEMCONTEXT must be defined in this column for the SEM_CONTAINS operator to use.

The sparql attribute is a string literal that defines the document search criteria, expressed in SPARQL format.

The optional policy attribute specifies the name of an extractor policy, usually to override the default policy. A semantic document index can have one or more extractor policies specified at index creation, and one of these policies is the default, which is used if the policy attribute is null in the call to SEM_CONTAINS.

The optional aliases attribute identifies one or more namespaces, including a default namespace, to be used for expansion of qualified names in the query pattern. Its data type is SEM_ALIASES, which has the following definition: TABLE OF SEM_ALIAS, where each SEM_ALIAS element identifies a namespace ID and namespace value. The SEM_ALIAS data type has the following definition: (namespace_id VARCHAR2(30), namespace_val VARCHAR2(4000))

The optional index_status attribute is relevant only when a dependent policy involving one or more entailments is being used for the SEM_CONTAINS invocation. The index_status value identifies the minimum required validity status of the entailments. The possible values are 0 (for VALID, the default), 1 (for INCOMPLETE), and 2 (for INVALID).

The optional ancoper attribute specifies a number as the binding to be used when the SEM_CONTAINS_SELECT ancillary operator is used with this operator in a query. The number specified for the ancoper attribute should be the same as number specified for the operbind attribute in the SEM_CONTAINS_SELECT ancillary operator.

The SEM_CONTAINS operator returns 1 for each document instance matching the specified search criteria, and returns 0 for all other cases.

For more information about using the SEM_CONTAINS operator, including an example, see Section 4.5.

4.4.1 SEM_CONTAINS_SELECT Ancillary Operator

You can use the SEM_CONTAINS_SELECT ancillary operator to return additional information about each document that matches some search criteria. This ancillary operator has a single numerical attribute (operbind) that associates an instance of the SEM_CONTAINS_SELECT ancillary operator with a SEM_CONTAINS operator by using the same value for the binding. This ancillary operator returns an object of type CLOB that contains the additional information from the matching document, formatted in SPARQL Query Results XML format.

The SEM_CONTAINS_SELECT ancillary operator has the following syntax:

SEM_CONTAINS_SELECT(
  operbind  NUMBER
 ) RETURN CLOB;

For more information about using the SEM_CONTAINS_SELECT ancillary operator, including examples, see Section 4.6.

4.4.2 SEM_CONTAINS_COUNT Ancillary Operator

You can use the SEM_CONTAINS_COUNT ancillary operator for a SEM_CONTAINS operator invocation. For each matched document, it returns the count of matching subgraphs for the SPARQL graph pattern specified in the SEM_CONTAINS invocation.

The SEM_CONTAINS_COUNT ancillary operator has the following syntax:

SEM_CONTAINS_COUNT(
  operbind  NUMBER
 ) RETURN NUMBER;

The following example excerpt shows the use of the SEM_CONTAINS_COUNT ancillary operator to return the count of matching subgraphs for each matched document:

SELECT docId, SEM_CONTAINS_COUNT(1) as matching_subgraph_count
FROM   Newsfeed
WHERE  SEM_CONTAINS (article, 
  '{ ?org   rdf:type          class:Organization  . 
     ?org   pred:hasCategory  cat:BusinessFinance }', .., 
   1)= 1;

4.5 Searching for Documents Using SPARQL Query Patterns

Documents that are semantically indexed (that is, indexed using the mdsys.SemContext index type) can be searched using SEM_CONTAINS operator within a standard SQL query. In the query, the SEM_CONTAINS operator must have at least two parameters, the first specifying the column in which the documents are stored and the second specifying the document search criteria expressed as a SPARQL query pattern, as in the following example:

SELECT docId FROM Newsfeed
WHERE  SEM_CONTAINS (article, 
  '{ ?org  rdf:type  <http://www.example.com/classes/Organization>  . 
     ?org  <http://example.com/pred/hasCategory>  
             <http://www.example.com/category/BusinessFinance> }'
           )= 1;

The SPARQL query pattern specified with the SEM_CONTAINS operator is matched against the individual graphs corresponding to each document, and a document is considered to match a search criterion if the triples from the corresponding graph satisfy the query pattern. In the preceding example, the SPARQL query pattern identifies the individual graphs (thus, the documents) that refer to an Organization that belong to BusinessFinance category. The SQL query returns the rows corresponding to the matching documents in its result set. The preceding example assumes that the URIs used in the query are generated by the underlying extractor, and that you (the user searching for documents) are aware of the properties and terms that are generated by the extractor in use.

When you create an index using a dependent extractor policy that includes one or more user-defined RDF models, the triples asserted in the user models are considered to be common to all the documents. Document searches involving such policies test the search criteria against the triples in individual graphs corresponding to the documents, combined with the triples in the user models. For example, the following query identifies all articles referring to organizations in the state of New Hampshire, using the geographical ontology (geo_ontology RDF Model from a preceding example) that maps cities to states:

SELECT docId FROM   Newsfeed
WHERE  SEM_CONTAINS (article, 
        '{ ?org     rdf:type          class:Organization  . 
           ?org     pred:hasLocation  ?city . 
           ?city    geo:hasState      state:NewHampshire }', 
        'SEM_EXTR_PLUS_GEOONT', 
               sem_aliases(                              
                  sem_alias('class', 'http://www.myorg.com/classes/'),
                  sem_alias('pred', 'http://www.myorg.com/pred/'),
                  sem_alias('geo', 'http://geoont.org/rel/'),
                  sem_alias('state', 'http://geoont.org/state/'))) = 1;

The preceding query, with a reference to the extractor policy SEM_EXTR_PLUS_GEOONT (created in an example in Section 4.2), combines the triples extracted from the indexed documents and the triples in the user model to find matching documents. In this example, the name of the extractor policy is optional if the corresponding index is created with just this policy or if this is the default extractor policy for the index. When the query pattern uses some qualified names, an optional parameter to the SEM_CONTAINS operator can specify the namespaces to be used for expanding the qualified names.

SPARQL-based document searches can make use of the SPARQL syntax that is supported through SEM_MATCH queries.

4.6 Bindings for SPARQL Variables in Matching Subgraphs in a Document (SEM_CONTAINS_SELECT Ancillary Operator)

You can use the SEM_CONTAINS_SELECT ancillary operator to return additional information about each document matched using the SEM_CONTAINS operator. Specifically, the bindings for the variables used in SPARQL-based document search criteria can be returned using this operator. This operator is ancillary to the SEM_CONTAINS operator, and a literal number is used as an argument to this operator to associate it with a specific instance of SEM_CONTAINS operator, as in the following example:

SELECT docId, SEM_CONTAINS_SELECT(1) as result
FROM   Newsfeed
WHERE  SEM_CONTAINS (article, 
  '{ ?org   rdf:type          class:Organization  . 
     ?org   pred:hasCategory  cat:BusinessFinance }', .., 
   1)= 1;

The SEM_CONTAINS_SELECT ancillary operator returns the bindings for the variables in SPARQL Query Results XML format, as CLOB data. The variables may be bound to multiple data instances from a single document, in which case all bindings for the variables are returned. The following example is an excerpt from the output of the preceding query: a value returned by the SEM_CONTAINS_SELECT ancillary operator for a document matching the specified search criteria.

<results>
  <result> 
     <binding name="ORG">
        <uri>http://newscorp.com/Org/AcmeCorp</uri>
     </binding>
  </result> 
  <result>
     <binding name="ORG">
       <uri>http://newscorp.com/Org/ABCCorp</uri>
     </binding>
  </result>
</results>

You can rank the search results by creating an instance of XMLType for the CLOB value returned by the SEM_CONTAINS_SELECT ancillary operator and applying an XPath expression to sort the results on some attribute values.

By default, the SEM_CONTAINS_SELECT ancillary operator returns bindings for all variables used in the SPARQL-based document search criteria. However, when the values for only a subset of the variables are relevant for a search, the SPARQL pattern can include a SELECT clause with space-separated list of variables for which the values should be returned, as in the following example:

SELECT docId, SEM_CONTAINS_SELECT(1) as result
FROM   Newsfeed
WHERE  SEM_CONTAINS (article, 
        'SELECT ?org  ?city 
         WHERE { ?org     rdf:type          class:Organization  . 
                 ?org     pred:hasLocation  ?city . 
                 ?city    geo:hasState      state:NewHampshire }', .., 
         1) = 1;

4.7 Improving the Quality of Document Search Operations

The quality of a document search operation depends on the quality of the information produced by the extractor used to index the documents. If the information extracted is incomplete, you may want to add some annotations to a document. You can use the SEM_RDFCTX.MAINTAIN_TRIPLES procedure to add annotations, in the form of RDF triples, to specific documents in order to improve the quality of search, as shown in the following example:

begin
  sem_rdfctx.maintain_triples(
     index_name      => 'ArticleIndex',
     where_clause    => 'docid in (1,15,20)',  
     rdfxml_content => sys.xmltype(
      '<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" 
                xmlns:rdfs="http://www.w3.org/2000/01/rdf-schema#"
                xmlns:pred="http://example.com/pred/">
       <rdf:Description rdf:about=" http://newscorp.com/Org/ExampleCorp">
         <pred:hasShortName 
               rdf:datatype="http://www.w3.org/2001/XMLSchema#string">
             Example
         </pred:hasShortName>
     </rdf:Description> 
    </rdf:RDF>'));
end;
/

The index name and the WHERE clause specified in the preceding example identify specific instances of the document to be annotated, and the RDF/XML content passed in is used to add additional triples to the individual graphs corresponding to those documents. This allows domain experts and user communities to improve the quality of search by adding relevant triples to annotate some documents.

4.8 Indexing External Documents

You can use semantic indexing on documents that are stored in a file system or on the network. In such cases, you store the references to external documents in a table column, and you create a semantic index on the column using an appropriate extractor policy.

To index external documents, define an extractor policy with appropriate preferences, using an XML document that is assigned to the preferences parameter of the SEM_RDFCTX.CREATE_POLICY procedure, as in the following example:

begin
  sem_rdfctx.create_policy (
       policy_name => 'SEM_EXTR_FROM_FILE',
       extractor   => mdsys.gatenlp_extractor()),
       preferences => sys.xmltype('<RDFCTXPreferences>
                                     <Datastore type="FILE"> 
                                        <Path>EXTFILES_DIR</Path>
                                     </Datastore>
                                   </RDFCTXPreferences>')); 
end;
/

The <Datastore> element in the preferences document specifies the type of repository used for the documents to be indexed. When the value for the type attribute is set to FILE, the <Path> element identifies a directory object in the database (created using the SQL statement CREATE DIRECTORY). A table column indexed using the specified extractor policy is expected to contain relative paths to individual files within the directory object, as shown in the following example:

CREATE TABLE newsfeed (docid       number, 
                       articleLoc  VARCHAR2(100)); 
INSERT INTO into newsfeed (docid, articleLoc) values
                     (1, 'article1.txt'); 
INSERT INTO newsfeed (docid, articleLoc) values
                     (2, 'folder/article2.txt'); 
 
CREATE INDEX ArticleIndex on newsfeed (articleLoc)
   INDEXTYPE IS mdsys.SemContext PARAMETERS ('SEM_EXTR_FROM_FILE');

To index documents that are accessed using HTTP protocol, create a extractor policy with preferences that set the type attribute of the <Datastore> element to URL and that list one or more hosts in the <Path> elements, as shown in the following excerpt:

<RDFCTXPreferences>
   <Datastore type="URL"> 
       <Path>http://cnn.com</Path>
       <Path>http://abc.com</Path>
   </Datastore>
</RDFCTXPreferences>

The schema in which a semantic index for external documents is created must have the necessary privileges to access the external objects, including access to any proxy server used to access documents outside the firewall, as shown in the following example:

-- Grant read access to the directory object for FILE data store -- 
grant read on directory EXTFILES_DIR to SEMUSR;
 
-- Grant connect access to set of hosts for URL data store -- 
begin
  dbms_network_acl_admin.create_acl (
                acl          => 'network_docs.xml',
                description  => 'Normal Access',
                principal    => 'SEMUSR',
                is_grant     => TRUE,
                privilege    => 'connect');
end;
/
 
begin
  dbms_network_acl_admin.assign_acl (
               acl        => 'network_docs.xml',
               host       =>  'cnn.com',
               lower_port => 1,
               upper_port => 10000);
end;
/

External documents that are semantically indexed in the database may be in one of the well-known formats such as Microsoft Word, RTF, and PDF. This takes advantage of the Oracle Text capability to extract plain text version from formatted documents using filters (see the CTX_DOC.POLICY_FILTER procedure, described in Oracle Text Reference). To semantically index formatted documents, you must specify the name of a CTX policy in the extractor preferences, as shown in the following excerpt:

<RDFCTXPreferences>
   <Datastore type="FILE" filter="CTX_FILTER_POLICY"> 
       <Path>EXTFILES_DIR</Path>
   </Datastore>
</RDFCTXPreferences>

In the preceding example, the CTX_FILTER_POLICY policy, created using the CTX_DDL.CREATE_POLICY procedure, must exist in your schema. The table columns that are semantically indexed using this preferences document can store paths to formatted documents, from which plain text is extracted using the specified CTX policy. The information extractor associated with the extractor policy then processes the plain text further, to extract the semantics in RDF/XML format.

4.9 Configuring the Calais Extractor type

The CALAIS_EXTRACTOR type, which is a subtype of the RDFCTX_WS_EXTRACTOR type, enables you to access a Web service end point anywhere on the network, including the one that is publicly accessible (OpenCalais.com). To do so, you must connect with SYSDBA privileges and configure the Calais extractor type with Web service end point, the SOAP action, and the license key by setting corresponding parameters, as shown in the following example:

begin
  sem_rdfctx.set_extractor_param (
     param_key   => 'CALAIS_WS_ENDPOINT',
     param_value => 'http://api1.opencalais.com/enlighten/calais.asmx',
     param_desc  => 'Calais web service end-point');
       
  sem_rdfctx.set_extractor_param (
     param_key   => 'CALAIS_KEY',
     param_value => '<Calais license key goes here>',
     param_desc  => 'Calais extractor license key');
 
  sem_rdfctx.set_extractor_param (
     param_key   => 'CALAIS_WS_SOAPACTION',
     param_value => 'http://clearforest.com/Enlighten',
     param_desc  => 'Calais web service SOAP Action');
end;

To enable access to a Web service outside the firewall, you must also set the parameter for the proxy host, as in the following example:

begin
  sem_rdfctx.set_extractor_param (
      param_key   => 'HTTP_PROXY',
      param_value => 'www-proxy.acme.com',
      param_desc  => 'Proxy server');
end;

4.10 Working with General Architecture for Text Engineering (GATE)

General Architecture for Text Engineering (GATE) is an open source natural language processor and information extractor (see http://gate.ac.uk). You can use GATE to perform semantic indexing of documents stored in the database. The extractor type mdsys.gatenlp_extractor is defined as a subtype of the RDFCTX_EXTRACTOR type. The implementation of this extractor type sends an unstructured document to a GATE engine over a TCP connection, receives corresponding annotations, and converts them into RDF following a user-specified XML style sheet.

The requests for information extraction are handled by a server socket implementation, which instantiates the GATE components and listens to extraction requests at a pre-determined port. The host and the post for the GATE listener are recorded in the database, as shown in the following example, for all instances of the mdsys.gatenlp_extractor type to use.

begin 
  sem_rdfctx.set_extractor_param (
     param_key   => 'GATE_NLP_HOST',
     param_value => 'gateserver.acme.com',
     param_desc  => 'Host for GATE NLP Listener ');
       
  sem_rdfctx.set_extractor_param (
     param_key   => 'GATE_NLP_PORT',
     param_value => '7687',
     param_desc  => 'Port for Gate NLP Listener');
end;

The server socket application receives an unstructured document and constructs an annotation set with the desired types of annotations. Each annotation in the set may be customized to include additional features, such as the relevant phrase from the input document and some domain specific features. The resulting annotation set is serialized into XML (using the annotationSetToXml method in the gate.corpora.DocumentXmlUtils Java package) and returned back to the socket client.

A sample Java implementation for the GATE listener is available for download from the code samples and examples page on OTN (see Section 1.11, "Semantic Data Examples (PL/SQL and Java)" for information about this page).

The mdsys.gatenlp_extractor implementation in the database receives the annotation set encoded in XML, and converts it to RDF/XML using an XML style sheet. You can replace the default style sheet (listed in Section 4.17) used by the mdsys.gatenlp_extractor implementation with a custom style sheet when you instantiate the type.

The following example creates an extractor policy that uses a custom style sheet to generate RDF from the annotation set produced by the GATE extractor:

begin
  sem_rdfctx.create_policy (policy_name => 'GATE_EXTR',
                            extractor   => mdsys.gatenlp_extractor(
      sys.XMLType('<?xml version="1.0"?> 
                 <xsl:stylesheet version="2.0" 
                    xmlns:xsl="http://www.w3.org/1999/XSL/Transform" >
                   ..
                 </xsl:stylesheet>')));
end;
/

4.11 Creating a New Extractor Type

You can create a new extractor type by extending the RDFCTX_EXTRACTOR or RDFCTX_WS_EXTRACTOR extractor type. The extractor type to be extended must be accessible using Web service calls. The schema in which the new extractor type is created must be granted additional privileges to allow creation of the subtype. For example, if a new extractor type is created in the schema RDFCTXU, you must enter the following commands to grant the UNDER and RDFCTX_ADMIN privileges to that schema:

GRANT under ON mdsys.rdfctx_extractor TO rdfctxu;
GRANT rdfctx_admin TO rdfctxu;

As an example, assume that an information extractor can process an incoming document and return an XML document that contains extracted information. To enable the information extractor to be invoked using a PL/SQL wrapper, you can create the corresponding extractor type implementation, as in the following example:

create or replace type rdfctxu.info_extractor under rdfctx_extractor (
  xsl_trans   sys.XMLtype,
  constructor function info_extractor (
                 xsl_trans  sys.XMLType ) return self as result,
  overriding member function getDescription return VARCHAR2,
  overriding member function rdfReturnType return VARCHAR2,
  overriding member function extractRDF(document CLOB,
                                        docId    VARCHAR2) return CLOB
)
/
 
create or replace type body rdfctxu.info_extractor as 
  constructor function info_extractor (
                 xsl_trans  sys.XMLType ) return self as result is
  begin
    self.extr_type := 'Info Extractor Inc.'; 
    -- XML style sheet to generate RDF/XML from proprietary XML documents
    self.xsl_trans := xsl_trans; 
    return;
  end info_extractor; 
 
  overriding member function getDescription return VARCHAR2 is
  begin
    return 'Extactor by Info Extractor Inc.';
  end getDescription;
 
  overriding member function rdfReturnType return VARCHAR2 is
  begin
    return 'RDF/XML';
  end rdfReturnType;
 
  overriding member function extractRDF(document CLOB,
                                        docId    VARCHAR2) return CLOB is
    ce_xmlt  sys.xmltype;
  begin
    EXECUTE IMMEDIATE 
      'begin :1 = info_extract_xml(doc => :2); end;'
       USING IN OUT ce_xmlt, IN document;
 
    -- Now pass the ce_xmlt through RDF/XML transformation -- 
    return ce_xmlt.transform(self.xsl_trans).getClobVal();
  end extractRdf;
 
end;

In the preceding example:

  • The implementation for the created info_extractor extractor type relies on the XML style sheet, set in the constructor, to generate RDF/XML from the proprietary XML schema used by the underlying information extractor.

  • The extractRDF function assumes that the info_extract_xml function contacts the desired information extractor and returns an XML document with the information extracted from the document that was passed in.

  • The XML style sheet is applied on the XML document to generate equivalent RDF/XML, which is returned by the extractRDF function.

4.12 Creating a Local Semantic Index on a Range-Partitioned Table

A local index can be created on a VARCHAR2 or CLOB column of a range-partitioned table by using the following syntax:

CREATE INDEX <index-name> … LOCAL;

The following example creates a range-partitioned table and a local semantic index on that table:

CREATE TABLE part_newsfeed (
  docid number, article CLOB, cdate DATE) 
partition by range (cdate)
(partition p1 values less than (to_date('01-Jan-2001')),
 partition p2 values less than (to_date('01-Jan-2004')),
 partition p3 values less than (to_date('01-Jan-2008')),
 partition p4 values less than (to_date('01-Jan-2012'))
);
 
CREATE INDEX ArticleLocalIndex on part_newsfeed (article)
   INDEXTYPE IS mdsys.SemContext PARAMETERS ('SEM_EXTR')
LOCAL;

Note that every partition of the local semantic index will have content generated for the same set of policies. When you use the ALTER INDEX statement on a local index to add or drop policies associated with a semantic index partition, you should try to keep the same set of policies associated with each partition. You can achieve this result by using ALTER INDEX statements in a loop over the set of partitions. (For more information about altering semantic indexes, see Section 4.13,)

4.13 Altering a Semantic Index

This section discusses using the ALTER INDEX statement with a semantic index. For a local semantic index, the ALTER INDEX statement applies to a specified partition. The general syntax of the ALTER INDEX command for a semantic index is as follows:

ALTER INDEX <index-name> REBUILD [PARTITION <index-partition-name>]
  [PARAMETERS ('-<action_for_policy> <policy-name>')];

4.13.1 Rebuilding Content for All Existing Policies in a Semantic Index

If the PARAMETERS clause is not included in the ALTER INDEX statement, the content of the semantic index (or index partition) is rebuilt for every policy presently associated with the index. The following are two examples:

ALTER INDEX ArticleIndex REBUILD;
ALTER INDEX ArticleLocalIndex REBUILD PARTITION p1;

4.13.2 Rebuilding to Add Content for a New Policy to a Semantic Index

Using add_policy for <action_for_policy>, you can add content for a new base policy or a dependent policy to a semantic index (or index partition). If a dependent policy is being added and if its base policy is not already a part of the index, then content for the base policy is also added implicitly (by invoking the extractor specified as part of the base policy definition). The following is an example:

ALTER INDEX ArticleIndex REBUILD PARAMETERS ('-add_policy MY_POLICY');

4.13.3 Rebuilding Content for an Existing Policy from a Semantic Index

Using rebuild_policy for <action_for_policy>, you can rebuild the content of the semantic index (or index partition) for an existing policy presently associated with the index. The following is an example:

ALTER INDEX ArticleIndex REBUILD PARAMETERS ('-rebuild_policy MY_POLICY');

4.13.4 Rebuilding to Drop Content for an Existing Policy from a Semantic Index

Using drop_policy for <action_for_policy>, you can drop content corresponding to an existing base policy or a dependent policy from a semantic index (or index partition). Note that dropping the content for a base policy will fail if it is the only policy for the index (or index partition) or if it is used by dependent policies associated with this index (or index partition).

The following example drops the content for a policy from an index:

ALTER INDEX ArticleIndex REBUILD PARAMETERS ('-drop_policy MY_POLICY');

4.14 Passing Extractor-Specific Parameters in CREATE INDEX and ALTER INDEX

The CREATE INDEX and ALTER INDEX statements allow the passing of parameters needed by extractors. These parameters are passed on to the extractor using the params parameter of the extractRdf and batchExtractRdf methods. The following two examples show their use:

CREATE INDEX ArticleIndex on Newsfeed (article)
  INDEXTYPE IS mdsys.SemContext PARAMETERS ('SEM_EXTR=(NE_ONLY)');

ALTER INDEX ArticleIndex REBUILD 
  PARAMETERS ('-add_policy MY_POLICY=(NE_ONLY)');

4.15 Performing Document-Centric Inference

Document-centric inference refers to the ability to infer from each document individually. It does not allow triples extracted from two different documents to be used together for inference. It contrasts with the more common corpus-centric inference, where new triples can be inferred from combinations of triples extracted from multiple documents.

Document-centric inference can be desirable in document search applications because inclusion of a document in the search result is based on the extracted and/or inferred triples for that document only, that is, triples extracted and/or inferred from any other documents in the corpus do not play any role in the selection of this document. (Document-centric inference might be preferred, for example, if there is inconsistency among documents because of differences in the reliability of the data or in the biases of the document creators.)

To perform document-centric inference, use named graph based local inference (explained in Section 2.2.11.2) by specifying options => 'LOCAL_NG_INF=T' in the call to the SEM_APIS.CREATE_ENTAILMENT procedure.

Entailments created through document-centric inference can be included as content of a semantic index by creating a dependent policy and adding that policy to the semantic index, as shown in Example 4-2.

Example 4-2 Using Document-Centric Inference

-- Create entailment 'extr_data_inf' using document-centric inference
-- assuming:
--   model_name for semantic index based on base policy: 'RDFCTX_MOD_1'
--    (model name is available from the RDFCTX_INDEX_POLICIES view; 
--     see Section 4.16.2, "RDFCTX_INDEX_POLICIES View")
--   ontology: dataOntology
--   rulebase: OWL2RL
-- options: 'LOCAL_NG_INF=T' (for document-centric inference)
BEGIN
sem_apis.create_entailment('extr_data_inf',
  models_in    => sem_models('RDFCTX_MOD_1', 'dataOntology'),
  rulebases_in => sem_rulebases('OWL2RL'),
  options      => 'LOCAL_NG_INF=T');
END;
/
-- Create a dependent policy to augment data extracted using base policy
-- with content of entailment extr_data_inf (computed in previous statement)
BEGIN
sem_rdfctx.create_policy (
  policy_name => 'SEM_EXTR_PLUS_DATA_INF',
  base_policy => 'SEM_EXTR',
  user_models => NULL,
  user_entailments => sem_models('extr_data_inf'));
END;
/
-- Add the dependent policy to the ARTICLEINDEX index.
EXECUTE sem_rdfctx.add_dependent_policy('ARTICLEINDEX','SEM_EXTR_PLUS_DATA_INF');

4.16 Metadata Views for Semantic Indexing

This section describes views that contain metadata about semantic indexing.

4.16.1 MDSYS.RDFCTX_POLICIES View

Information about extractor policies defined in the current schema is maintained in the MDSYS.RDFCTX_POLICIES view, which has the columns shown in Table 4-1 and one row for each extractor policy.

Table 4-1 MDSYS.RDFCTX_POLICIES View Columns

Column NameData TypeDescription

POLICY_OWNER

VARCHAR2(32)

Owner of the extractor policy

POLICY_NAME

VARCHAR2(32)

Name of the extractor policy

EXTRACTOR

MDSYS.RDFCTX_EXTRACTOR

Instance of extractor type

IS_DEPENDENT

VARCHAR2(3)

Contains YES if the extractor policy is dependent on a base policy; contains NO if the extractor policy is not dependent on a base policy.

BASE_POLICY

VARCHAR2(32)

For a dependent policy, the name of the base policy

USER_MODELS

MDSYS.RDF_MODELS

For a dependent policy, a list of the RDF models included in the policy


4.16.2 RDFCTX_INDEX_POLICIES View

Information about semantic indexes defined in the current schema and the extractor policies used to create the index is maintained in the MDSYS.RDFCTX_POLICIES view, which has the columns shown in Table 4-2 and one row for each combination of semantic index and extractor policy.

Table 4-2 MDSYS.RDFCTX_INDEX_POLICIES View Columns

Column NameData TypeDescription

INDEX_OWNER

VARCHAR2(32)

Owner of the semantic index

INDEX_NAME

VARCHAR2(32)

Name of the semantic index

INDEX_PARTITION

VARCHAR2(32)

Name of the index partition (for LOCAL index only)

POLICY_NAME

VARCHAR2(32)

Name of the extractor policy

EXTR_PARAMETERS

VARCHAR2(100)

Parameters specified for the extractor

IS_DEFAULT

VARCHAR2(3)

Contains YES if POLICY_NAME is the default extractor policy for the index; contains NO if POLICY_NAME is not the default extractor policy for the index.

STATUS

VARCHAR2(10)

Contains VALID if the index is valid, INPROGRESS if the index is being created, or FAILED if a system failure occurred during the creation of the index.

RDF_MODEL

VARCHAR2(32)

Name of the RDF model maintaining the index data


4.16.3 RDFCTX_INDEX_EXCEPTIONS View

Information about exceptions encountered while creating or maintaining semantic indexes in the current schema is maintained in the MDSYS.RDFCTX_INDEX_EXCEPTIONS view, which has the columns shown in Table 4-3 and one row for each exception.

Table 4-3 MDSYS.RDFCTX_INDEX_EXCEPTIONS View Columns

Column NameData TypeDescription

INDEX_OWNER

VARCHAR2(32)

Owner of the semantic index associated with the exception

INDEX_NAME

VARCHAR2(32)

Name of the semantic index associated with the exception

POLICY_NAME

VARCHAR2(32)

Name of the extractor policy associated with the exception

DOC_IDENTIFIER

VARCHAR2(38)

Row identifier (rowid) of the document associated with the exception

EXCEPTION_TYPE

VARCHAR2(13)

Type of exception

EXCEPTION_CODE

NUMBER

Error code associated with the exception

EXCEPTION_TEXT

CLOB

Text associated with the exception

EXTRACTED_AT

TIMESTAMP

Time at which the exception occurred


4.17 Default Style Sheet for GATE Extractor Output

This section lists the default XML style sheet that the mdsys.gatenlp_extractor implementation uses to convert the annotation set (encoded in XML) into RDF/XML. (This extractor is explained in Section 4.10.)

<?xml version="1.0"?> 
  <xsl:stylesheet version="2.0" 
                   xmlns:xsl="http://www.w3.org/1999/XSL/Transform" > 
     <xsl:output encoding="utf-8" indent="yes"/> 
     <xsl:param name="docbase">http://xmlns.oracle.com/rdfctx/</xsl:param>
     <xsl:param name="docident">0</xsl:param>
     <xsl:param name="classpfx">
       <xsl:value-of select="$docbase"/>
       <xsl:text>class/</xsl:text> 
     </xsl:param>
     <xsl:template match="/">
        <rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" 
                 xmlns:rdfs="http://www.w3.org/2000/01/rdf-schema#"
                 xmlns:owl="http://www.w3.org/2002/07/owl#" 
                 xmlns:prop="http://xmlns.oracle.com/rdfctx/property/">  
        <xsl:for-each select="AnnotationSet/Annotation"> 
          <rdf:Description> 
            <xsl:attribute name="rdf:about"> 
              <xsl:value-of select="$docbase"/>
              <xsl:text>docref/</xsl:text>
              <xsl:value-of select="$docident"/>
              <xsl:text>/</xsl:text>
              <xsl:value-of select="@Id"/>
            </xsl:attribute>
            <xsl:for-each select="./Feature"> 
              <xsl:choose>
                <xsl:when test="./Name[text()='majorType']"> 
                  <rdf:type> 
                    <xsl:attribute name="rdf:resource"> 
                       <xsl:value-of select="$classpfx"/>
                       <xsl:text>major/</xsl:text>
                       <xsl:value-of select="translate(./Value/text(),
                                                       ' ', '#')"/>
                    </xsl:attribute>  
                  </rdf:type>
                </xsl:when>
                <xsl:when test="./Name[text()='minorType']"> 
                  <xsl:element name="prop:hasMinorType"> 
                    <xsl:attribute name="rdf:resource"> 
                       <xsl:value-of select="$docbase"/>
                       <xsl:text>minorType/</xsl:text>
                       <xsl:value-of select="translate(./Value/text(),
                                                       ' ', '#')"/>
                    </xsl:attribute>  
                  </xsl:element> 
                </xsl:when>
                <xsl:when test="./Name[text()='kind']"> 
                  <xsl:element name="prop:hasKind"> 
                    <xsl:attribute name="rdf:resource"> 
                       <xsl:value-of select="$docbase"/>
                       <xsl:text>kind/</xsl:text>
                       <xsl:value-of select="translate(./Value/text(),
                                                       ' ', '#')"/>
                    </xsl:attribute>  
                  </xsl:element> 
                </xsl:when>
                <xsl:when test="./Name[text()='locType']"> 
                  <xsl:element name="prop:hasLocType"> 
                    <xsl:attribute name="rdf:resource"> 
                       <xsl:value-of select="$docbase"/>
                       <xsl:text>locType/</xsl:text>
                       <xsl:value-of select="translate(./Value/text(),
                                                       ' ', '#')"/>
                    </xsl:attribute>  
                  </xsl:element> 
                </xsl:when>
                <xsl:when test="./Name[text()='entityValue']"> 
                  <xsl:element name="prop:hasEntityValue"> 
                    <xsl:attribute name="rdf:datatype"> 
                      <xsl:text>
                         http://www.w3.org/2001/XMLSchema#string
                      </xsl:text>
                    </xsl:attribute> 
                    <xsl:value-of select="./Value/text()"/>
                  </xsl:element> 
                </xsl:when>
                <xsl:otherwise> 
                  <xsl:element name="prop:has{translate(
                                        substring(./Name/text(),1,1),
                                        'abcdefghijklmnopqrstuvwxyz',
                                        'ABCDEFGHIJKLMNOPQRSTUVWXYZ')}{
                                      substring(./Name/text(),2)}"> 
                     <xsl:attribute name="rdf:datatype"> 
                        <xsl:text>
                          http://www.w3.org/2001/XMLSchema#string
                        </xsl:text> 
                     </xsl:attribute> 
                    <xsl:value-of select="./Value/text()"/>
                  </xsl:element> 
                </xsl:otherwise> 
              </xsl:choose>
            </xsl:for-each> 
          </rdf:Description> 
        </xsl:for-each>
        </rdf:RDF> 
      </xsl:template>
   </xsl:stylesheet>
PKa!!PK.AOEBPS/sdo_rdf_newfeat.htmXa What's New in Semantic Technologies?

What's New in Semantic Technologies?

This section describes new and changed semantic technologies features for Oracle Database Release 11.

Release 11.2 Features

The following are new and changed features for Oracle Database 11g Release 2 (11.2).

This release also includes the features that were supplied in the interim patch 7600122 for Release 11.1.0.7.0, which are listed in "Features Added for Release 11.1.0.7 (November, 2008)".


Release 11.2.0.2: Required Actions if Semantic Technologies Installation is Invalid:

Further action may be required if your Semantic Technologies installation is invalid after upgrading to Release 11.2.0.2.0. For information, see Section A.1.4.

Jena Adapter for Oracle Database (Updated November, 2011)

The following features are included in an updated Jena Adapter kit that was made available in November 2011:

The Jena Adapter for Oracle Database provides a Java-based interface to Oracle Semantic Technologies by implementing the well-known Jena Graph and Model APIs. For information about downloading and using the Jena Adapter, see Chapter 7.


Note:

Updates of the Jena Adapter have been made available since its initial release. For information about the new features in an update, see the "readme" file included in the download .zip file.

Named Graph Support Enhancements (Added in Patch 9825019 in May 2011; Included in Release 11.2.0.3)

With Patch 9825019 (SEMANTIC TECHNOLOGIES 11G R2 FIX BUNDLE 3), features have been added to support the following operations with named graph data:

  • Loading

  • Querying

  • Inferencing

Section 1.3.9 provides an overview of named graph support. That section also includes links to sections about performing specific operations with named graph data.

Spatial Support (Added in Patch 9825019 in May 2011; Included in Release 11.2.0.3)

With Patch 9825019 (SEMANTIC TECHNOLOGIES 11G R2 FIX BUNDLE 3), Oracle Database Semantic Technologies supports storage of spatial geometry data encoded as orageo:WKTLiteral typed literals, and it provides a set of query functions for spatial operations.

For information about representing, indexing, and querying spatial data in RDF, see Section 1.6.6.

Data Type Indexing (Added in Patch 9825019 in May 2011; Included in Release 11.2.0.3)

With Patch 9825019 (SEMANTIC TECHNOLOGIES 11G R2 FIX BUNDLE 3), data type indexes (indexes on the values of typed literals stored in a semantic network) may significantly improve the performance of SEM_MATCH queries involving certain types of FILTER expressions.

Section 1.9 explains how to use data type indexes, and it provides links to reference information about new procedures to create, alter, and drop a data type index.

Jena Adapter SPARQL Gateway Support (Updated September, 2011)

SPARQL Gateway (described in Section 7.16, "SPARQL Gateway and Semantic Data") is added in the September 2011 update to the Jena Adapter. (If you are using a previous version of the Jena Adapter and if you do not want to use SPARQL Gateway, you do not need to download and install the September 2011 update.)

Enhanced Optimizer Hint Support in SEM_MATCH (11.2.0.2)

Effective with Release 11.2.0.2, you can embed inline HINT0 query optimizer hints using SPARQL comments in SEM_MATCH, which allows optimizer hints to be associated with non-root BGPs in a SPARQL query. In addition, new hints have been introduced to influence joins between BGPs. For information, see Section 1.6.4, "Inline Query Optimizer Hints".

Full-Text Indexing and Searching for RDF Terms (11.2.0.2)

Effective with Release 11.2.0.2, you can create an Oracle Text index on the MDSYS.RDF_VALUE$ table and use the orardf:textContains SPARQL FILTER function to execute efficient full-text searches in semantic queries. For information, see Section 1.6.5, "Full-Text Search".

OLS Triple-Level Security (Added in Patch 9819833 in June 2010)

Oracle Label Security (OLS) for RDF data provides the triple-level security option, in addition to the resource-level security previously available. Triple-level provides superior performance and ease of use, and is described in Section 5.2.1, "Triple-Level Security".

OWL 2 RL Support (Added in Patch 9819833 in June 2010)

The OWL 2 RL profile is supported by the addition of the system-defined rulebase OWL2RL, as explained in Section 2.1.2, "Supported OWL Subsets".

Features Added in Patch to Release 11.2 (Released March 2010)

The following features are included in a Release 11.2 Semantic Technologies patch that was made available on the Oracle Technology Network (OTN) in March 2010. You must apply that patch in order to be able to use these features.

In addition, Section A.1.2, "Upgrading Semantic Technologies Support from Release 11.1" has been expanded to include Section A.1.2.1, "Handling of Empty RDF Literals".

Sesame Adapter for Oracle Database

The Sesame Adapter for Oracle Database integrates the popular Sesame Java APIs with Oracle Semantic Technologies support. For information about downloading and using the Sesame Adapter, see Chapter 8.


Note:

Support for the Sesame Adapter was provided in a patch that was made available in January, 2010, on the Oracle Technology Network (OTN), and is updated in a patch made available in March or April, 2010.

Semantic Indexing for Documents

You can use semantic indexing to enable queries on information extracted from unstructured documents. Documents indexed semantically can be searched using the SEM_CONTAINS operator within a standard SQL query. For information about using semantic indexing, see Chapter 4.

The SEM_RDFCTX PL/SQL package contains subprograms to manage extractor policies and semantic indexes created for documents. This package is documented in Chapter 12.

Virtual Private Database (VPD) and Oracle Label Security (OLS) Support

You can enforce a fine-grained access control mechanism for RDF data by using either the Virtual Private Database (VPD) or Oracle Label Security (OLS) feature of Oracle Database, as explained in Chapter 5.

Workspace Manager Support for RDF Data

You can use Oracle Workspace Manager to version-enable RDF data in the semantic data store, as explained in Chapter 6.

New Procedures to Enable, Downgrade, and Remove Semantic Technologies Support

The procedures for enabling, downgrading, and removing semantic technologies support in the database have been redesigned. These procedures are explained in Appendix A.

Note that you still must perform certain actions and meet prerequisites before you can use any types, synonyms, or PL/SQL packages related to Oracle semantic technologies support. These actions and prerequisites are explained in Section A.1.

Optimized owl:sameAs Inference

You can optimize the performance of owl:sameAs inference, as explained in Section 2.2.8.

Optimizing Entailment Performance Through Compact Structures

You can specify RAW8=T in the options parameter to the SEM_APIS.CREATE_ENTAILMENT procedure, to cause RAW8 datatypes to be used instead of NUMBER in many intermediate tables created during the inference process. This option can improve entailment performance by up to 30% in some cases.

Incremental Inference

You can use incremental inference update entailments (rules indexes) efficiently after triple additions, as explained in Section 2.2.9.

Parallel Inference

You can use parallel inference to improve inference performance by taking advantage of the capabilities of a multi-core or multi-CPU architectures, as explained in Section 2.2.10.

Filter and Union Support in Curly Brace Syntax for SEM_MATCH

You can now specify FILTER or UNION, or both, in addition to OPTIONAL as keywords in the curly brace syntax supported for the SEM_MATCH table function. These keywords are explained in Section 1.6.2.

Simple Knowledge Organization System (SKOS) Support

You can perform inferencing based on a core subset of the Simple Knowledge Organization System (SKOS) data model, as explained in Chapter 3.

Inference Beyond OWLPrime

You can specify additional inference components to cover OWL constructs, such as owl:intersectionOf and owl:unionOf, as explained in Table 9-1, "Inferencing Keywords for inf_components_in Parameter" in the Usage Notes for the SEM_APIS.CREATE_ENTAILMENT procedure in Chapter 9.

Systematized Nomenclature of Medicine - Clinical Terms (SNOMED CT) Support

The current release includes a built-in rulebase that supports the expressiveness of Systematized Nomenclature of Medicine - Clinical Terms (SNOMED CT) terminology. To include this support, specify the SNOMED keyword in the inf_components_in parameter in the call to the SEM_APIS.CREATE_ENTAILMENT procedure .

New SEM_APIS Package Subprograms

The following subprograms have been added to the SEM_APIS PL/SQL package, which is documented in Chapter 9.

Parameters for SEM_PERF.GATHER_STATISTICS

The SEM_PERF.GATHER_STATS procedure now accepts two parameters, both optional: just_on_values_table and parallel. This procedure is documented in Chapter 11.

SEM_APIS.*_RULES_INDEX Subprograms Deprecated

In the SEM_APIS PL/SQL package, the subprograms containing "_RULES_INDEX" in their names are deprecated and removed from this manual. Instead, you should use subprograms containing "_ENTAILMENT" in their names:

The old "*_RULES_INDEX" formats will continue to work, but you are encouraged to switch to using the "*_ENTAILMENT" subprograms.

Note that several metadata views still include "RULES_INDEX" in their names.

The terms entailment and rules index are synonyms in this manual, although entailment is used predominantly

Release 11.1 Features

The following are new and changed features for Oracle Database 11g Release 1 (11.1).

Features Added for Release 11.1.0.7 (November, 2008)

This section describes features that are included in interim patch 7600122 for Release 11.1.0.7.0, which was made available on My Oracle Support in November, 2008.

  • Support for virtual models (see Section 1.3.8)

  • Curly brace syntax for SEM_MATCH graph pattern, including support for the OPTIONAL construct (see Section 1.6.2)

  • Using HINT0 ("hint-zero" ) in a SEM_MATCH query (see Section 1.6)

  • New columns returned from SEM_MATCH: id, _prefix, _suffix (see Section 1.6)

  • Ability to create and manage semantic network indexes on models and rules indexes (see Section 1.8)

  • SEM_APIS.BULK_LOAD_FROM_STAGING_TABLE procedure: New options in the flags parameter join_hint for tasks IZC, MBV, and MBT; parallel=n)

  • Simplification of staging table definition (fewer columns) and privilege requirements (see Section 1.7.1)

  • Simpler event tracing during bulk load (see the information about the new RDF$ET_TAB table in Section 1.7.1.2)

Storage Model Enhancements and Migration

The storage model has been enhanced to support OWL inferencing: some internal data structures and indexes have been changed, added, and removed. These changes also result in enhanced performance.

Because of the extent of these changes, if you have semantic data that you used with the previous release, you must upgrade that data to migrate it to the new format before you can use any new features for this release. For more information, see Appendix A, "Enabling, Downgrading, or Removing Semantic Technologies Support".

Support for OWL Inferencing

Support has been added to support storing, validating, and querying Web Ontology Language (OWL)-based ontologies. Support is provided for a subset of the OWL DL language.

To query ontology data, you can use table functions and operators that examine semantic relationships, such as SEM_MATCH, SEM_RELATED, and SEM_DISTANCE.

New Bulk Loading Interface for Improved Performance

You can improve performance for bulk loading of semantic data in bulk using a staging table and calling the SEM_APIS.BULK_LOAD_FROM_STAGING_TABLE procedure. For more information, see Section 1.7.1.

Ontology-Assisted Querying of Relational Data

You can go beyond syntactic matching to perform semantic relatedness-based querying of relational data, by associating an ontology with the data and using the new SEM_RELATED operator (and optionally its SEM_DISTANCE ancillary operator). The new SEM_INDEXTYPE index type improves performance for semantic queries.

Required Procedure for Using Semantic Technology Support

Before you can use any types, synonyms, or PL/SQL packages related to Oracle semantic technologies support, you must enable support for semantic technologies. For more information, see Appendix A, "Enabling, Downgrading, or Removing Semantic Technologies Support".

PK7, ]aXaPK.A OEBPS/toc.htm Table of Contents

Contents

List of Examples

List of Figures

List of Tables

Title and Copyright Information

Preface

What's New in Semantic Technologies?

Part I Conceptual and Usage Information

1 Oracle Database Semantic Technologies Overview

2 OWL Concepts

3 Simple Knowledge Organization System (SKOS) Support

4 Semantic Indexing for Documents

5 Fine-Grained Access Control for RDF Data

6 Workspace Manager Support for RDF Data

7 Jena Adapter for Oracle Database

8 Sesame Adapter for Oracle Database

Part II Reference and Supplementary Information

9 SEM_APIS Package Subprograms

10 SEM_OLS Package Subprograms

11 SEM_PERF Package Subprograms

12 SEM_RDFCTX Package Subprograms

13 SEM_RDFSA Package Subprograms

A Enabling, Downgrading, or Removing Semantic Technologies Support

B SEM_MATCH Support for Spatial Queries

Glossary

Index

PK)CPK.AOEBPS/sem_ols_ref.htm- SEM_OLS Package Subprograms

10 SEM_OLS Package Subprograms

The SEM_OLS package contains subprograms (functions and procedures) related to triple-level security to RDF data, using Oracle Label Security (OLS). To use the subprograms in this chapter, you should understand the conceptual and usage information in Chapter 1, "Oracle Database Semantic Technologies Overview" and Chapter 5, "Fine-Grained Access Control for RDF Data".

This chapter provides reference information about the subprograms, listed in alphabetical order.


SEM_OLS.APPLY_POLICY_TO_APP_TAB

Format

SEM_OLS.APPLY_POLICY_TO_APP_TAB(

     policy_name IN VARCHAR2,

     schema_name IN VARCHAR2,

     table_name IN VARCHAR2,

     predicate IN VARCHAR2 DEFAULT NULL);

Description

Applies an OLS policy to an application table.

Parameters

policy_name

Name of an existing OLS policy.

schema_name

Name of the schema containing the application table.

table_name

Name of the application table.

predicate

An additional predicate to combine with the label-based predicate.

Usage Notes

When you use triple-level security, OLS is applied to each semantic model in the network. That is, label security is applied to the relevant internal tables and to all the application tables; there is no need to manually apply policies to the application tables of existing semantic models. However, if you need to create additional models after applying the OLS policy, you must use the SEM_OLS.APPLY_POLICY_TO_APP_TAB procedure to apply OLS to the application table before creating the model.

You must have the following to execute this procedure: EXECUTE privilege for the SA_POLICY_ADMIN package, and the policy_DBA role.

Before executing this procedure, you must have executed the SEM_RDFSA.APPLY_OLS_POLICY procedure specifying SEM_RDFSA.TRIPLE_LEVEL_ONLY for the rdfsa_options parameter.

To remove the OLS policy from the application table, use the SEM_OLS.REMOVE_POLICY_FROM_APP_TAB procedure.

For information about support for OLS, see Section 5.2.

Examples

The following example applies an OLS policy named defense to the MY_SCHEMA.MY_APP_TABLE application table.

begin
  sem_ols.apply_policy_to_app_table(
        policy_name => 'defense',
        schema_name => 'my_schema',
        table_name  => 'my_app_table');
end;
/

SEM_OLS.REMOVE_POLICY_FROM_APP_TAB

Format

SEM_OLS.REMOVE_POLICY_FROM_APP_TAB(

     policy_name IN VARCHAR2,

     schema_name IN VARCHAR2,

     table_name IN VARCHAR2);

Description

Permanently removes or detaches the OLS policy from an application table.

Parameters

policy_name

Name of the existing OLS policy.

schema_name

Name of the schema containing the application table.

table_name

Name of the application table.

Usage Notes

If you have dropped a semantic model and you no longer need to protect the application table, you can use this procedure.

You must have the following to execute this procedure: EXECUTE privilege for the SA_POLICY_ADMIN package, and the policy_DBA role.

Before executing this procedure, you must have executed the SEM_RDFSA.APPLY_OLS_POLICY procedure specifying SEM_RDFSA.TRIPLE_LEVEL_ONLY for the rdfsa_options parameter.

An exception is generated if the associated model exists. In this case, if you want to execute this procedure, you must first drop the model.

For information about support for OLS, see Section 5.2.

Examples

The following example removes the OLS policy named defense from the MY_SCHEMA.MY_APP_TABLE application table.

begin
  sem_ols.remove_policy_from_app_table(
        policy_name => 'defense',
        schema_name => 'my_schema',
        table_name  => 'my_app_table');
end;
/
PKo_2-PK.AOEBPS/workspace_mgr.htmr5 Workspace Manager Support for RDF Data

6 Workspace Manager Support for RDF Data

RDF data stored in the Oracle Database semantic data store can be version-enabled using Oracle Database 11g Workspace Manager. Using Workspace Manager, you can create new versions of RDF data through data modification operations while maintaining a copy of the old data. For detailed usage and reference information about Workspace Manager, see Oracle Database Workspace Manager Developer's Guide.

The unit of versioning for the semantic data store is a model, which is in turn associated with an application table that resides in a user schema. Traditional Workspace Manager interfaces are used to manage a version-enabled RDF model. (However, you cannot use version-enabling on a model that participates in a virtual model. Virtual models are described in Section 1.3.8)

Creating the first version-enabled RDF model in a database re-creates an index on the underlying triple store that holds data for all RDF models. Thus, the time needed to create the first version-enabled RDF model depends on the amount of existing data in these models.

This chapter discuss the key characteristics of a version-enabled RDF model and provides some examples. It contains the following major sections:

6.1 Enabling Workspace Manager Support for RDF Data

Workspace Manager support for RDF data is not installed in the database by default. Instead, you must run a script named sdordfwm.sql after you enable semantic technologies support in the database (explained in Section A.1, "Enabling Semantic Technologies Support"). You only need to run sdordfwm.sql once for the database. To run this script, connect to the database as SYSDBA and enter the following statement as appropriate for your operating system:

  • Linux: SQL> @$ORACLE_HOME/md/admin/sdordfwm.sql

  • Windows: SQL> @%ORACLE_HOME%\md\admin\sdordfwm.sql

If you want to check whether Workspace Manager support is enabled for RDF data, enter the following query and see if the value returned is INSTALLED:

SELECT value FROM mdsys.rdf_parameter 
  WHERE namespace = 'COMPONENT' and attribute = 'RDFOWM';

6.1.1 Removing Workspace Manager Support for RDF Data

After you have enabled Workspace Manager support for RDF data, you might need to remove that support in certain circumstances. For example, before you downgrade or remove semantic technologies support in the database, you must remove the Workspace Manager support for RDF data. (For information about downgrading and removing semantic technologies support, see Appendix A, "Enabling, Downgrading, or Removing Semantic Technologies Support".)

To remove Workspace Manager support for RDF data, perform the following steps:

  1. Connect to the database as the SYS user with SYSDBA privileges (SYS AS SYSDBA, and enter the SYS account password when prompted).

  2. Start SQL*Plus, and enter the following statement:

    • Linux: @$ORACLE_HOME/md/admin/sdordfwm_rm.sql

    • Windows: @%ORACLE_HOME%\md\admin\sdordfwm_rm.sql

After the sdordfwm_rm.sql script completes successfully, you can downgrade or remove semantic technologies support in the database.

6.2 Version-Enabling an RDF Model

To version-enable an RDF model, you must create any necessary entailments (rules indexes) on it (using the SEM_APIS.CREATE_ENTAILMENT procedure), and then version-enable its associated application table. For example, if an RDF model named contracts is associated with an application table named CONTRACTS_RDF_DATA, you can perform the version enabling as shown in the following example:

begin
  dbms_wm.enableVersioning (table_name => 'contracts_rdf_data'); 
end;
/

The application table must have a primary key, and you cannot specify the HISTORY or VALID TIME options.

The data stored in the application table and the corresponding RDF data in the semantic data store are versioned logically. The standard data manipulation operations (insert, update, delete) on a version-enabled application table maintain the version information for both the affected application table rows and the corresponding RDF triples in the semantic data store. Queries accessing the RDF data using the SEM_MATCH operator take the state of the active workspace and the relevant versions of data into account when constructing the result set.

You cannot use continually refreshed workspaces and multiparent workspaces when working with RDF data. However, other workspace operations (such as creating new workspaces, creating savepoints, setting access modes, and refreshing and merging workspaces) proceed as you would expect.

You can use Workspace Manager locks on the application table rows to avoid conflicting data manipulation operations in concurrent workspaces. The conflicts in such cases are detected based on the primary keys for the locked rows, and they may not directly translate to any meaningful locks on the RDF data, such as a lock on a sub-graph or a node. (See Section 6.4 for best practices involving version-enabled RDF data.)

6.3 Inferring from Version-Enabled RDF Models

You can use an RDF model to create one or more entailments through inference using system-defined and user-defined rulebases. When such a model is version-enabled, the associated entailments that store the inferred data are also version-enabled.

However, unlike the RDF models, which are versioned logically, the entailments are versioned physically. Thus, a workspace with some unmerged changes to an RDF model may maintain a private version of the entailment, which duplicates all the inferred triples that may already exist in the entailment corresponding to the parent workspace. Figure 6-1 shows a physically versioned entailment where private copies of the index are maintained for workspaces W1 and W2, which are child workspaces of the LIVE workspace.

Figure 6-1 Physical Versioning of Entailment (Rules Index)

Description of Figure 6-1 follows
Description of "Figure 6-1 Physical Versioning of Entailment (Rules Index)"

A workspace created from a parent workspace with a valid entailment will share the entailment with its parent until some workspace-specific data manipulation operations occur on the corresponding models. The first data manipulation operation within a workspace creates a private copy of the entailment and marks it INVALID. Workspace-private versions of inference data can be rebuilt using the SEM_APIS.CREATE_ENTAILMENT procedure, as shown in the following example:

begin
  sem_apis.create_entailment(
            index_name_in   => 'contracts_rdfs',
            models_in       => SDO_RDF_Models('contracts'),
            rulebases_in    => SDO_RDF_Rulebases('RDFS')); 
end;
/

Any subsequent additions to the RDF model within the workspace will mark the entailment as INCOMPLETE, although you can return it to a VALID status using the SEM_APIS.CREATE_ENTAILMENT procedure. When a workspace is merged, the workspace-private entailment is dropped, and the corresponding index in the parent workspace is marked INVALID.

6.4 Merging and Refreshing Workspaces in Version-Enabled RDF Models

A workspace modifying data stored in a version-enabled RDF model may contain additional changes to other RDF models or relational tables. The semantics for the merging or refreshing relational data modified in a workspace remain unchanged. Similarly, the merging or refreshing of the application table is subject to conflict analysis based on the primary key defined in the application table. Triples added and deleted in concurrent workspaces are not flagged as conflicts unless they are associated with the same primary key in the application table. For example, a parent and a child workspace may assert the same triple with different primary keys, and during a workspace merge both the triples are accepted. You can avoid such conflicts by generating the application table primary keys from the triple components.

When you work with RDF data, the types of conflicts that must be flagged vary depending on the application requirements. The conflicts include physical conflicts that exist between two triples, and logical conflicts that exist in a graph to which the triples belong. For example, two concurrent workspaces asserting a triple that captures the value of a specific contract physically conflict with each other. Similarly, a workspace that assigned a new project to a department (with an appropriate triple) may logically conflict with other workspace that reduced the budget for the department.

You can avoid these conflicts by acquiring appropriate Workspace Manager locks on the application table rows using the DBMS_WM.lockRows subprogram. For example, while adding a new triple that describes a specific resource, the application logic may acquire row locks for all application table rows that describe the same resource; in this case, failure to acquire the lock indicates a conflict.

If business conflicts between concurrent workspaces can be manually detected and reconciled, you can use the <application table name>_DIFF Workspace Manager view to obtain a summary of all the changes in two workspaces. For a version-enabled application table named contracts_rdf_data, the query in following example fetches the triples for all the rows that are modified either in W2 workspace or the LIVE workspace:

begin
  dbms_wm.SetDiffVersions(workspace1 => 'W2',
                          workspace2 => 'LIVE');
end;
/
 
SELECT appt.apptable_pkey, appt.triple.get_triple() triple, 
       wm_diffver, wm_code FROM contracts_rdf_data_diff appt;

For an explanation of the possible values for the WM_DIFFVER and WM_CODE columns, see the section about xxx_DIFF views in Oracle Database Workspace Manager Developer's Guide.

For any conflicts detected using this technique, you can resolve them by executing compensating data manipulation operations for the conflicting data modifications.

6.5 Special Considerations When Using Workspace Manager Support for RDF Data

Certain Semantic Technologies features are not compatible with Workspace Manager for RDF data.

The following network-level features are not supported on semantic networks containing version-enabled RDF models:

  • Oracle Label Security (OLS) for RDF data

  • Virtual Private Database (VPD) for RDF data

  • Semantic indexing for documents

  • SEM_APIS.DROP_SEM_NETWORK with the cascade option

The following model-level features are not supported on version-enabled RDF models:

6.6 Usage Flow Example: Versioning Semantic Models

Example 6-1 shows how entailments can be handled in different workspaces. Comments within the example explain the individual operations.

Note the following restrictions and considerations for version-enabled RDF models, which are reflected in Example 6-1:

  • Incremental inference is not supported for entailments involving version-enabled RDF models.

  • Although an already created entailment can be made VALID by calling SEM_APIS.CREATE_ENTAILMENT, a new entailment cannot be created once the application table corresponding to an RDF model has been version-enabled.

  • The pattern argument for SEM_MATCH needs to use the old (non-curly-brace) syntax.

  • The following option must be used in the SEM_APIS.CREATE_ENTAILMENT invocation involving version-enabled models: 'EPT=1' (or options => 'USER_RULES=T,EPT=1' when user-defined rules are used) .

Example 6-1 Versioning a Semantic Model

set pagesize 10000
 
-- 1. Enabling Workspace Manager Support for RDF Data
conn / as sysdba
@?/md/admin/sdordfwm.sql
 
set echo on
column x format a10
column y format a10
 
column owner format a10
column index_name format a12
column index_view_name format a18
column status format a10
 
-- create sem network and a test user
exec sem_apis.create_sem_network('SYSAUX');
grant connect,resource,unlimited tablespace to rdfuser identified by rdfuser;
 
-- 2. Create table and model: then Insert triples (that will lead to some inferred triples, when entailed)
 
conn rdfuser/rdfuser
 
CREATE TABLE emp_rdf_data(id number,triple sdo_rdf_triple_s);
ALTER TABLE emp_rdf_data add CONSTRAINT emp_rdf_data_PK PRIMARY KEY (ID);
 
EXEC SEM_APIS.CREATE_SEM_MODEL('emp0','emp_rdf_data','triple');
 
insert into emp_rdf_data values (0.1, sdo_rdf_triple_s('emp0','<Man>','rdfs:subClassOf','<Person>'));
insert into emp_rdf_data values (0.2, sdo_rdf_triple_s('emp0','<Woman>','rdfs:subClassOf','<Person>'));
insert into emp_rdf_data values (1, sdo_rdf_triple_s('emp0','<John>','rdf:type','<Man>'));
 
-- 3. Create entailments: one with user-def rules only and the other with OWL
 
exec sdo_rdf_inference.create_rulebase('family_rb');
 
insert into mdsys.rdfr_family_rb values (
'R1',
'(?x rdf:type <Man>)',
NULL,
'(?x rdf:type <Male>)',
NULL);
 
insert into mdsys.rdfr_family_rb values (
'R2',
'(?x rdf:type <Woman>)',
NULL,
'(?x rdf:type <Female>)',
NULL);
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_u_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('family_rb'),
options => 'USER_RULES=T,EPT=1');
end;
/
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('owlprime'),
options => 'EPT=1');
end;
/
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 4. Activate versioning and re-try the queries
 
EXEC dbms_wm.enableVersioning (table_name => 'emp_rdf_data');
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
 
-- 4a. [fail] try to create a post-versioning entailment for testing
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_idx_2',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('owl2rl'),
options => 'EPT=1');
end;
/
 
 
-- 5. Create Workspace and move into that workspace
 
exec dbms_wm.createworkspace('W1');
exec dbms_wm.gotoWorkspace('W1');
 
-- 5a. [fail] in this workspace, try to create a post-versioning entailment for testing
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_idx_3',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('rdfs'),
options => 'EPT=1');
end;
/
 
-- 6. Workspace inherits the entailment from LIVE workspace (no physical copy yet)
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 7. Insert some triples => causes LIVE entailments to be copied to this workspace and marked INVALID here
insert into emp_rdf_data values (2.1, sdo_rdf_triple_s('emp0','<Mary>','rdf:type','<Woman>'));
commit;
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 8. Go back to LIVE workspace to confirm that the LIVE entailments are still VALID
 
disconnect
connect rdfuser/rdfuser
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 9. Back to new workspace: make its entailments VALID
 
exec dbms_wm.gotoWorkspace('W1');
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_u_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('family_rb'),
options => 'USER_RULES=T,EPT=1');
end;
/
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('owlprime'),
options => 'EPT=1');
end;
/
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 10. Go back to LIVE workspace to confirm that the entailments there are still VALID
 
disconnect
connect rdfuser/rdfuser
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 11. Insert a new row into LIVE model => causes LIVE entailments to become INVALID
 
insert into emp_rdf_data values (3, sdo_rdf_triple_s('emp0','<Gary>','rdf:type','<Man>'));
commit;
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 12. Back to new workspace: confirm that its entailment is still VALID
 
exec dbms_wm.gotoWorkspace('W1');
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 13. Back to LIVE workspace: make the entailments VALID again
 
disconnect
connect rdfuser/rdfuser
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_u_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('family_rb'),
options => 'USER_RULES=T,EPT=1');
end;
/
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('owlprime'),
options => 'EPT=1');
end;
/
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 14. Insert a new row into LIVE model => causes LIVE entailments to become INCOMPLETE
 
insert into emp_rdf_data values (4, sdo_rdf_triple_s('emp0','<Lory>','rdf:type','<Man>'));
commit;
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 15. make the entailments in LIVE workspace VALID again
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_u_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('family_rb'),
options => 'USER_RULES=T,EPT=1');
end;
/
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('owlprime'),
options => 'EPT=1');
end;
/
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 16. merge workspace: this should make the LIVE entailments INVALID
conn rdfuser/rdfuser
exec dbms_wm.mergeWorkspace('W1');
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- 17. remove workspace
exec dbms_wm.removeWorkspace('W1');
 
-- 18. disable versioning, refresh the INVALID entailments to make them VALID, and then query
exec dbms_wm.disableVersioning('emp_rdf_data');
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_u_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('family_rb'),
options => 'USER_RULES=T,EPT=1');
end;
/
 
begin
sem_apis.create_entailment(
index_name_in => 'emp0_idx',
models_in => SDO_RDF_Models('emp0'),
rulebases_in => SDO_RDF_Rulebases('owlprime'),
options => 'EPT=1');
end;
/
 
select * from mdsys.rdf_rules_index_info;
select count(*) from mdsys.rdfi_emp0_u_idx;
select x,y from TABLE(SEM_MATCH('(?x rdf:type ?y)',sem_models('emp0'),sem_rulebases('family_rb'),null,null)) order by 1,2;
select count(*) from mdsys.rdfi_emp0_idx;
select x from TABLE(SEM_MATCH('(?x rdf:type <Person>)',sem_models('emp0'),sem_rulebases('OWLPRIME'),null,null)) order by 1;
 
-- cleanup
exec sem_apis.drop_entailment('EMP0_IDX');
exec sem_apis.drop_entailment('EMP0_U_IDX');
exec sem_apis.drop_sem_model('EMP0');
exec sem_apis.drop_rulebase('FAMILY_RB');
drop table emp_rdf_data;
conn / as sysdba
drop user rdfuser cascade;
PK SEM_RDFSA Package Subprograms

13 SEM_RDFSA Package Subprograms

The SEM_RDFSA package contains subprograms (functions and procedures) for providing fine-grained access control to RDF data, using either a virtual private database (VPD) or Oracle Label Security (OLS). To use the subprograms in this chapter, you should understand the conceptual and usage information in Chapter 1, "Oracle Database Semantic Technologies Overview" and Chapter 5, "Fine-Grained Access Control for RDF Data".

This chapter provides reference information about the subprograms, listed in alphabetical order.


SEM_RDFSA.ADD_VPD_CONSTRAINT

Format

SEM_RDFSA.ADD_VPD_CONSTRAINT(

     policy_name IN VARCHAR2,

     constr_name IN VARCHAR2,

     match_pattern IN VARCHAR2,

     apply_pattern IN VARCHAR2,

     constr_group IN VARCHAR2 DEFAULT NULL);

Description

Adds a data access constraint to a VPD policy.

Parameters

policy_name

Name of an existing VPD policy.

constr_name

Unique case-insensitive name for the constraint to be defined in the VPD policy.

match_pattern

Match pattern that determines the target of the constraint.

apply_pattern

The constraint's apply pattern that enforces the access restriction.

constr_group

Case-insensitive name for the group to which the constraint belongs.

Usage Notes

This procedure adds a data access constraint to a VPD policy. Only the owner of a VPD policy can add a new constraint to the policy. The newly added constraint comes into effect immediately and it is enforced for queries on all RDF models associated with the VPD policy.

A constraint group may be specified for each policy such that they can be selectively activated and deactivated at runtime based on the application context (as explained in Section 5.1.1). A constraint with no constraint group is always active. See Section 5.1.3 for details about the match and apply patterns passed to this procedure.

See also the MDSYS.RDFVPD_POLICY_CONSTRAINTS view, which is described in Section 5.1.6.

For information about support for VPD, see Section 5.1.

Examples

The following example adds data access constraint for the RDF Class Contracts to the VPD policy and assigns it a constraint group named user:

begin
  sem_rdfsa.add_vpd_constraint(
          policy_name   => 'contracts_policy',
          constr_name   => 'andy_constraint_1',
          match_pattern =>'{?contract  rdf:type
                              <http://www.myorg.com/classes/Contract>}',
          apply_pattern => '{?contract  pred:hasMember      
              "sys_context('sa$appctx','app_user_uri'}"^^orardf:instruction }', 
          constr_group  => 'user');
end;
/

The preceding example makes use of secure application context to identify the user issuing the query and binds its value in the apply pattern to ensure that the user has access to a specific contract. The secure application context with the namespace sa$appctx is expected to initialize the value for the attribute app_user_uri to the URI identifying the user logged in.

The following example adds a data access constraint for the RDF property hasContractValue to the same VPD policy and assigns it to a constraint group named vp.

begin
  sem_rdfsa.add_vpd_constraint(
          policy_name   => 'contracts_policy',
          constr_name   => 'vp_access_to_contr_value',
          match_pattern => '{?contract  pred:hasContractValue ?cvalue }',
          apply_pattern => '{?contract  pred:drivenBy         ?dept .
                             ?dept      pred:hasVP
              "sys_context('sa$appctx','app_user_uri'}"^^orardf:instruction }', 
          constr_group  => 'vp');
end;
/

SEM_RDFSA.APPLY_OLS_POLICY

Format

SEM_RDFSA.APPLY_OLS_POLICY(

     policy_name IN VARCHAR2,

     rdfsa_options IN NUMBER DEFAULT SEM_RDFSA.SECURE_SUBJECT,

     table_options IN VARCHAR2 DEFAULT 'ALL_CONTROL',

     label_function IN VARCHAR2 DEFAULT NULL,

     predicate IN VARCHAR2 DEFAULT NULL);

Description

Applies an OLS policy to the semantic data store.

Parameters

policy_name

Name of an existing OLS policy.

rdfsa_options

Options specifying the mode of fine-grained access control to be enabled for RDF data. The default option for securing RDF data involves assigning sensitivity labels for the resources appearing the triples' subject position. You can override the defaults by using the rdfsa_options parameter and specifying one of the constants defined in Table 13-1 in the Usage Notes.

table_options

Policy enforcement options. The default value (ALL_CONTROL) is the only supported value for this procedure.

label_function

A string invoking a function to return a label value to use as the default.

predicate

An additional predicate to combine with the label-based predicate.

Usage Notes

The OLS policy specified with this procedure must be created with CTXT1 as the column name, and it should use default policy options. For information about policy options, see Oracle Label Security Administrator's Guide.

This procedure invokes the sa_policy_admin.apply_table_policy procedure on multiple tables defined in the MDSYS schema. The parameters table_options, label_function, and predicate for the SEM_RDFSA.APPLY_OLS_POLICY procedure have same semantics as the parameters with same names in the sa_policy_admin.apply_table_policy procedure.

For the rdfsa_options parameter, you can specify the package constant for the desired option. Table 13-1 lists these constants and their descriptions.

Table 13-1 SEM_RDFSA Package Constants for rdfsa_options Parameter

ConstantDescription

SEM_RDFSA.SECURE_SUBJECT

Assigns sensitivity labels for the resources appearing the triples' subject position.

SEM_RDFSA.SECURE_PREDICATE

Assigns sensitivity labels for the resources appearing the triples' predicate position.

SEM_RDFSA.SECURE_OBJECT

Assigns sensitivity labels for the resources appearing the triples' object position.

SEM_RDFSA.TRIPLE_LEVEL_ONLY

Applies triple-level security. Provides good performance, and eliminates the need to assign labels to individual resources. (Requires that Patch 9819833, available from My Oracle Support, be installed.)

SEM_RDFSA.OPT_DEFINE_BEFORE_USE

Restricts the use of an RDF resource in a triple before the sensitivity label is defined for the resource. If this option is not specified, the user's initial row label is used as the default label for the resource upon first use.

SEM_RDFSA.OPT_RELAX_TRIPLE_LABEL

Relaxes the dominating relationship that exists between the triple label and the labels associated with all its components. With this option, a triple can be defined if the user has READ access to all the triple components and the triple label may not bear any relationship with the component labels. Without this option, the triple label should at least cover the label for all its components.


You can specify a function in the label_function parameter to generate custom labels for newly inserted triples. The label function is associated with the MDSYS.RDF_LINK$ table, and the columns in this table may be configured as parameters to the label function as shown in the following example:

fgac_admin.new_triple_label(:new.model_id,
                            :new.start_node_id,
                            :new.p_value_id,
                            :new.canon_end_node_id)'

Because the OLS policy is applied to more than one table with different structures, the only valid column reference in any predicates assigned to the predicate parameter is that of the label column: CTXT1. If OLS is enabled for a semantic data store with existing data, you can specify a predicate of the form 'OR CTXT1 is null' to be able to continue using this data with no access restrictions.

An OLS-enabled semantic data store uses sensitivity labels for all the RDF triples organized in multiple models. User access to such triples, through model views and SEM_MATCH queries, is restricted by the OLS policy. Additionally, independent of a user owning the application table, access to the triple column (of type SDO_RDF_TRIPLE_S) in the table is restricted to users with FULL access privileges with the OLS policy.

The triples are inserted into a specific RDF model using the INSERT privileges on the corresponding application table. A sensitivity label for the new triple is generated using the user's session context (initial row label) or the label function. The triple is validated for any RDF policy violations using labels associated with the triple components. Although the triple information may not be accessed trough the application table, the model view may be queried to access the triples, while enforcing the OLS policy restrictions. If you have the necessary policy privileges (such as writeup, writeacross), you can update the CTXT1 column in the model view to reset the label assigned to the triple. The new label is automatically validated for any RDF policy violations involving the triple components. Update privilege on the CTXT1 column of the model view is granted to the owner of the model, and this user may selectively grant this privilege to other users.

If the RDF models are created in schemas other than the user with FULL access, necessary privileges on the model objects -- specifically, read/write access on the application table, read access to the model view, and write access to the CTXT1 column in the model view -- can be granted to such users for maintenance operations. These operations include bulk loading into the model, resetting any sensitivity labels assigned to the triples, and creating entailments using the model.

To disable the OLS policy, use the SEM_RDFSA.DISABLE_OLS_POLICY procedure.

For information about support for OLS, see Section 5.2.

Examples

The following example enable secure access to RDF data with secure subject and secure predicate options.

begin
  sem_rdfsa.apply_ols_policy(
        policy_name   => 'defense',
        rdfsa_options => sem_rdfsa.SECURE_SUBJECT+
                         sem_rdfsa.SECURE_PREDICATE); 
end;
/

The following example extends the preceding example by specifying a Define Before Use option, which allows a user to define a triple only if the triple components secured (Subject, Predicate or Object) are predefined with an associated sensitivity label. This configuration is effective if the user inserting the triple does not have execute privileges on the SEM_RDFSA package.

begin
  sem_rdfsa.apply_ols_policy(
        policy_name   => 'defense',
        rdfsa_options => sem_rdfsa.SECURE_SUBJECT+
                         sem_rdfsa.SECURE_PREDICATE+
                         sem_rdfsa.OPT_DEFINE_BEFORE_USE); 
end;
/

SEM_RDFSA.APPLY_VPD_POLICY

Format

SEM_RDFSA.APPLY_VPD_POLICY(

     policy_name IN VARCHAR2,

     model_name IN VARCHAR2);

Description

Applies a VPD policy to an RDF model.

Parameters

policy_name

Name of an existing VPD policy.

model_name

Name of the model to which to apply the VPD policy.

Usage Notes

This procedure applies a VPD policy to an RDF model. The owner of a VPD policy can apply it to any model in the database instance. The owner must also have EXECUTE privileges on the SYS.DBMS_RLS package.

After a policy is applied to an RDF model, the data stored in the model and the data inferred from this model can be accessed only using SPARQL query patterns within a SEM_MATCH operation. All other forms of data access (such as through model views or using classic graph pattern syntax in a SEM_MATCH query) are not permitted.

A VPD policy may not applied to a model participating in a virtual model. Similarly, a virtual model may not be defined to include any VPD-enabled models

See also the MDSYS.RDFVPD_POLICIES view, which is described in Section 5.1.4.

For information about support for VPD, see Section 5.1.

Examples

The following example applies a VPD policy to an RDF model.

begin
  sem_rdfsa.apply_vpd_policy(
      policy_name => 'contracts_policy',
      model_name  => 'contracts');
end;
/

SEM_RDFSA.CREATE_VPD_POLICY

Format

SEM_RDFSA.CREATE_VPD_POLICY(

     policy_name IN VARCHAR2,

     namespace_map IN RDF_ALIASES DEFAULT NULL,

    policy_context IN VARCHAR2 DEFAULT NULL);

Description

Creates a new VPD policy in a user schema.

Parameters

policy_name

Name for the VPD policy.

namespace_map

A mapping of namespaces and their prefixes to be used in the VPD constraints.

policy_context

Name of the context created specifically to manage constraint groups defined in the VPD policy.

Usage Notes

You must have EXECUTE privileges on the SYS.DBMS_RLS package.

The namespace map associated with the policy may be used to define namespace prefixes that are subsequently used in the policy's metadata statements added using the SEM_RDFSA.MAINT_VPD_METADATA procedure and the policy's data access constraints defined using the SEM_RDFSA.ADD_VPD_CONSTRAINT procedure.

The name of the application context assigned to the policy_context parameter is used to manage the constraint groups dynamically at runtime. Such context must be created using the CREATE CONTEXT command before associating it with the VPD policy. Each data access constraint defined in the VPD policy may specify the name of a constraint group to which it belongs. At runtime, the name of the constraint group is matched with the name of the policy_context value to selectively activate the group. (For more information, see Section 5.1.1).

See also the MDSYS.RDFVPD_POLICIES view, which is described in Section 5.1.4.

For information about support for VPD, see Section 5.1.

Examples

The following example creates a VPD policy with a namespace map and also associates a policy context to manage constraint groups.

create context contracts_appctx using sec_admin.contracts_appctx;
begin
  sem_rdfsa.create_vpd_policy(
        policy_name  => 'contracts_policy',
        namespace_map => 
          mdsys.rdf_aliases(                                                    
            mdsys.rdf_alias('','http://www.myorg.com/classes/'),                                                     
            mdsys.rdf_alias('pred','http://www.myorg.com/pred/'),
            mdsys.rdf_alias('emp','http://www.myorg.com/employee/')),
        policy_context => 'contracts_appctx');
end;
/

SEM_RDFSA.DELETE_VPD_CONSTRAINT

Format

SEM_RDFSA.DELETE_VPD_CONSTRAINT(

     policy_name IN VARCHAR2,

     constr_name IN VARCHAR2);

Description

Deletes a data access constraint from a VPD policy.

Parameters

policy_name

Name of an existing VPD policy.

constr_name

Unique case-insensitive name for the constraint to be deleted from the VPD policy.

Usage Notes

This procedure deletes a data access constraint from a VPD policy. Only the owner of a VPD policy can delete a constraint from the policy.

See also the MDSYS.RDFVPD_POLICY_CONSTRAINTS view, which is described in Section 5.1.6.

For information about support for VPD, see Section 5.1.

Examples

The following example adds data access constraint named vp_access_to_contr_value from the VPD policy named contracts_policy:

begin
  sem_rdfsa.delete_vpd_policy(
      policy_name   => 'contracts_policy',
      constr_name   => 'vp_access_to_contr_value'); 
end;
/

SEM_RDFSA.DISABLE_OLS_POLICY

Format

SEM_RDFSA.DISABLE_OLS_POLICY;

Description

Disables the OLS policy that has been previously applied to or enabled on the semantic data store.

Parameters

(None.)

Usage Notes

You can use this procedure to disable temporarily the OLS policy that had been applied to or enabled for the semantic data store. The user disabling the policy should have the necessary privileges to administer OLS policies and should also have access to the OLS policy applied to RDF data.

The sensitivity labels assigned to various RDF resources and triples are preserved and the OLS policy may be re-enabled to enforce them. New resources with specific labels can be added, or labels for existing triples and resources can be updated when the OLS policy is disabled.

To apply an OLS policy, use the SEM_RDFSA.APPLY_OLS_POLICY procedure; to enable an OLS policy that had been disabled, use the SEM_RDFSA.ENABLE_OLS_POLICY procedure.

For information about support for OLS, see Section 5.2.

Examples

The following example disables the OLS policy for the semantic data store.

begin
  sem_rdfsa.disable_ols_policy;
end;
/

SEM_RDFSA.DROP_VPD_POLICY

Format

SEM_RDFSA.DROP_VPD_POLICY(

     policy_name IN VARCHAR2);

Description

Drops an unused VPD policy.

Parameters

policy_name

Name of an existing VPD policy.

Usage Notes

This procedure permanently drops an unused VPD policy along with its metadata and constraints. You must be the owner of a VPD policy or a user with a DBA role to perform this operation.

For information about support for VPD, see Section 5.1.

Examples

The following example adds data access constraint named vp_access_to_contr_value from the VPD policy named contracts_policy:

begin
  sem_rdfsa.delete_vpd_policy(
      policy_name   => 'contracts_policy',
      constr_name   => 'vp_access_to_contr_value'); 
end;
/

SEM_RDFSA.ENABLE_OLS_POLICY

Format

SEM_RDFSA.ENABLE_OLS_POLICY;

Description

Enables the OLS policy that has been previously disabled.

Parameters

(None.)

Usage Notes

You can use this procedure to enable the OLS policy that had been disabled for the semantic data store. The user enabling the policy should have the necessary privileges to administer OLS policies and should also have access to the OLS policy applied to RDF data.

To disable an OLS policy, use the SEM_RDFSA.DISABLE_OLS_POLICY procedure.

For information about support for OLS, see Section 5.2.

Examples

The following example enables the OLS policy for the semantic data store.

begin
  sem_rdfsa.enable_ols_policy;
end;
/

SEM_RDFSA.MAINT_VPD_METADATA

Format

SEM_RDFSA.MAINT_VPD_METADATA(

     policy_name IN VARCHAR2,

     t_subject IN VARCHAR2,

     t_predicate IN VARCHAR2,

     t_object IN VARCHAR2,

     action IN VARCHAR2 DEFAULT 'ADD');

Description

Maintains the VPD metadata by accepting RDF schema statements with their subject, predicate, and object terms.

Parameters

policy_name

Name of an existing VPD policy.

t_subject

Subject of the metadata triple.

t_predicate

Predicate of the metadata triple.

t_object

Object of the metadata triple.

action

Maintenance operation to be performed: ADD (add the specified triple to the metadata) or DELETE (delete the specified triple from the metadata).

Usage Notes

This procedure maintains the metadata associated with the VPD policy by accepting RDF Schema statements and performing appropriate actions on its dictionary. Only the owner of the VPD policy or a user with FULL access to the VPD policy can maintain its metadata.

The types metadata statements that you can add using this procedure are determined by the terms assigned to the t_predicate parameter. The accepted values for this parameter are:

http://www.w3.org/2000/01/rdf-schema#domain
http://www.w3.org/2000/01/rdf-schema#range
http://www.w3.org/2000/01/rdf-schema#subClassOf
http://www.w3.org/2000/01/rdf-schema#subPropertyOf
http://www.w3.org/2002/07/owl#equivalentProperty

MDSYS.RDFVPD_PREDICATE_MDATA and MDSYS.RDFVPD_RESOURCE_REL views, which are described in Section 5.1.7 and Section 5.1.8, respectively.

For information about support for VPD, see Section 5.1.

Examples

The following example specifies Contract as a subclass of Project and adds domain information for the hasContractValue property:

begin
  sem_rdfsa.maint_vpd_metadata(
      policy_name => 'contracts_policy',
      t_subject   => '<http://www.myorg.com/classes/Project>',
      t_predicate => '<http://www.w3.org/2000/01/rdf-schema#subClassOf>',
      t_object    => '<http://www.myorg.com/classes/Contract>',
      action      => 'add');
end;
 
begin
  sem_rdfsa.maint_vpd_metadata(
      policy_name => 'contracts_policy',
      t_subject   => '<http://www.myorg.com/property/hasConfValue>',
      t_predicate => 'rdfs:domain',
      t_object    => '<http://www.myorg.com/classes/Contract>');
end;

SEM_RDFSA.REMOVE_OLS_POLICY

Format

SEM_RDFSA.REMOVE_OLS_POLICY;

Description

Permanently removes or detaches the OLS policy from the semantic data store.

Parameters

(None.)

Usage Notes

You should have the necessary privileges to administer OLS policies, and you should also have access to the OLS policy applied to RDF data. Once the OLS policy is detached from the semantic data store, all the sensitivity labels previously assigned to the triples and resources are lost.

This operation drops objects that are specifically created to maintain the RDF security policies.

To apply an OLS policy, use the SEM_RDFSA.APPLY_OLS_POLICY procedure.

For information about support for OLS, see Section 5.2.

Examples

The following= example removes the OLS policy that had been previously applied to the semantic data store.

begin
  sem_rdfsa.remove_ols_policy;
end;
/

SEM_RDFSA.REMOVE_VPD_POLICY

Format

SEM_RDFSA.REMOVE_VPD_POLICY(

     policy_name IN VARCHAR2,

     model_name IN VARCHAR2);

Description

Removes (detaches) a VPD policy from an RDF model.

Parameters

policy_name

Name of an existing VPD policy.

model_name

Name of the model to which to apply the VPD policy.

Usage Notes

This procedure permanently removes or detaches a VPD policy from an RDF Model. You must be the owner of the VPD policy or a user with a DBA role to perform this operation.

After a VPD policy is removed, fine-grained access control for the specific model is turned off and the users with SELECT privileges on the model view can access all the data in the model.

For information about support for VPD, see Section 5.1.

Examples

The following example removes a VPD policy from an RDF model.

begin
  sem_rdfsa.remove_vpd_policy(
      policy_name => 'contracts_policy',
      model_name  => 'contracts');
end;
/

SEM_RDFSA.RESET_MODEL_LABELS

Format

SEM_RDFSA.RESET_MODEL_LABELS(

     model_name IN VARCHAR2);

Description

Resets the labels associated with a model or with global resources; requires that the associated model or models be empty.

Parameters

model_name

Name of the model for which the labels should be reset, or the string RDF$GLOBAL to reset the labels associated with all global resources.

Usage Notes

If you specify a model name, the model must be empty. If you specify RDF$GLOBAL, all the models must be empty (that is, no triples in the RDF repository).

You must have FULL access privilege with the OLS policy applied to the semantic data store.

For information about support for OLS, see Section 5.2.

Examples

The following example removes all resources and their labels associated with the Contracts model.

begin
   sem_rdfsa.reset_model_labels(model_name => 'Contracts');
end;
/

SEM_RDFSA.SET_PREDICATE_LABEL

Format

SEM_RDFSA.SET_PREDICATE_LABEL(

     model_name IN VARCHAR2,

     predicate IN VARCHAR2,

     label_string IN VARCHAR2);

Description

Sets a sensitivity label for a predicate at the model level or for the whole repository.

Parameters

model_name

Name of the model to which the predicate belongs, or the string RDF$GLOBAL if the same label should applied for the use of the predicate in all models.

predicate

Predicate for which the label should be assigned.

label_string

OLS row label in string representation.

Usage Notes

If you specify a model name, you must have read access to the model and execute privileges on the SEM_RDFSA package to perform this operation. If you specify RDF$GLOBAL, you must have FULL access privilege with the OLS policy applied to RDF data.

You must have access to the specified label and OLS policy privilege to overwrite an existing label if a label already exists for the predicate. The SECURE_PREDICATE option must be enabled for RDF data.

If an existing predicate label is updated with this operation, the labels for the triples using this predicate must all dominate the new predicate label. The only exception is when the OPT_RELAX_TRIPLE_LABEL option is chosen for the OLS-enabled RDF data.

If you specify RDF$GLOBAL, a global predicate with a unique sensitivity label across models is created. If the same predicate is previously defined in one or more models, the global label dominates all such labels and the model-specific labels are replaced for the given predicate.

After a label for a predicate is set, new triples with the predicate can be added only if the triple label (which may be initialized from user's initial row label or using a label function) dominates the predicate's sensitivity label. This dominance relationship can be relaxed with the OPT_RELAX_TRIPLE_LABEL option, in which case the user should at least have read access to the predicate to be able to define a new triple using the predicate.

For information about support for OLS, see Section 5.2.

Examples

The following example sets a predicate label for Contracts model and another predicate label for all models in the database instance.

begin
  sem_rdfsa.set_predicate_label( 
         model_name   => 'contracts',
         predicate    => '<http://www.myorg.com/pred/hasContractValue>',
         label_string => 'TS:US_SPCL');
end;  
/
 
begin
  sem_rdfsa.set_predicate_label(
         model_name   => 'rdf$global',
         predicate    => '<http://www.myorg.com/pred/hasStatus>',
         label_string => 'SE:US_SPCL:US');
end;
/

SEM_RDFSA.SET_RDFS_LABEL

Format

SEM_RDFSA.SET_RDFS_LABEL(

     label_string IN VARCHAR2,

     inf_override IN VARCHAR2);

Description

Sets a sensitivity label for RDFS schema elements.

Parameters

label_string

OLS row label in string representation, to be used as the sensitivity label for all RDF schema constructs.

inf_override

OLS row label to be used as the override for generating labels for inferred triples.

Usage Notes

This procedure sets or resets the sensitivity label associated with the RDF schema resources, often recognized by http://www.w3.org/1999/02/22-rdf-syntax-ns# and http://www.w3.org/2000/01/rdf-schema# prefixes for their URIs. You can assign a sensitivity label with restricted access to these resources, so that operations such as creating new RDF classes and adding new properties can be restricted to users with higher privileges.

You must have FULL access privilege with policy applied to RDF data.

RDF schema elements implicitly use the relaxed triple label option, so that the triples using RDFS and OWL constructs for subject, predicate, or object are not forced to have a sensitivity label that dominates the labels associated with the schema constructs. Therefore, a user capable of defining new RDF classes and properties must least have read access to the schema elements.

When RDF schema elements are referred to in the inferred triples, the system-defined and custom label generators consider the inference override label in determining the appropriate label for the inferred triples. If a custom label generator is used, this override label is passed instead of the actual label when an RDF schema element is involved.

For information about support for OLS, see Section 5.2.

Examples

The following example sets a label with a unique compartment for all RDF schema elements. A user capable of defining new RDF classes and properties is expected to have an exclusive membership to the compartment.

begin
  sem_rdfsa.set_rdfs_label( 
         label_string  => 'SE:RDFS:',
         inf_override  => 'SE:US_SPCL:US');
end;  
/

SEM_RDFSA.SET_RESOURCE_LABEL

Format

SEM_RDFSA.SET_RESOURCE_LABEL(

     model_name IN VARCHAR2,

     resource_uri IN VARCHAR2,

     label_string IN VARCHAR2,

     resource_pos IN VARCHAR2 DEFAULT 'S');

Description

Sets a sensitivity label for a resource that may be used in the subject and/or object position of a triple.

Parameters

model_name

Name of the model to which the resource belongs, or the string RDF$GLOBAL if the same label should applied for using the resource in all models.

resource_uri

URI for the resource that may be used as subject or object in one or more triples.

label_string

OLS row label in string representation.

resource_pos

Position of the resource within a triple: S, O, or S,O. You can specify up to two separate labels for the same resource, one to be considered when the resource is used in the subject position of a triple and the other to be considered when it appears in the object position. The values 'S', 'O' or 'S,O' set a label for the resource in subject, object or both subject and object positions, respectively.

Usage Notes

If you specify a model name, you must have read access to the model and execute privileges on the SEM_RDFSA package to perform this operation. If you specify RDF$GLOBAL, you must have FULL access privilege with the OLS policy applied to RDF data.

You must have access to the specified label and OLS policy privilege to overwrite an existing label if a label already exists for the predicate. The SECURE_PREDICATE option must be enabled for RDF data.

If an existing resource label is updated with this operation, the labels for the triples using this resource in the specified position must all dominate the new resource label. The only exception is when the OPT_RELAX_TRIPLE_LABEL option is chosen for the OLS-enabled RDF data.

If you specify RDF$GLOBAL, a global resource with a unique sensitivity label across models is created. If the same resource is previously defined in one or more models with the same triple position, the global label dominates all such labels and the model-specific labels are replaced for the given resource in that position.

After a label for a predicate is set, new triples using the resource in the specified position can be added only if the triple label dominates the resource's sensitivity label. This dominance relationship can be relaxed with OPT_RELAX_TRIPLE_LABEL option, in which case, the user should at least have read access to the resource.

For information about support for OLS, see Section 5.2.

Examples

The following example sets sensitivity labels for multiple resources based on their position.

begin
  sem_rdfsa.set_resource_label(
         model_name   => 'contracts',
         resource_uri => '<http://www.myorg.com/contract/projectHLS>',
         label_string => 'SE:US_SPCL:US',
         resource_pos => 'S,O');
end;
/
 
begin
  sem_rdfsa.set_resource_label(
       model_name   => 'rdf$global',
       resource_uri => '<http://www.myorg.com/contract/status/Complete>',
       label_string => 'SE:US_SPCL:US',
       resource_pos => 'O');
end;
/

SEM_RDFSA.SET_RULE_LABEL

Format

SEM_RDFSA.SET_RULE_LABEL(

     rule_base IN VARCHAR2,

     rule_name IN VARCHAR2,

     label_string IN VARCHAR2);

Description

Sets sensitivity label for a rule belonging to a rulebase.

Parameters

rule_base

Name of an existing RDF rulebase.

rule_name

Name of the rule belonging to the rulebase.

label_string

OLS row label in string representation.

Usage Notes

The sensitivity label assigned to the rule is used to generate the label for the inferred triples when an appropriate label generator option is chosen.

You must have access have access to the rulebase, and you must have FULL access privilege with the OLS policy can assign labels for system-defined rules in the RDFS rulebase.

There is no support for labels assigned to user-defined rules.

For information about support for OLS, see Section 5.2.

Examples

The following example assigns a sensitivity label for an RDFS rule.

begin
sem_rdfsa.set_rule_label (rule_base    => 'RDFS',
                          rule_name    => 'RDF-AXIOMS',
                          label_string => 'SE:US_SPCL:');
end;
/
PKjyxPK.AOEBPS/sem_spatial_orageo.htm SEM_MATCH Support for Spatial Queries

B SEM_MATCH Support for Spatial Queries

This appendix provides reference information for SPARQL extension functions for performing spatial queries in SEM_MATCH. To use these functions, you must understand the concepts explained in Section 1.6.6, "Spatial Support".

This appendix includes the following functions:


orageo:area

Format

orageo:area(geom1 : orageo:WKTLiteral, unit : Literal) : xsd:decimal

Description

Returns the area of geom1 in terms of the specified unit of measure.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

unit

Unit of measurement: a quoted string with an SDO_UNIT value from the MDSYS.SDO_DIST_UNITS table (for example, "unit=SQ_KM"). See the section about unit of measurement support in Oracle Spatial Developer's Guide for more information about unit of measurement specification.

Usage Notes

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_AREA function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons with areas greater than 10,000 square kilometers.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:area(?cgeom, "unit=SQ_KM") > 10000) }'
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:buffer

Format

orageo:buffer(geom1 : orageo:WKTLiteral, distance : xsd:decimal, unit : Literal) : orageo:WKTLiteral

Description

Returns a buffer polygon at a specified distance around or inside a geometry.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

distance

Distance value. Distance value. If the value is positive, the buffer is generated around geom1; if the value is negative (valid only for polygons), the buffer is generated inside geom1.

unit

Unit of measurement: a quoted string with an SDO_UNIT value from the MDSYS.SDO_DIST_UNITS table (for example, "unit=KM"). See the section about unit of measurement support in Oracle Spatial Developer's Guide for more information about unit of measurement specification.

Usage Notes

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_BUFFER function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons that are completely inside a 100 kilometer buffer around a specified point.

SELECT name, cdist
FROM table(sem_match(
'{ # HINT0={LEADING(?cgeom)}
   ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (
     orageo:relate(?cgeom, 
       orageo:buffer("POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
                     100, "unit=KM"),
      "mask=inside")) }'
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:centroid

Format

orageo:centroid(geom1 : orageo:WKTLiteral) : orageo:WKTLiteral

Description

Returns a point geometry that is the centroid of geom1. (The centroid is also known as the "center of gravity.")

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

Usage Notes

For an input geometry consisting of multiple objects, the result is weighted by the area of each polygon in the geometry objects. If the geometry objects are a mixture of polygons and points, the points are not used in the calculation of the centroid. If the geometry objects are all points, the points have equal weight.

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_CENTROID function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons with centroids within 200 kilometers of a specified point.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:withinDistance(orageo:centroid(?cgeom), 
      "POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
      "distance=200 unit=KM")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:convexHull

Format

orageo:convexHull(geom1 : orageo:WKTLiteral) : orageo:WKTLiteral

Description

Returns a polygon-type object that represents the convex hull of geom1. (The convex hull is a simple convex polygon that completely encloses the geometry object, using as few straight-line sides as possible to create the smallest polygon that completely encloses the geometry object.)

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

Usage Notes

A convex hull is a convenient way to get an approximation of a complex geometry object.

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_CONVEX_HULL function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons whose convex hull contains a specified point.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:relate(orageo:convexHull(?cgeom), 
      "POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
      "mask=contains")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:difference

Format

orageo:difference(geom1 : orageo:WKTLiteral, geom2 : orageo:WKTLiteral) : orageo:WKTLiteral

Description

Returns a geometry object that is the topological difference (MINUS operation) of geom1 and geom2.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

geom2

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

Usage Notes

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_UNION function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons whose centroid is inside the difference of two specified polygons.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:relate(orageo:centroid(?cgeom), 
      orageo:difference("Polygon((-83.6 34.1, -83.2 34.1, -83.2 34.5, 
                         -83.6 34.5, -83.6 34.1))"^^orageo:WKTLiteral,
                        "Polygon((-83.2 34.3, -83.0 34.3, -83.0 34.5, 
                         -83.2 34.5, -83.2 34.3))"^^orageo:WKTLiteral),
      "mask=inside")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:distance

Format

orageo:distance(geom1 : orageo:WKTLiteral, geom2 : orageo:WKTLiteral, unit : Literal) : xsd:decimal

Description

Returns the distance between the nearest pair of points or segments of geom1 and geom2 in terms of the specified unit of measure.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

geom2

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

unit

Unit of measurement: a quoted string with an SDO_UNIT value from the MDSYS.SDO_DIST_UNITS table (for example, "unit=KM"). See the section about unit of measurement support in Oracle Spatial Developer's Guide for more information about unit of measurement specification.

Usage Notes

Use orageo:withinDistance instead of orageo:distance whenever possible, because orageo:withinDistance has a more efficient index-based implementation.

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_DISTANCE function in Oracle Spatial Developer's Guide.

Example

The following example finds the ten nearest U.S. Congressional districts to a specified point and orders them by distance from the point.

SELECT name, cdist
FROM table(sem_match(
'SELECT ?name ?cdist
 WHERE
 { # HINT0={LEADING(?cgeom)}
   ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:nearestNeighbor(?cgeom, 
      "POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
      "sdo_num_res=10")) }
 ORDER BY ASC(orageo:distance(?cgeom,
                "POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
                "unit=KM"))'
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '))
ORDER BY sem$rownum;

orageo:intersection

Format

orageo:intersection(geom1 : orageo:WKTLiteral, geom2 : orageo:WKTLiteral) : orageo:WKTLiteral

Description

Returns a geometry object that is the topological intersection (AND operation) of geom1 and geom2.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

geom2

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

Usage Notes

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_INTERSECTION function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons whose centroid is inside the intersection of two specified polygons.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:relate(orageo:centroid(?cgeom), 
      orageo:intersection("Polygon((-83.6 34.1, -83.2 34.1, -83.2 34.5, 
                           -83.6 34.5, -83.6 34.1))"^^orageo:WKTLiteral,
                          "Polygon((-83.2 34.3, -83.0 34.3, -83.0 34.5, 
                           -83.2 34.5, -83.2 34.3))"^^orageo:WKTLiteral),
      "mask=inside")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:length

Format

orageo:length(geom1 : orageo:WKTLiteral, unit : Literal) : xsd:decimal

Description

Returns the length or perimeter of geom1 in terms of the specified unit of measure.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

unit

Unit of measurement: a quoted string with an SDO_UNIT value from the MDSYS.SDO_DIST_UNITS table (for example, "unit=KM"). See the section about unit of measurement support in Oracle Spatial Developer's Guide for more information about unit of measurement specification.

Usage Notes

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_LENGTH function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons with lengths (perimeters) greater than 1000 kilometers.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:legnth(?cgeom, "unit=KM") > 1000) }'
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:mbr

Format

orageo:mbr(geom1 : orageo:WKTLiteral) : orageo:WKTLiteral

Description

Returns the minimum bounding rectangle of geom1, that is, the single rectangle that minimally encloses geom1.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

Usage Notes

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_MBR function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons whose minimum bounding rectangle contains a specified point.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:relate(orageo:mbr(?cgeom), 
      "POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
      "mask=contains")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:nearestNeighbor

Format

orageo:nearestNeighbor(geom1: orageo:WKTLiteral, geom2 : orageo:WKTLiteral, param : Literal) : xsd:boolean

Description

Returns true if geom1 is a nearest neighbor of geom2, where the size of the nearest neighbors set is specified by param; returns false otherwise.

Parameters

geom1

Geometry object. Specified as a query variable.

geom2

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

param

Determines the behavior of the operator. See the Usage Notes for the available keyword-value pairs.

Usage Notes

In the param parameter, the available keyword-value pairs are:

  • distance=n specifies the maximum allowable distance for the nearest neighbor search.

  • sdo_num_res=n specifies the size of the set for the nearest neighbor search.

  • unit=unit specifies the unit of measurement to use with distance value. If you do not specify a value, the unit of measurement associated with the data is used.

geom1 must be a local variable (that is, a variable that appears in the basic graph pattern that contains the orageo:nearestNeighbor spatial filter).

It is a good idea to use a 'LEADING(?var)' HINT0 hint when your query involves a restrictive orageo:relate spatial filter on ?var.

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_NN operator in Oracle Spatial Developer's Guide.

Example

The following example finds the ten nearest U.S. Congressional districts to a specified point.

SELECT name, cdist
FROM table(sem_match(
'{ # HINT0={LEADING(?cgeom)}
   ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:nearestNeighbor(?cgeom, 
      "POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
      "sdo_num_res=10")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:relate

Format

orageo:relate(geom1: orageo:WKTLiteral, geom2 : orageo:WKTLiteral, param : Literal) : xsd:boolean

Description

Returns true if geom1 and geom2 satisfy the topological spatial relation specified by the param parameter; returns false otherwise.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

geom2

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

param

Specifies a list of mask relationships to check. See the list of keywords in the Usage Notes.

Usage Notes

The following param values (mask relationships) can be tested:

  • ANYINTERACT: Returns TRUE if the objects are not disjoint.

  • CONTAINS: Returns CONTAINS if the second object is entirely within the first object and the object boundaries do not touch; otherwise, returns FALSE.

  • COVEREDBY: Returns COVEREDBY if the first object is entirely within the second object and the object boundaries touch at one or more points; otherwise, returns FALSE.

  • COVERS: Returns COVERS if the second object is entirely within the first object and the boundaries touch in one or more places; otherwise, returns FALSE.

  • DISJOINT: Returns DISJOINT if the objects have no common boundary or interior points; otherwise, returns FALSE.

  • EQUAL: Returns EQUAL if the objects share every point of their boundaries and interior, including any holes in the objects; otherwise, returns FALSE.

  • INSIDE: Returns INSIDE if the first object is entirely within the second object and the object boundaries do not touch; otherwise, returns FALSE.

  • ON: Returns ON if the boundary and interior of a line (the first object) is completely on the boundary of a polygon (the second object); otherwise, returns FALSE.

  • OVERLAPBDYDISJOINT: Returns OVERLAPBDYDISJOINT if the objects overlap, but their boundaries do not interact; otherwise, returns FALSE.

  • OVERLAPBDYINTERSECT: Returns OVERLAPBDYINTERSECT if the objects overlap, and their boundaries intersect in one or more places; otherwise, returns FALSE.

  • TOUCH: Returns TOUCH if the two objects share a common boundary point, but no interior points; otherwise, returns FALSE.

Values for param can be combined using the logical Boolean operator OR. For example, 'INSIDE + TOUCH' returns INSIDE+TOUCH if the relationship between the geometries is INSIDE or TOUCH or both INSIDE and TOUCH; it returns FALSE if the relationship between the geometries is neither INSIDE nor TOUCH.

When invoking orageo:relate with a query variable and a constant geometry, always use the query variable as the first parameter and the constant geometry as the second parameter.

For best performance, geom1 should be a local variable (that is, a variable that appears in the basic graph pattern that contains the orageo:relate spatial filter).

It is a good idea to use a 'LEADING(?var)' HINT0 hint when your query involves a restrictive orageo:relate spatial filter on ?var.

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_RELATE operator in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district that contains a specified point.

SELECT name, cdist
FROM table(sem_match(
'{ # HINT0={LEADING(?cgeom)}
   ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:relate(?cgeom, 
      "POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
      "mask=contains")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '
));

orageo:union

Format

orageo:union(geom1 : orageo:WKTLiteral, geom2 : orageo:WKTLiteral) : orageo:WKTLiteral

Description

Returns a geometry object that is the topological union (OR operation) of geom1 and geom2.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

geom2

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

Usage Notes

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_UN ION function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons whose centroid is inside the union of two specified polygons.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:relate(orageo:centroid(?cgeom), 
      orageo:union("Polygon((-83.6 34.1, -83.2 34.1, -83.2 34.5, 
                    -83.6 34.5, -83.6 34.1))"^^orageo:WKTLiteral,
                   "Polygon((-83.2 34.3, -83.0 34.3, -83.0 34.5, 
                    -83.2 34.5, -83.2 34.3))"^^orageo:WKTLiteral),
      "mask=inside")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:withinDistance

Format

orageo:withinDistance(geom1 : orageo:WKTLiteral, geom2 : orageo:WKTLiteral, distance : xsd:decimal, unit : Literal) : xsd:boolean

Description

Returns true if the distance between geom1 and geom2 is less than or equal to distance when measured in unit; returns false otherwise.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

geom2

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

distance

Distance value.

unit

Unit of measurement: a quoted string with an SDO_UNIT value from the MDSYS.SDO_DIST_UNITS table (for example, "unit=KM"). See the section about unit of measurement support in Oracle Spatial Developer's Guide for more information about unit of measurement specification.

Usage Notes

When invoking this function with a query variable and a constant geometry, always use the query variable as the first parameter and the constant geometry as the second parameter.

For best performance, geom1 should be a local variable (that is, a variable that appears in the basic graph pattern that contains the orageo:withinDistance spatial filter).

It is a good idea to use a 'LEADING(?var)' HINT0 hint when your query involves a restrictive orageo:withinDistance spatial filter on ?var.

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_WITHIN_DISTANCE operator in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional districts that are within 100 kilometers of a specified point.

SELECT name, cdist
FROM table(sem_match(
'{ # HINT0={LEADING(?cgeom)}
   ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:withinDistance(?cgeom, 
      "POINT(-71.46444 42.7575)"^^orageo:WKTLiteral,
      100, "KM")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));

orageo:xor

Format

orageo:xor(geom1 : orageo:WKTLiteral, geom2 : orageo:WKTLiteral) : orageo:WKTLiteral

Description

Returns a geometry object that is the topological symmetric difference (XOR operation) of geom1 and geom2.

Parameters

geom1

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

geom2

Geometry object. Specified as a query variable or a constant orageo:WKTLiteral value.

Usage Notes

See Section 1.6.6 for information about representing , indexing, and querying spatial data in RDF.

See also the SDO_GEOM.SDO_XOR function in Oracle Spatial Developer's Guide.

Example

The following example finds the U.S. Congressional district polygons whose centroid is inside the symmetric difference of two specified polygons.

SELECT name, cdist
FROM table(sem_match(
'{ ?person usgovt:name ?name .
   ?person pol:hasRole ?role .
   ?role pol:forOffice ?office .
   ?office pol:represents ?cdist .
   ?cdist orageo:hasExactGeometry ?cgeom 
   FILTER (orageo:relate(orageo:centroid(?cgeom), 
      orageo:xor("Polygon((-83.6 34.1, -83.2 34.1, -83.2 34.5, 
                  -83.6 34.5, -83.6 34.1))"^^orageo:WKTLiteral,
                 "Polygon((-83.2 34.3, -83.0 34.3, -83.0 34.5, 
                  -83.2 34.5, -83.2 34.3))"^^orageo:WKTLiteral),
      "mask=inside")) } '
,sem_models('gov_all_vm'), null, 
,sem_aliases(
   sem_alias('usgovt','http://www.rdfabout.com/rdf/schema/usgovt/'),
   sem_alias('pol','http://www.rdfabout.com/rdf/schema/politico/'))
,null, null, null, ' ALLOW_DUP=T '));
PK rhPK.A OEBPS/lot.htm3 List of Tables

List of Tables

PK#PK.AOEBPS/owl_concepts.htm OWL Concepts

2 OWL Concepts

This chapter describes concepts related to the support for a subset of the Web Ontology Language (OWL). It builds on the information in Chapter 1, and it assumes that you are familiar with the major concepts associated with OWL, such as ontologies, properties, and relationships. For detailed information about OWL, see the OWL Web Ontology Language Reference at http://www.w3.org/TR/owl-ref/.

This chapter contains the following major sections:

2.1 Ontologies

An ontology is a shared conceptualization of knowledge in a particular domain. It consists of a collection of classes, properties, and optionally instances. Classes are typically related by class hierarchy (subclass/ superclass relationship). Similarly, the properties can be related by property hierarchy (subproperty/ superproperty relationship). Properties can be symmetric or transitive, or both. Properties can also have domain, ranges, and cardinality constraints specified for them.

RDFS-based ontologies only allow specification of class hierarchies, property hierarchies, instanceOf relationships, and a domain and a range for properties.

OWL ontologies build on RDFS-based ontologies by additionally allowing specification of property characteristics. OWL ontologies can be further classified as OWL-Lite, OWL-DL, and OWL Full. OWL-Lite restricts the cardinality minimum and maximum values to 0 or 1. OWL-DL relaxes this restriction by allowing minimum and maximum values. OWL Full allows instances to be also defined as a class, which is not allowed in OWL-DL and OWL-Lite ontologies.

Section 2.1.2 describes OWL capabilities that are supported and not supported with semantic data.

2.1.1 Example: Cancer Ontology

Figure 2-1 shows part of a cancer ontology, which describes the classes and properties related to cancer. One requirement is to have a PATIENTS data table with a column named DIAGNOSIS, which must contain a value from the Diseases_and_Disorders class hierarchy.

Figure 2-1 Cancer Ontology Example

Description of Figure 2-1 follows
Description of "Figure 2-1 Cancer Ontology Example"

In the cancer ontology shown in Figure 2-1, the diagnosis Immune_System_Disorder includes two subclasses, Autoimmune_Disease and Immunodeficiency_Syndrome. The Autoimmune_Disease diagnosis includes the subclass Rheumatoid_Arthritis; and the Immunodeficiency_Syndrome diagnosis includes the subclass T_Cell_Immunodeficiency, which includes the subclass AIDS.

The data in the PATIENTS table might include the PATIENT_ID and DIAGNOSIS column values shown in Table 2-1.

Table 2-1 PATIENTS Table Example Data

PATIENT_IDDIAGNOSIS

1234

Rheumatoid_Arthritis

2345

Immunodeficiency_Syndrome

3456

AIDS


To query ontologies, you can use the SEM_MATCH table function (described in Section 1.6) or the SEM_RELATED operator and its ancillary operators (described in Section 2.3).

2.1.2 Supported OWL Subsets

This section describes OWL vocabulary subsets that are supported.

Oracle Database supports the RDFS++, OWLSIF, and OWLPrime vocabularies, which have increasing expressivity, as well as OWL 2 RL. Each supported vocabulary has a corresponding rulebase; however, these rulebases do not need to be populated because the underlying entailment rules of these three vocabularies are internally implemented. The supported vocabularies are as follows:

  • RDFS++: A minimal extension to RDFS; which is RDFS plus owl:sameAs and owl:InverseFunctionalProperty.

  • OWLSIF: OWL with IF Semantic, with the vocabulary and semantics proposed for pD* semantics in Completeness, decidability and complexity of entailment for RDF Schema and a semantic extension involving the OWL vocabulary, by H.J. Horst, Journal of Web Semantics 3, 2 (2005), 79–115.

  • OWLPrime: The following OWL capabilities:

    • Basics: class, subclass, property, subproperty, domain, range, type

    • Property characteristics: transitive, symmetric, functional, inverse functional, inverse

    • Class comparisons: equivalence, disjointness

    • Property comparisons: equivalence

    • Individual comparisons: same, different

    • Class expressions: complement

    • Property restrictions: hasValue, someValuesFrom, allValuesFrom

      As with pD*, the supported semantics for these value restrictions are only intensional (IF semantics).

  • OWL 2 RL: Described in the "OWL 2 RL" section of the W3C OWL 2 Web Ontology Language Profiles recommendation (http://www.w3.org/TR/owl-profiles/#OWL_2_RL) as: "The OWL 2 RL profile is aimed at applications that require scalable reasoning without sacrificing too much expressive power. It is designed to accommodate both OWL 2 applications that can trade the full expressivity of the language for efficiency, and RDF(S) applications that need some added expressivity from OWL 2."

    The system-defined rulebase OWL2RL supports all the standard production rules defined for OWL 2 RL. As with OWLPRIME, users will not see any rules in this OWL2RL rulebase. Note that the rulebase OWL2RL will be created automatically if it does not already exist.

    The following code excerpt uses the OWL2RL rulebase:

    CREATE TABLE m1_tpl (triple SDO_RDF_TRIPLE_S) COMPRESS;
    EXECUTE sem_apis.create_sem_model('m1','m1_tpl','triple');
    -- Insert data into model M1. Details omitted
    ...
    -- Now run inference using the OWL2RL rulebase
    EXECUTE sem_apis.create_entailment('m1_inf',sem_models('m1'),sem_rulebases('owl2rl'));
    

    Note that inference-related optimization, such as parallel inference and RAW8, are all applicable when the OWL2RL rulebase is used.

Table 2-2 lists the RDFS/OWL vocabulary constructs included in each supported rulebase.

Table 2-2 RDFS/OWL Vocabulary Constructs Included in Each Supported Rulebase

Rulebase NameRDFS/OWL Constructs Included

RDFS++

all RDFS vocabulary constructs

owl:InverseFunctionalProperty

owl:sameAs

OWLSIF

all RDFS vocabulary constructs

owl:FunctionalProperty

owl:InverseFunctionalProperty

owl:SymmetricProperty

owl:TransitiveProperty

owl:sameAs

owl:inverseOf

owl:equivalentClass

owl:equivalentProperty

owl:hasValue

owl:someValuesFrom

owl:allValuesFrom

OWLPrime

rdfs:subClassOf

rdfs:subPropertyOf

rdfs:domain

rdfs:range

owl:FunctionalProperty

owl:InverseFunctionalProperty

owl:SymmetricProperty

owl:TransitiveProperty

owl:sameAs

owl:inverseOf

owl:equivalentClass

owl:equivalentProperty

owl:hasValue

owl:someValuesFrom

owl:allValuesFrom

owl:differentFrom

owl:disjointWith

owl:complementOf

OWL2RL

(As described in http://www.w3.org/TR/owl-profiles/#OWL_2_RL)


2.2 Using OWL Inferencing

You can use entailment rules to perform native OWL inferencing. This section creates a simple ontology, performs native inferencing, and illustrates some more advanced features.

2.2.1 Creating a Simple OWL Ontology

Example 2-1 creates a simple OWL ontology, inserts one statement that two URIs refer to the same entity, and performs a query using the SEM_MATCH table function.

Example 2-1 Creating a Simple OWL Ontology

SQL> CREATE TABLE owltst(id number, triple sdo_rdf_triple_s);
Table created.
 
SQL> EXECUTE sem_apis.create_sem_model('owltst','owltst','triple');
PL/SQL procedure successfully completed.
 
SQL> INSERT INTO owltst VALUES (1, sdo_rdf_triple_s('owltst',
       'http://example.com/name/John', 'http://www.w3.org/2002/07/owl#sameAs', 
       'http://example.com/name/JohnQ'));
1 row created.
 
SQL> commit;
 
SQL> -- Use SEM_MATCH to perform a simple query.
SQL> select s,p,o from table(SEM_MATCH('(?s ?p  ?o)', SEM_Models('OWLTST'),
           null,  null, null ));

2.2.2 Performing Native OWL inferencing

Example 2-2 calls the SEM_APIS.CREATE_ENTAILMENT procedure. You do not need to create the rulebase and add rules to it, because the OWL rules are already built into the Oracle semantic technologies inferencing engine.

Example 2-2 Performing Native OWL Inferencing

SQL> -- Invoke the following command to run native OWL inferencing that
SQL> -- understands the vocabulary defined in the preceding section.
SQL>
SQL>  EXECUTE sem_apis.create_entailment('owltst_idx', sem_models('owltst'), sem_rulebases('OWLPRIME'));
PL/SQL procedure successfully completed.
 
SQL> -- The following view is generated to represent the entailed graph (rules index).
SQL> desc mdsys.semi_owltst_idx;
 
SQL> -- Run the preceding query with an additional rulebase parameter to list
SQL> -- the original graph plus the inferred triples.
SQL> SELECT s,p,o FROM table(SEM_MATCH('(?s ?p  ?o)', SEM_MODELS('OWLTST'),
           SEM_RULEBASES('OWLPRIME'),  null, null ));

2.2.3 Performing OWL and User-Defined Rules inferencing

Example 2-3 creates a user-defined rulebase, inserts a simplified uncleOf rule (stating that the brother of one's father is one's uncle), and calls the SEM_APIS.CREATE_ENTAILMENT procedure.

Example 2-3 Performing OWL and User-Defined Rules Inferencing

SQL> -- First, insert the following assertions.
 
SQL> INSERT INTO owltst VALUES (1, sdo_rdf_triple_s('owltst',
       'http://example.com/name/John', 'http://example.com/rel/fatherOf',
       'http://example.com/name/Mary'));
 
 
SQL> INSERT INTO owltst VALUES (1, sdo_rdf_triple_s('owltst',
       'http://example.com/name/Jack', 'http://example.com/rel/brotherOf',
       'http://example.com/name/John'));
 
SQL> -- Create a user-defined rulebase.
 
SQL> EXECUTE sem_apis.create_rulebase('user_rulebase');
 
SQL> -- Insert a simple "uncle" rule.
 
SQL> INSERT INTO mdsys.semr_user_rulebase VALUES ('uncle_rule', 
'(?x <http://example.com/rel/brotherOf> ?y)(?y <http://example.com/rel/fatherOf> ?z)',
NULL, '(?x <http://example.com/rel/uncleOf> ?z)', null);
 
SQL>  -- In the following statement, 'USER_RULES=T' is required, to
SQL> --  include the original graph plus the inferred triples.
SQL> EXECUTE sem_apis.create_entailment('owltst2_idx', sem_models('owltst'),
          sem_rulebases('OWLPRIME','USER_RULEBASE'), 
          SEM_APIS.REACH_CLOSURE, null, 'USER_RULES=T');
 
SQL> -- In the result of the following query, :Jack :uncleOf :Mary is inferred.
SQL> SELECT s,p,o FROM table(SEM_MATCH('(?s ?p  ?o)',
           SEM_MODELS('OWLTST'),
           SEM_RULEBASES('OWLPRIME','USER_RULEBASE'),  null, null ));

2.2.4 Generating OWL inferencing Proofs

OWL inference can be complex, depending on the size of the ontology, the actual vocabulary (set of language constructs) used, and the interactions among those language constructs. To enable you to find out how a triple is derived, you can use proof generation during inference. (Proof generation does require additional CPU time and disk resources.)

To generate the information required for proof, specify PROOF=T in the call to the SEM_APIS.CREATE_ENTAILMENT procedure, as shown in the following example:

EXECUTE sem_apis.create_entailment('owltst_idx', sem_models('owltst'), -
  sem_rulebases('owlprime'), SEM_APIS.REACH_CLOSURE, 'SAM', 'PROOF=T');

Specifying PROOF=T causes a view to be created containing proof for each inferred triple. The view name is the entailment name prefixed by MDSYS.SEMI_. Two relevant columns in this view are LINK_ID and EXPLAIN (the proof). The following example displays the LINK_ID value and proof of each generated triple (with LINK_ID values shortened for simplicity):

SELECT link_id || ' generated by ' || explain as 
          triple_and_its_proof FROM mdsys.semi_owltst_idx;
 
TRIPLE_AND_ITS_PROOF
--------------------------------------------------------------------
8_5_5_4 generated by 4_D_5_5 : SYMM_SAMH_SYMM
8_4_5_4 generated by 8_5_5_4 4_D_5_5 : SAM_SAMH
. . .

A proof consists of one or more triple (link) ID values and the name of the rule that is applied on those triples:

link-id1 [link-id2 ... link-idn] : rule-name

To get the full subject, predicate, and object URIs for proofs, you can query the model view and the entailment (rules index) view. Example 2-4 displays the LINK_ID value and associated triple contents using the model view MDSYS.SEMM_OWLTST and the entailment view MDSYS.SEMI_OWLTST_IDX.

Example 2-4 Displaying Proof Information

SELECT to_char(x.triple.rdf_m_id, 'FMXXXXXXXXXXXXXXXX') ||'_'||
       to_char(x.triple.rdf_s_id, 'FMXXXXXXXXXXXXXXXX') ||'_'||
       to_char(x.triple.rdf_p_id, 'FMXXXXXXXXXXXXXXXX') ||'_'||
       to_char(x.triple.rdf_c_id, 'FMXXXXXXXXXXXXXXXX'),
       x.triple.get_triple()
  FROM (
   SELECT sdo_rdf_triple_s(
           t.canon_end_node_id,
           t.model_id, 
           t.start_node_id,
           t.p_value_id,
           t.end_node_id) triple
     FROM (select * from mdsys.semm_owltst union all
           select * from mdsys.semi_owltst_idx
          ) t  
    WHERE t.link_id IN ('4_D_5_5','8_5_5_4')
  ) x;
 
   LINK_ID  X.TRIPLE.GET_TRIPLE()(SUBJECT, PROPERTY, OBJECT)
----------  --------------------------------------------------------------
4_D_5_5 SDO_RDF_TRIPLE('<http://example.com/name/John>', '<http://www.w3.org/2002/07/owl#sameAs>', '<http://example.com/name/JohnQ>')
8_5_5_4 SDO_RDF_TRIPLE('<http://example.com/name/JohnQ>', '<http://www.w3.org/2002/07/owl#sameAs>', '<http://example.com/name/John>')

In Example 2-4, for the proof entry 8_5_5_4 generated by 4_D_5_5 : SYMM_SAMH_SYMM for the triple with LINK_ID = 8_5_5_4, it is inferred from the triple with 4_D_5_5 using the symmetricity of owl:sameAs.

If the entailment status is INCOMPLETE and if the last entailment was generated without proof information, you cannot invoke SEM_APIS.CREATE_ENTAILMENT with PROOF=T. In this case, you must first drop the entailment and create it again specifying PROOF=T.

2.2.5 Validating OWL Models and Entailments

An OWL ontology may contain errors, such as unsatisfiable classes, instances belonging to unsatisfiable classes, and two individuals asserted to be same and different at the same time. You can use the SEM_APIS.VALIDATE_MODEL and SEM_APIS.VALIDATE_ENTAILMENT functions to detect inconsistencies in the original data model and in the entailment, respectively.

Example 2-5 shows uses the SEM_APIS.VALIDATE_ENTAILMENT function, which returns a null value if no errors are detected or a VARRAY of strings if any errors are detected.

Example 2-5 Validating an Entailment

SQL>  -- Insert an offending triple.
SQL>  insert into owltst values (1, sdo_rdf_triple_s('owltst',
             'urn:C1', 'http://www.w3.org/2000/01/rdf-schema#subClassOf', 'http://www.w3.org/2002/07/owl#Nothing'));
 
SQL> -- Drop entailment first.
SQL>  exec sem_apis.drop_entailment('owltst_idx');
PL/SQL procedure successfully completed.
 
SQL> -- Perform OWL inferencing.
SQL> exec sem_apis.create_entailment('owltst_idx', sem_models('OWLTST'), sem_rulebases('OWLPRIME'));
PL/SQL procedure successfully completed.
 
SQL > set serveroutput on; 
SQL > -- Now invoke validation API: sem_apis.validate_entailment
SQL > 
declare 
  lva mdsys.rdf_longVarcharArray; 
  idx int; 
begin 
  lva := sem_apis.validate_entailment(sem_models('OWLTST'), sem_rulebases('OWLPRIME')) ; 
 
  if (lva is null) then
   dbms_output.put_line('No errors found.');
  else 
    for idx in 1..lva.count loop 
      dbms_output.put_line('Offending entry := ' || lva(idx)) ; 
    end loop ; 
  end if;
end ; 
/ 
 
SQL> -- NOTE: The LINK_ID value and the numbers in the following
SQL> -- line are shortened for simplicity in this example. -- 

          Offending entry  := 1 10001 (4_2_4_8 2 4 8) Unsatisfiable class.

Each item in the validation report array includes the following information:

  • Number of triples that cause this error (1 in Example 2-5)

  • Error code (10001 Example 2-5)

  • One or more triples (shown in parentheses in the output; (4_2_4_8 2 4 8) in Example 2-5).

    These numbers are the LINK_ID value and the ID values of the subject, predicate, and object.

  • Descriptive error message (Unsatisfiable class. in Example 2-5)

The output in Example 2-5 indicates that the error is caused by one triple that asserts that a class is a subclass of an empty class owl:Nothing.

2.2.6 Using SEM_APIS.CREATE_ENTAILMENT for RDFS Inference

In addition to accepting OWL vocabularies, the SEM_APIS.CREATE_ENTAILMENT procedure accepts RDFS rulebases. The following example shows RDFS inference (all standard RDFS rules are defined in http://www.w3.org/TR/rdf-mt/):

EXECUTE sem_apis.create_entailment('rdfstst_idx', sem_models('my_model'), sem_rulebases('RDFS'));

Because rules RDFS4A, RDFS4B, RDFS6, RDFS8, RDFS10, RDFS13 may not generate meaningful inference for your applications, you can deselect those components for faster inference. The following example deselects these rules.

EXECUTE sem_apis.create_entailment('rdfstst_idx', sem_models('my_model'), sem_rulebases('RDFS'), SEM_APIS.REACH_CLOSURE, -
 'RDFS4A-, RDFS4B-, RDFS6-, RDFS8-, RDFS10-, RDFS13-');

2.2.7 Enhancing Inference Performance

This section describes suggestions for improving the performance of inference operations.

  • Collect statistics before inferencing. After you load a large RDF/OWL data model, you should execute the SEM_PERF.GATHER_STATS procedure. See the Usage Notes for that procedure (in Chapter 11) for important usage information.

  • Allocate sufficient temporary tablespace for inference operations. OWL inference support in Oracle relies heavily on table joins, and therefore uses significant temporary tablespace.

See also Section 2.2.8, "Optimizing owl:sameAs Inference".

2.2.8 Optimizing owl:sameAs Inference

You can optimize inference performance for large owl:sameAs cliques by specifying 'OPT_SAMEAS=T' in the options parameter when performing OWLPrime entailment. (A clique is a graph in which every node of it is connected to, bidirectionally, every other node in the same graph.)

According to OWL semantics, the owl:sameAs construct is treated as an equivalence relation, so it is reflexive, symmetric, and transitive. As a result, during inference a full materialization of owl:sameAs-related entailments could significantly increase the size of the inferred graph. Consider the following example triple set:

:John  owl:sameAs  :John1 .
:John  owl:sameAs  :John2 .
:John2 :hasAge     "32" .

Applying OWLPrime inference (with the SAM component specified) to this set would generate the following new triples:

:John1  owl:sameAs  :John .
:John2  owl:sameAs  :John .
:John1  owl:sameAs  :John2 .
:John2  owl:sameAs  :John1 .
:John   owl:sameAs  :John .
:John1  owl:sameAs  :John1 .
:John2  owl:sameAs  :John2 .
:John   :hasAge     "32" . 
:John1  :hasAge     "32" .

In the preceding example, :John, :John1 and :John2 are connected to each other with the owl:sameAs relationship; that is, they are members of an owl:sameAs clique. To provide optimized inference for large owl:sameAs cliques, you can consolidate owl:sameAs triples without sacrificing correctness by specifying 'OPT_SAMEAS=T' in the options parameter when performing OWLPrime entailment. For example:

EXECUTE sem_apis.create_entailment('M_IDX',sem_models('M'),
   sem_rulebases('OWLPRIME'),null,null,'OPT_SAMEAS=T');

When you specify this option, for each owl:sameAs clique, one resource from the clique is chosen as a canonical representative and all of the inferences for that clique are consolidated around that resource. Using the preceding example, if :John1 is the clique representative, after consolidation the inferred graph would contain only the following triples:

:John1 owl:sameAs :John1 .
:John1 :hasAge    "32" .

Some overhead is incurred with owl:sameAs consolidation. During inference, all asserted models are copied into the inference partition, where they are consolidated together with the inferred triples. Additionally, for very large asserted graphs, consolidating and removing duplicate triples incurs a large runtime overhead, so the OPT_SAMEAS=T option is recommended only for ontologies that have a large number of owl:sameAs relationships and large clique sizes.

After the OPT_SAMEAS=T option has been used for an entailment, all subsequent uses of SEM_APIS.CREATE_ENTAILMENT for that entailment must also use OPT_SAMEAS=T, or an error will be reported. To disable optimized sameAs handling, you must first drop the entailment.

Clique membership information is stored in a view named MDSYS.SEMCL_entailment-name, where entailment-name is the name of the entailment (rules index). Each MDSYS.SEMCL_entailment-name view has the columns shown in Table 2-3.

Table 2-3 MDSYS.SEMCL_entailment_name View Columns

Column NameData TypeDescription

MODEL_ID

NUMBER

ID number of the inferred model

VALUE_ID

NUMBER)

ID number of a resource that is a member of the owl:sameAs clique identified by CLIQUE_ID

CLIQUE_ID

NUMBER

ID number of the clique representative for the VALUE_ID resource


To save space, the MDSYS.SEMCL_entailment-name view does not contain reflexive rows like (CLIQUE_ID, CLIQUE_ID).

2.2.8.1 Querying owl:sameAs Consolidated Inference Graphs

At query time, if the entailment queried was created using the OPT_SAMEAS=T option, the results are returned from an owl:sameAs-consolidated inference partition. The query results are not expanded to include the full owl:sameAs closure.

In the following example query, the only result returned would be :John1, which is the canonical clique representative.

SELECT A FROM TABLE (
  SEM_MATCH ('(?A :hasAge "32")',SEM_MODELS('M'),  
    SEM_RULEBASES('OWLPRIME'),NULL, NULL));

With the preceding example, even though :John2 :hasAge "32" occurs in the model, it has been replaced during the inference consolidation phase where redundant triples are removed. However, you can expand the query results by performing a join with the MDSYS.SEMCL_rules-index-name view that contains the consolidated owl:sameAs information. For example, to get expanded result set for the preceding SEM_MATCH query, you can use the following expanded query:

SELECT V.VALUE_NAME A_VAL FROM TABLE (
  SEM_MATCH ('(?A :hasAge "32")',SEM_MODELS('M'), 
    SEM_RULEBASES('OWLPRIME'), NULL, NULL)) Q,
    MDSYS.RDF_VALUE$ V, MDSYS.SEMCL_M_IDX C
  WHERE V.VALUE_ID  = C.VALUE_ID 
     AND C.CLIQUE_ID = Q.A$RDFVID
  UNION ALL
    SELECT A A_VAL FROM TABLE (
      SEM_MATCH ('(?A :hasAge "32")',SEM_MODELS('M'),  
        SEM_RULEBASES('OWLPRIME'),NULL, NULL));

Or, you could rewrite the preceding expanded query using a left outer join, as follows:

SELECT V.VALUE_NAME A_VAL FROM TABLE (
  SEM_MATCH ('(?A <http://hasAge> "33")',SEM_MODELS('M'), 
   SEM_RULEBASES('OWLPRIME'), NULL, NULL)) Q,
  MDSYS.RDF_VALUE$ V, 
   (SELECT value_id, clique_id FROM MDSYS.SEMCL_M_IDX 
  UNION ALL
    SELECT DISTINCT clique_id, clique_id 
      FROM MDSYS.SEMCL_M_IDX) C
 WHERE Q.A$RDFVID  = c.clique_id  (+)
   AND V.VALUE_ID  = nvl(C.VALUE_ID, Q.A$RDFVID);

2.2.9 Performing Incremental Inference

Incremental inference can be used to update entailments (rules indexes) efficiently after triple additions. There are two ways to enable incremental inference for an entailment:

  • Specify the options parameter value INC=T when creating the entailment. For example:

    EXECUTE sem_apis.create_entailment ('M_IDX',sem_models('M'),
      sem_rulebases('OWLPRIME'),null,null, 'INC=T');
    
  • Use the SEM_APIS.ENABLE_INC_INFERENCE procedure.

    If you use this procedure, the entailment must have a VALID status. Before calling the procedure, if you do not own the models involved in the entailment, you must ensure that the respective model owners have used the SEM_APIS.ENABLE_CHANGE_TRACKING procedure to enable change tracking for those models.

When incremental inference is enabled for an entailment, the parameter INC=T must be specified when invoking the SEM_APIS.CREATE_ENTAILMENT procedure for that entailment.

Incremental inference for an entailment depends on triggers for the application tables of the models involved in creating the entailment. This means that incremental inference works only when triples are inserted in the application tables underlying the entailment using conventional path loads, unless you specify the triples by using the delta_in parameter in the call to the SEM_APIS.CREATE_ENTAILMENT procedure, as in the following example, in which the triples from model M_NEW will be added to model M, and entailment M_IDX will be updated with the new inferences:

EXECUTE sem_apis.create_entailment('M_IDX', sem_models('M'),
  sem_rulebases('OWLPRIME''), SEM_APIS.REACH_CLOSURE, null, null,
  sem_models('M_NEW'));
  

Another way to bypass the conventional path loading requirement when using incremental inference is to set the UNDO_RETENTION parameter to cover the intervals between entailments when you perform bulk loading. For example, if the last entailment was created 6 hours ago, the UNDO_RETENTION value should be set to greater than 6 hours; if it is less than that, then (given a heavy workload and limited undo space) it is not guaranteed that all relevant undo information will be preserved for incremental inference to apply. In such cases, the SEM_APIS.CREATE_ENTAILMENT procedure falls back to regular (non-incremental) inference.

To check if change tracking is enabled on a model, use the SEM_APIS.GET_CHANGE_TRACKING_INFO procedure. To get additional information about incremental inference for an entailment, use the SEM_APIS.GET_INC_INF_INFO procedure.

The following restrictions apply to incremental inference:

  • It does not work with optimized owl:sameAs handling (OPT_SAMEAS), user-defined rules, VPD-enabled models, or version-enabled models.

  • It supports only the addition of triples. With updates or deletions, the entailment will be completely rebuilt.

  • It depends on triggers on application tables.

  • Column types (RAW8 or NUMBER) used in incremental inference must be consistent. For instance, if RAW8=T is used to build the entailment initially, then for every subsequent SEM_APIS.CREATE_ENTAILMENT call the same option must be used. To change the column type to NUMBER, you must drop and rebuild the entailment.

2.2.10 Using Parallel Inference

Parallel inference can improve inference performance by taking advantage of the capabilities of a multi-core or multi-CPU architectures. To use parallel inference, specify the DOP (degree of parallelism) keyword and an appropriate value when using the SEM_APIS.CREATE_ENTAILMENT procedure. For example:

EXECUTE sem_apis.create_entailment('M_IDX',sem_models('M'), 
      sem_rulebases('OWLPRIME'), sem_apis.REACH_CLOSURE, null, 'DOP=4');

Specifying the DOP keyword causes parallel execution to be enabled for an Oracle-chosen set of inference components

The success of parallel inference depends heavily on a good hardware configuration of the system on which the database is running. The key is to have a "balanced" system that implements the best practices for database performance tuning and Oracle SQL parallel execution. For example, do not use a single 1 TB disk for an 800 GB database, because executing SQL statements in parallel on a single physical disk can even be slower than executing SQL statements in serial mode. Parallel inference requires ample memory; for each CPU core, you should have at least 4 GB of memory.

Parallel inference is best suited for large ontologies; however, inference performance can also improve for small ontologies.

There is some transient storage overhead associated with using parallel inference. Parallel inference builds a source table that includes all triples based on all the source RDF/OWL models and existing inferred graph. This table might use an additional 10 to 30 percent of storage compared to the space required for storing data and index of the source models.

2.2.11 Using Named Graph Based Inferencing (Global and Local)

The default inferencing in Oracle Database takes all asserted triples from all the source model or models provided and applies semantic rules on top of all the asserted triples until an inference closure is reached. Even if the given source models contain one or more multiple named graphs, it makes no difference because all assertions, whether part of a named graph or not, are treated the same as if they come from a single graph. (For an introduction to named graph support in Oracle Database Semantic Technologies, see Section 1.3.9.)

This default inferencing can be thought of as completely "global" in that it does not consider named graphs at all.

However, if you use named graphs, you can override the default inferencing and have named graphs be considered by using either of the following features:

  • Named graph based global inference (NGGI), which treats all specified named graphs as a unified graph. NGGI lets you narrow the scope of triples to be considered, while enabling great flexibility; it is explained in Section 2.2.11.1.

  • Named graph based local inference (NGLI), which treats each specified named graph as a separate entity. NGLI is explained in Section 2.2.11.2.

For using NGGI and NGLI together, see a recommended usage flow in Section 2.2.11.3.

You specify NGGI or NGLI through certain parameters and options to the SEM_APIS.CREATE_ENTAILMENT procedure when you create an entailment (rules index).

2.2.11.1 Named Graph Based Global Inference (NGGI)

Named graph based global inference (NGGI) enables you to narrow the scope of triples used for inferencing at the named graph level (as opposed to the model level). It also enables great flexibility in selecting the scope; for example, you can include triples from zero or more named graphs and/or from the default graph, and you can include all triples with a null graph name from specified models.

For example, in a hospital application you may only want to apply the inference rules on all the information contained in a set of named graphs describing patients of a particular hospital. If the patient-related named graphs contains only instance-related assertions (ABox), you can specify one or multiple additional schema related-models (TBox), as in Example 2-6.

Example 2-6 Named Graph Based Global Inference

EXECUTE sem_apis.create_entailment(
  'patients_inf',
  models_in         => sem_models('patients','hospital_ontology'),
  rulebases_in      => sem_rulebases('owl2rl'),
  passes            => SEM_APIS.REACH_CLOSURE,
  inf_components_in => null,
  options           => 'DOP=4,RAW8=T',
  include_default_g => sem_models('hospital_ontology'),
  include_named_g   => sem_graphs('<urn:hospital1_patient1>','<urn:hospital1_patient2>'),
  inf_ng_name       => '<urn:inf_graph_for_hospital1>'
  );

In Example 2-6:

  • Two models are involved: patients contains a set of named graphs where each named graph holds triples relevant to a particular patient, and hospital_ontology contains schema information describing concepts and relationships that are defined for hospitals. These two models together are the source models, and they set up an overall scope for the inference.

  • The include_default_g parameter causes all triples with a NULL graph name in the specified models to participate in NGGI. In this example, all triples with a NULL graph name in model hospital_ontology will be included in NGGI.

  • The include_named_g parameter causes all triples from the specified named graphs (across all source models) to participate in NGGI. In this example, triples from named graphs <urn:hospital1_patient1> and <urn:hospital1_patient2> will be included in NGGI.

  • The inf_ng_name parameter assigns graph name <urn:inf_graph_for_hospital1> to all the new triples inferred by NGGI.

2.2.11.2 Named Graph Based Local Inference (NGLI)

Named graph based local inference (NGLI) treats each named graph as a separate entity instead of viewing the graphs as a single unified graph. Inference logic is performed within the boundary of each entity. You can specify schema-related assertions (TBox) in a default graph, and that default graph will participate the inference of each named graph. For example, inferred triples based on a graph with name G1 will be assigned the same graph name G1 in the inferred data partition.

Assertions from any two separate named graphs will never jointly produce any new assertions.

For example, assume the following:

  • Graph G1 includes the following assertion:

    :John  :hasBirthMother  :Mary .
    
  • Graph G2 includes the following assertion:

    :John  :hasBirthMother  :Bella .
    
  • The default graph includes the assertion that :hasBirthMother is an owl:FunctionalProperty. (This assertion has a null graph name.)

In this example, named graph based local inference (NGLI) will not infer that :Mary is owl:sameAs :Bella because the two assertions are from two distinct graphs, G1 and G2. By contrast, a named graph based global inference (NGGI) that includes G1, G2, and the functional property definition would be able to infer that :Mary is owl:sameAs :Bella.

Example 2-7 shows NGLI.

Example 2-7 Named Graph Based Local Inference

EXECUTE sem_apis.create_entailment(
  'patients_inf',
  models_in         => sem_models('patients','hospital_ontology'),
  rulebases_in      => sem_rulebases('owl2rl'),
  passes            => SEM_APIS.REACH_CLOSURE,
  inf_components_in => null,
  options           => 'LOCAL_NG_INF=T'
);

In Example 2-7:

  • The two models patients and hospital_ontology together are the source models, and they set up an overall scope for the inference, similar to the case of global inference in Example 2-6. All triples with a null graph name are treated as part of the common schema (TBox). Inference is performed within the boundary of every single named graph combined with the common schema.

  • Then options parameter keyword-value pair LOCAL_NG_INF=T specifies that named graph based local inference (NGLI) is to be performed.

Note that, by design, NGLI does not apply to the default graph itself. However, you can easily apply named graph based global inference (NGGI) on the default graph and set the inf_ng_name parameter to null. In this way, the TBox inference is precomputed, improving the overall performance and storage consumption.

NGLI does not allow the following:

  • Inferring new relationships based on a mix of triples from multiple named graphs

  • Inferring new relationships using only triples from the default graph.

To get the inference that you would normally expect, you should keep schema assertions and instance assertions separate. Schema assertions (for example, :A rdfs:subClassOf :B and :p1 rdfs:subPropertyOf :p2) should be stored in the default graph as unnamed triples (with null graph names). By contrast, instance assertions (for example, :X :friendOf :Y) should be stored in one of the named graphs.

For a discussion and example of using NGLI to perform document-centric inference with semantically indexed documents, see Section 4.15, "Performing Document-Centric Inference".

NGLI currently does not work together with proof generation, user-defined rules, optimized owl:sameAs handling, or incremental inference.

2.2.11.3 Using NGGI and NGLI Together

The following is a recommended usage flow for using NGGI and NGLI together. It assumes that TBox and ABox are stored in two separate models, that TBox contains schema definitions and all triples in the TBox have a null graph name, but that ABox consists of a set of named graphs describing instance-related data.

  1. Invoke NGGI on the TBox by itself. For example:

    EXECUTE sem_apis.create_entailment(
        'TEST_INF',
        sem_models('abox','tbox'),
        sem_rulebases('owl2rl'),
        SEM_APIS.REACH_CLOSURE,
        include_default_g=>sem_models('tbox')
    );
    
  2. Invoke NGLI for all named graphs. For example:

    EXECUTE sem_apis.create_entailment(
        'TEST_INF',
        sem_models('abox','tbox'),
        sem_rulebases('owl2rl'),
        SEM_APIS.REACH_CLOSURE,
        options => 'LOCAL_NG_INF=T,ENTAIL_ANYWAY=T'
    );
    

    Note that ENTAIL_ANYWAY=T is specified because the NGGI call in step 1will set the status of inferred graph to VALID, and the SEM_APIS.CREATE_ENTAILMENT procedure call in step 2 will quit immediately unless ENTAIL_ANYWAY=T is specified.

2.2.12 Performing Selective Inferencing (Advanced Information)

Selective inferencing is component-based inferencing, in which you limit the inferencing to specific OWL components that you are interested in. To perform selective inferencing, use the inf_components_in parameter to the SEM_APIS.CREATE_ENTAILMENT procedure to specify a comma-delimited list of components. The final inferencing is determined by the union of rulebases specified and the components specified.

Example 2-8 limits the inferencing to the class hierarchy from subclass (SCOH) relationship and the property hierarchy from subproperty (SPOH) relationship. This example creates an empty rulebase and then specifies the two components ('SCOH,SPOH') in the call to the SEM_APIS.CREATE_ENTAILMENT procedure.

Example 2-8 Performing Selective Inferencing

EXECUTE sem_apis.create_rulebase('my_rulebase');
 
EXECUTE sem_apis.create_entailment('owltst_idx', sem_models('owltst'), sem_rulebases('my_rulebase'), SEM_APIS.REACH_CLOSURE, 'SCOH,SPOH');

The following component codes are available: SCOH, COMPH, DISJH, SYMMH, INVH, SPIH, MBRH, SPOH, DOMH, RANH, EQCH, EQPH, FPH, IFPH, DOM, RAN, SCO, DISJ, COMP, INV, SPO, FP, IFP, SYMM, TRANS, DIF, SAM, CHAIN, HASKEY, ONEOF, INTERSECT, INTERSECTSCOH, MBRLST, PROPDISJH, SKOSAXIOMS, SNOMED, SVFH, THINGH, THINGSAM, UNION, RDFP1, RDFP2, RDFP3, RDFP4, RDFP6, RDFP7, RDFP8AX, RDFP8BX, RDFP9, RDFP10, RDFP11, RDFP12A, RDFP12B, RDFP12C, RDFP13A, RDFP13B, RDFP13C, RDFP14A, RDFP14BX, RDFP15, RDFP16, RDFS2, RDFS3, RDFS4a, RDFS4b, RDFS5, RDFS6, RDFS7, RDFS8, RDFS9, RDFS10, RDFS11, RDFS12, RDFS13

The rules corresponding to components with a prefix of RDFP can be found in Completeness, decidability and complexity of entailment for RDF Schema and a semantic extension involving the OWL vocabulary, by H.J. Horst.

The syntax for deselecting a component is component_name followed by a minus (-) sign. For example, the following statement performs OWLPrime inference without calculating the subClassOf hierarchy:

EXECUTE sem_apis.create_entailment('owltst_idx', sem_models('owltst'), sem_rulebases('OWLPRIME'), SEM_APIS.REACH_CLOSURE, 'SCOH-');

By default, the OWLPrime rulebase implements the transitive semantics of owl:sameAs. OWLPrime does not include the following rules (semantics):

U   owl:sameAs   V  .
U     p    X  .        ==>   V  p   X   .
 
U   owl:sameAs   V  .
X     p    U  .        ==>   X   p   V   .

The reason for not including these rules is that they tend to generate many assertions. If you need to include these assertions, you can include the SAM component code in the call to the SEM_APIS.CREATE_ENTAILMENT procedure.

2.3 Using Semantic Operators to Query Relational Data

You can use semantic operators to query relational data in an ontology-assisted manner, based on the semantic relationship between the data in a table column and terms in an ontology. The SEM_RELATED semantic operator retrieves rows based on semantic relatedness. The SEM_DISTANCE semantic operator returns distance measures for the semantic relatedness, so that rows returned by the SEM_RELATED operator can be ordered or restricted using the distance measure. The index type MDSYS.SEM_INDEXTYPE allows efficient execution of such queries, enabling scalable performance over large data sets.

2.3.1 Using the SEM_RELATED Operator

Referring to the cancer ontology example in Section 2.1.1, consider the following query that requires semantic matching: Find all patients whose diagnosis is of the type 'Immune_System_Disorder'. A typical database query of the PATIENTS table (described in Section 2.1.1) involving syntactic match will not return any rows, because no rows have a DIAGNOSIS column containing the exact value Immune_System_Disorder. For example the following query will not return any rows:

SELECT diagnosis FROM patients WHERE diagnosis = 'Immune_System_Disorder';

However, many rows in the patient data table are relevant, because their diagnoses fall under this class. Example 2-9 uses the SEM_RELATED operator (instead of lexical equality) to retrieve all the relevant rows from the patient data table. (In this example, the term Immune_System_Disorder is prefixed with a namespace, and the default assumption is that the values in the table column also have a namespace prefix. However, that might not always be the case, as explained in Section 2.3.5.)

Example 2-9 SEM_RELATED Operator

SELECT diagnosis FROM patients 
  WHERE SEM_RELATED (diagnosis, 
    '<http://www.w3.org/2000/01/rdf-schema#subClassOf>',
    '<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    sem_models('medical_ontology'), sem_rulebases('owlprime')) = 1;

The SEM_RELATED operator has the following attributes:

SEM_RELATED(
  sub  VARCHAR2,
  predExpr  VARCHAR2,
  obj  VARCHAR2,
  ontologyName  SEM_MODELS,
  ruleBases  SEM_RULEBASES,
  index_status  VARCHAR2,
  lower_bound INTEGER,
  upper_bound INTEGER
 ) RETURN INTEGER;

The sub attribute is the name of table column that is being searched. The terms in the table column are typically the subject in a <subject, predicate, object> triple pattern.

The predExpr attribute represents the predicate that can appear as a label of the edge on the path from the subject node to the object node.

The obj attribute represents the term in the ontology for which related terms (related by the predExpr attribute) have to be found in the table (in the column specified by the sub attribute). This term is typically the object in a <subject, predicate, object> triple pattern. (In a query with the equality operator, this would be the query term.)

The ontologyName attribute is the name of the ontology that contains the relationships between terms.

The rulebases attribute identifies one or more rulebases whose rules have been applied to the ontology to infer new relationships. The query will be answered based both on relationships from the ontology and the inferred new relationships when this attribute is specified.

The index_status optional attribute lets you query the data even when the relevant entailment (created when the specified rulebase was applied to the ontology) does not have a valid status. If this attribute is null, the query returns an error if the entailment does not have a valid status. If this attribute is not null, it must be the string VALID, INCOMPLETE, or INVALID, to specify the minimum status of the entailment for the query to succeed. Because OWL does not guarantee monotonicity, the value INCOMPLETE should not be used when an OWL Rulebase is specified.

The lower_bound and upper_bound optional attributes let you specify a bound on the distance measure of the relationship between terms that are related. See Section 2.3.2 for the description of the distance measure.

The SEM_RELATED operator returns 1 if the two input terms are related with respect to the specified predExpr relationship within the ontology, and it returns 0 if the two input terms are not related. If the lower and upper bounds are specified, it returns 1 if the two input terms are related with a distance measure that is greater than or equal to lower_bound and less than or equal to upper_bound.

2.3.2 Using the SEM_DISTANCE Ancillary Operator

The SEM_DISTANCE ancillary operator computes the distance measure for the rows filtered using the SEM_RELATED operator. The SEM_DISTANCE operator has the following format:

SEM_DISTANCE (number) RETURN NUMBER;

The number attribute can be any number, as long as it matches the number that is the last attribute specified in the call to the SEM_RELATED operator (see Example 2-10). The number is used to match the invocation of the ancillary operator SEM_DISTANCE with a specific SEM_RELATED (primary operator) invocation, because a query can have multiple invocations of primary and ancillary operators.

Example 2-10 expands Example 2-9 to show several statements that include the SEM_DISTANCE ancillary operator, which gives a measure of how closely the two terms (here, a patient's diagnosis and the term Immune_System_Disorder) are related by measuring the distance between the terms. Using the cancer ontology described in Section 2.1.1, the distance between AIDS and Immune_System_Disorder is 3.

Example 2-10 SEM_DISTANCE Ancillary Operator

SELECT diagnosis, SEM_DISTANCE(123) FROM patients 
  WHERE SEM_RELATED (diagnosis, 
    '<http://www.w3.org/2000/01/rdf-schema#subClassOf>',
    '<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    sem_models('medical_ontology'), sem_rulebases('owlprime'), 123) = 1;
 
SELECT diagnosis FROM patients 
  WHERE SEM_RELATED (diagnosis,
    '<http://www.w3.org/2000/01/rdf-schema#subClassOf>',
    '<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    sem_models('m,[edical_ontology'), sem_rulebases('owlprime'), 123) = 1
  ORDER BY SEM_DISTANCE(123);
 
SELECT diagnosis, SEM_DISTANCE(123) FROM patients 
  WHERE SEM_RELATED (diagnosis,
    '<http://www.w3.org/2000/01/rdf-schema#subClassOf>',
    '<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    sem_models('medical_ontology'), sem_rulebases('owlprime'), 123) = 1 
  WHERE SEM_DISTANCE(123) <= 3;

Example 2-11 uses distance information to restrict the number of rows returned by the primary operator. All rows with a term related to the object attribute specified in the SEM_RELATED invocation, but with a distance of greater than or equal to 2 and less than or equal to 4, are retrieved.

Example 2-11 Using SEM_DISTANCE to Restrict the Number of Rows Returned

SELECT diagnosis FROM patients 
  WHERE SEM_RELATED (diagnosis,
    '<http://www.w3.org/2000/01/rdf-schema#subClassOf>',
    '<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    sem_models('medical_ontology'), sem_rulebases('owlprime'), 2, 4) = 1;

In Example 2-11, the lower and upper bounds are specified using the lower_bound and upper_bound parameters in the SEM_RELATED operator instead of using the SEM_DISTANCE operator. The SEM_DISTANCE operator can be also be used for restricting the rows returned, as shown in the last SELECT statement in Example 2-10.

2.3.2.1 Computation of Distance Information

Distances are generated for the following properties during inference (entailment): OWL properties defined as transitive properties, and RDFS subClassOf and RDFS subPropertyOf properties. The distance between two terms linked through these properties is computed as the shortest distance between them in a hierarchical class structure. Distances of two terms linked through other properties are undefined and therefore set to null.

Each transitive property link in the original model (viewed as a hierarchical class structure) has a distance of 1, and the distance of an inferred triple is generated according to the number of links between the two terms. Consider the following hypothetical sample scenarios:

  • If the original graph contains C1 rdfs:subClassOf C2 and C2 rdfs:subClassOf C3, then C1 rdfs:subClassof of C3 will be derived. In this case:

    • C1 rdfs:subClassOf C2: distance = 1, because it exists in the model.

    • C2 rdfs:subClassOf C3: distance = 1, because it exists in the model.

    • C1 rdfs:subClassOf C3: distance = 2, because it is generated during inference.

  • If the original graph contains P1 rdfs:subPropertyOf P2 and P2 rdfs:subPropertyOf P3, then P1 rdfs:subPropertyOf P3 will be derived. In this case:

    • P1 rdfs:subPropertyOf P2: distance = 1, because it exists in the model.

    • P2 rdfs:subPropertyOf P3: distance = 1, because it exists in the model.

    • P1 rdfs:subPropertyOf P3: distance = 2, because it is generated during inference.

  • If the original graph contains C1 owl:equivalentClass C2 and C2 owl:equivalentClass C3, then C1 owl:equivalentClass C3 will be derived. In this case:

    • C1 owl:equivalentClass C2: distance = 1, because it exists in the model.

    • C2 owl:equivalentClass C3: distance = 1, because it exists in the model.

    • C1 owl:equivalentClass C3: distance = 2, because it is generated during inference.

The SEM_RELATED operator works with user-defined rulebases. However, using the SEM_DISTANCE operator with a user-defined rulebase is not yet supported, and will raise an error.

2.3.3 Creating a Semantic Index of Type MDSYS.SEM_INDEXTYPE

When using the SEM_RELATED operator, you can create a semantic index of type MDSYS.SEM_INDEXTYPE on the column that contains the ontology terms. Creating such an index will result in more efficient execution of the queries. The CREATE INDEX statement must contain the INDEXTYPE IS MDSYS.SEM_INDEXTYPE clause, to specify the type of index being created.

Example 2-12 creates a semantic index named DIAGNOSIS_SEM_IDX on the DIAGNOSIS column of the PATIENTS table using the Cancer_Ontology ontology.

Example 2-12 Creating a Semantic Index

CREATE INDEX diagnosis_sem_idx
  ON patients (diagnosis) 
  INDEXTYPE IS MDSYS.SEM_INDEXTYPE;

The column on which the index is built (DIAGNOSIS in Example 2-12) must be the first parameter to the SEM_RELATED operator, in order for the index to be used. If it not the first parameter, the index is not used during the execution of the query.

To improve the performance of certain semantic queries, you can cause statistical information to be generated for the semantic index by specifying one or more models and rulebases when you create the index. Example 2-13 creates an index that will also generate statistics information for the specified model and rulebase. The index can be used with other models and rulebases during query, but the statistical information will be used only if the model and rulebase specified during the creation of the index are the same model and rulebase specified in the query.

Example 2-13 Creating a Semantic Index Specifying a Model and Rulebase

CREATE INDEX diagnosis_sem_idx
  ON patients (diagnosis) 
  INDEXTYPE IS MDSYS.SEM_INDEXTYPE('ONTOLOGY_MODEL(medical_ontology), 
    RULEBASE(OWLPrime)');

The statistical information is useful for queries that return top-k results sorted by semantic distance. Example 2-14 shows such a query.

Example 2-14 Query Benefitting from Generation of Statistical Information

SELECT /*+ FIRST_ROWS */ diagnosis FROM patients 
  WHERE SEM_RELATED (diagnosis,
    '<http://www.w3.org/2000/01/rdf-schema#subClassOf>',
    '<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    sem_models('medical_ontology'), sem_rulebases('owlprime'), 123) = 1
  ORDER BY SEM_DISTANCE(123);

2.3.4 Using SEM_RELATED and SEM_DISTANCE When the Indexed Column Is Not the First Parameter

If an index of type MDSYS.SEM_INDEXTYPE has been created on a table column that is the first parameter to the SEM_RELATED operator, the index will be used. For example, the following query retrieves all rows that have a value in the DIAGNOSIS column that is a subclass of (rdfs:subClassOf) Immune_System_Disorder.

SELECT diagnosis FROM patients 
  WHERE SEM_RELATED (diagnosis, 
    '<http://www.w3.org/2000/01/rdf-schema#subClassOf>',
    '<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    sem_models('medical_ontology'), sem_rulebases('owlprime')) = 1;

Assume, however, that this query instead needs to retrieve all rows that have a value in the DIAGNOSIS column for which Immune_System_Disorder is a subclass. You could rewrite the query as follows:

SELECT diagnosis FROM patients 
  WHERE SEM_RELATED
    ('<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    '<http://www.w3.org/2000/01/rdf-schema#subClassOf>', 
    diagnosis, 
    sem_models('medical_ontology'), sem_rulebases('owlprime')) = 1;

However, in this case a semantic index on the DIAGNOSIS column will not be used, because it is not the first parameter to the SEM_RELATED operator. To cause the index to be used, you can change the preceding query to use the inverseOf keyword, as follows:

SELECT diagnosis FROM patients 
  WHERE SEM_RELATED (diagnosis,
    'inverseOf(http://www.w3.org/2000/01/rdf-schema#subClassOf)',
    '<http://www.example.org/medical_terms/Immune_System_Disorder>', 
    sem_models('medical_ontology'), sem_rulebases('owlprime')) = 1;

This form causes the table column (on which the index is built) to be the first parameter to the SEM_RELATED operator, and it retrieves all rows that have a value in the DIAGNOSIS column for which Immune_System_Disorder is a subclass.

2.3.5 Using URIPREFIX When Values Are Not Stored as URIs

By default, the semantic operator support assumes that the values stored in the table are URIs. These URIs can be from different namespaces. However, if the values in the table do not have URIs, you can use the URIPREFIX keyword to specify a URI when you create the semantic index. In this case, the specified URI is prefixed to the value in the table and stored in the index structure. (One implication is that multiple URIs cannot be used).

Example 2-15 creates a semantic index that uses a URI prefix.

Example 2-15 Specifying a URI Prefix During Semantic Index Creation

CREATE INDEX diagnosis_sem_idx
  ON patients (diagnosis) 
  INDEXTYPE IS MDSYS.SEM_INDEXTYPE
  PARAMETERS('URIPREFIX(<http://www.example.org/medical/>)');

Note that the slash (/) character at the end of the URI is important, because the URI is prefixed to the table value (in the index structure) without any parsing.

PK۳,,PK.AOEBPS/sem_sesame.htm Sesame Adapter for Oracle Database

8 Sesame Adapter for Oracle Database

The Sesame Adapter for Oracle Database (referred to here as the Sesame Adapter) integrates the popular Sesame Java APIs with Oracle Semantic Technologies support. Sesame, created by OpenRDF (http://www.openrdf.org/), "is an open source framework for storage, inferencing and querying of RDF data."

This chapter assumes that you are familiar with major concepts explained in Chapter 1, "Oracle Database Semantic Technologies Overview" and Chapter 2, "OWL Concepts". It also assumes that you are familiar with the overall capabilities and use of the Sesame Java framework. For information about the Sesame framework, see the Sesame Documentation page at http://www.openrdf.org/documentation.jsp.

The Sesame Adapter extends the semantic data management capabilities of Oracle Database Release 11.2 RDF/OWL.

This chapter includes the following major topics:

8.1 Sesame Adapter Overview

Like the Jena Adapter (described in Chapter 7), the Sesame Adapter provides a Java-based interface to Oracle semantic data through an open source framework and related tools. However, the Sesame Adapter adds support for context to the data:

  • Jena statements deal with triples (subject, predicate, object).

  • Sesame statements deal with triples and contexts; a Sesame statement is a quad consisting of a triple and a context. For example, with hospital patient data, the context might contain the patient's URI.

The Sesame Adapter provides a Java API for interacting with semantic data stored in Oracle Database. It also provides integration with the following Sesame tools:

  • Sesame Server, which provides an HTTP SPARQL endpoint.

  • Sesame Console, a command-line tool for creating and managing repositories. The available commands include the following:

    connect  Connects to a (local or remote) set of repositories
    disconnect  Disconnects from the current set of repositories
    create  Creates a new repository
    drop  Drops a repository
    open  Opens a repository to work on, takes a repository ID as argument
    close  Closes the current repository
    show  Displays an overview of various resources
    load  Loads a data file into a repository, takes a file path or URL as argument
    verify  Verifies the syntax of an RDF data file, takes a file path or URL as argument
    clear  Removes data from a repository
    serql  Evaluates the SeRQL query, takes a query as argument
    sparql  Evaluates the SPARQL query, takes a query as argument
    set  Allows various console parameters to be set
    exit, quit  Exits the console
    
  • OpenRDF Workbench, a graphical user interface to the Sesame Server.

The features provided by the Sesame Adapter include:

  • Loading (bulk and incremental), exporting, and removing statements, with and without context

  • Querying of data, with and without context

  • Inferencing, where inferred data is considered during query execution (if includeInferred=true), and where inferred triples have a null context

  • Managing namespaces

The Sesame Adapter for Oracle Database implements various interfaces of the Sesame Storage and Inference Layer (SAIL) API. For example, the class OracleSailConnection is an Oracle implementation of the Sesame SailConnection interface, and the class OracleSailStore is an Oracle implementation of the Sesame Sail interface.

A typical usage flow for the Sesame Adapter might include the excerpt shown in Example 8-1.

Example 8-1 Sample Usage flow for Sesame Adapter

String query =
" PREFIX foaf: <http://xmlns.com/foaf/0.1/> "
" PREFIX dc: <http://purl.org/dc/elements/1.1/> "
" select ?s ?p ?o ?name WHERE {?s ?p ?o . OPTIONAL {?o foaf:name ?name .} } ";

SailRepository sr = new SailRepository(
   new OracleSailStore(oracle, modelName,…attachmentForInference));
RepositoryConnection repConn = sr.getConnection();
// Data loading can happen here.
repConn.add(alice, friendOf, bob, context1);
 
TupleQuery tq = repConn.prepareTupleQuery(QueryLanguage.SPARQL, query);
TupleQueryResult tqr = tq.evaluate();
while(tqr.hasNext()) {
    psOut.print((tqr.next().toString()));
};
tqr.close()
repConn.close();
sr.shutDown();

8.2 Setup and Configuration for the Sesame Adapter

Setting up and configuring the Sesame Adapter for Oracle Database involves the following actions:

  1. Setting Up the Software Environment

  2. Setting Up the SPARQL Service

  3. Setting Up the Semantic Technologies Environment

8.2.1 Setting Up the Software Environment

To use the Sesame Adapter, you must first ensure that the system environment has the necessary software, including Oracle Database 11g Release 2 with the Spatial and Partitioning options and with Semantic Technologies support enabled, Sesame version 2.3.1, the Sesame Adapter, and JDK 6. You can set up the software environment by performing these actions:

  1. Install Oracle Database Release 11.2 Enterprise Edition with the Oracle Spatial and Partitioning Options.

  2. Enable the support for Semantic Technologies, as explained in Section A.1.

  3. Download the Sesame 2.3.1 onejar library and include it in your classpath, as explained in the "Setting up to use the Sesame libraries" chapter of the User Guide for Sesame 2.2. (See http://www.openrdf.org/doc/sesame2/users/ch04.html, or a more recent version if available.)

  4. Install the Sesame 2.3.1 Server, as explained in the "Server software installation" chapter of the User Guide for Sesame 2.2. (See http://www.openrdf.org/doc/sesame2/users/ch06.html, or a more recent version if available.)

  5. Download the Sesame Adapter (sesame_adapter_for_release11.2.zip) from the Oracle Database Semantic Technologies page (http://www.oracle.com/technetwork/database/options/semantic-tech/and click Downloads), and unzip it into a temporary directory, such as (on a Linux system) /tmp/sesame_adapter. (If this temporary directory does not already exist, create it before the unzip operation.)

    The Sesame Adapter directories and files have the following structure:

     examples/
       examples/Example1.java
       examples/Example2.java
       .
       .
       .
    lib/
       lib/sdordfsesame.jar
    web/
       web/web.xml
    javadoc/
       javadoc/javadoc.zip
    
  6. Copy ojdbc6.jar into <Sesame_DIR>/lib (Linux) or <Sesame_DIR>\lib (Windows). (ojdbc6.jar is in $ORACLE_HOME/jdbc/lib or %ORACLE_HOME%\jdbc\lib.)

  7. Copy sdordf.jar into <Sesame_DIR>/lib (Linux) or <Sesame_DIR>\lib (Windows). (sdordf.jar is in $ORACLE_HOME/md/jlib or %ORACLE_HOME%\md\jlib.)

  8. If JDK 6 is not already installed, install it.

  9. If the JAVA_HOME environment variable does not already refer to the JDK 6 installation, define it accordingly. For example:

    setenv JAVA_HOME /usr/local/packages/jdk16/
    
  10. If the SPARQL service to support the SPARQL protocol is not set up, set it up as explained in Section 8.2.2.

After setting up the software environment, ensure that your Semantic Technologies environment can enable you to use the Sesame Adapter to perform queries, as explained in Section 8.2.3.

8.2.2 Setting Up the SPARQL Service

Setting up a SPARQL endpoint using the Sesame Adapter involves downloading the Sesame Server, with which client applications can communicate over HTTP protocol. The Sesame Server (and OpenRDF Workbench) is bundled with the Sesame 2.3.1 release package. This section explains how to set up a SPARQL service using a servlet deployed in WebLogic Server. The number and complexity of the steps reflect the fact that Oracle is not permitted to bundle all the dependent third-party libraries in a .war or .ear file.

  1. Download and Install Oracle WebLogic Server 11g Release 1 (10.3.1). For details, see http://www.oracle.com/technology/products/weblogic/.

  2. Ensure that you have Java 6 installed, because it is required by Sesame.

  3. Download the Sesame 2.3.1 release package (in .zip format) from http://sourceforge.net/projects/sesame/files/Sesame%202/.

  4. Unpack the .zip file into a temporary directory. For example:

    mkdir /tmp/sesame
    cp openrdf-sesame-2.3.1-sdk.zip /tmp/sesame/
    cd /tmp/sesame
    unzip openrdf-sesame-2.3.1-sdk.zip
    
  5. Ensure that you have downloaded and unzipped the Sesame Adapter for Oracle Database, as explained in Section 8.2.1.

  6. Create a directory named openrdf-sesame.war at the same level as the Sesame_adapter directory, and go to it. For example:

    /tmp/openrdf-sesame.war
    cd /tmp/openrdf-sesame.war
    
  7. Unzip necessary files into the directory created in the preceding step:

    cd /tmp/openrdf-sesame.war
    unzip /tmp/sesame/openrdf-sesame-2.3.1/war/openrdf-sesame.war
    
  8. Copy necessary files, as follows. (Assume that ORACLE_HOME points to the home directory of a Release 11.2 Oracle Database.)

    cp /tmp/Sesame_adapter/web/web.xml /tmp/openrdf-sesame.war/WEB-INF/web.xml 
    cp /tmp/Sesame_adapter/lib/*.jar   /tmp/openrdf-sesame.war/WEB-INF/lib
    cp $ORACLE_HOME/md/jlib/sdordf.jar /tmp/openrdf-sesame.war/WEB-INF/lib
    cp $ORACLE_HOME/jdbc/lib/ojdbc6.jar /tmp/openrdf-sesame.war/WEB-INF/lib
    
  9. Using the WebLogic Server Administration console, create a J2EE data source named OracleSemDS. During the data source creation, you can specify a user and password for the database schema that contains the relevant semantic data against which SPARQL queries are to be executed.

    If you need help in creating this data source, see Section 8.2.2.1, "Creating the Required Data Source Using WebLogic Server".

  10. Go to the autodeploy directory of WebLogic Server and copy files, as follows. (For information about auto-deploying applications in development domains, see: http://download.oracle.com/docs/cd/E11035_01/wls100/deployment/autodeploy.html)

    cd <domain_name>/autodeploy
    cp -rf  /tmp/openrdf-sesame.war  <domain_name>/autodeploy
    

    In the preceding example, <domain_name> is the name of a WebLogic Server domain.

    Note that while you can run a WebLogic Server domain in two different modes, development and production, only development mode allows you use the auto-deployment feature.

  11. Check the files and the directory structure, as in the following example:

    autodeploy/% ls -1R ./openrdf-sesame.war/
     
    openrdf-sesame.war/:
    favicon.ico
    favicon.png
    images
    META-INF
    styles
    WEB-INF
     
    openrdf-sesame.war/images:
    DEBUG.png
    ERROR.png
    INFO.png
    logo.png
    productgroup.png
    transparent.png
    WARN.png
     
    openrdf-sesame.war/META-INF:
    MANIFEST.MF
    maven
     
    openrdf-sesame.war/META-INF/maven:
    info.aduna.appbase
    org.openrdf.sesame
     
    openrdf-sesame.war/META-INF/maven/info.aduna.appbase:
    aduna-appbase-webapp-system
     
    openrdf-sesame.war/META-INF/maven/info.aduna.appbase/aduna-appbase-webapp-system:
    pom.properties
    pom.xml
     
    openrdf-sesame.war/META-INF/maven/org.openrdf.sesame:
    sesame-http-server
     
    openrdf-sesame.war/META-INF/maven/org.openrdf.sesame/sesame-http-server:
    pom.properties
    pom.xml
     
    openrdf-sesame.war/styles:
    basic
    default
    msie-png-alpha.css
    msie-png-alpha.png
    w3-html40-recommended.css
     
    openrdf-sesame.war/styles/basic:
    all.css
     
    openrdf-sesame.war/styles/default:
    images
    msie-minheight.css
    print.css
    screen.css
     
    openrdf-sesame.war/styles/default/images:
    bg_body.png
    bg_header.png
    bg_html.png
    hr_menu.png
     
    openrdf-sesame.war/WEB-INF:
    aduna-webapp-servlet.xml
    aduna-webapp-system-servlet.xml
    classes
    includes
    lib
    openrdf-http-server-servlet.xml
    sample-web.xml
    taglibs
    tags
    urlrewrite.xml
    views
    web.xml
     
    openrdf-sesame.war/WEB-INF/classes:
    logback.xml
     
    openrdf-sesame.war/WEB-INF/includes:
    components
    stylesheets.html.jspf
    taglibs.jspf
     
    openrdf-sesame.war/WEB-INF/includes/components:
    bodyStart.html.jspf
    bodyStop.html.jspf
    ContentHeader.html.jspf
    Footer.html.jspf
    Header.html.jspf
    head.html.jspf
    htmlStart.html.jspf
    htmlStop.html.jspf
    logfilterform.html.jspf
    logpaginationfooter.html.jspf
    logpaginationheader.html.jspf
    Message.html.jspf
    Navigation.html.jspf
    page.html.jspf
    tabs.html.jspf
     
    openrdf-sesame.war/WEB-INF/lib:
    activation-1.1.jar
    aduna-appbase-core-3.9.0.jar
    aduna-appbase-logging-api-3.9.0.jar
    aduna-appbase-logging-file-3.9.0.jar
    aduna-appbase-webapp-base-core-3.9.0.jar
    aduna-appbase-webapp-system-core-3.9.0.jar
    aduna-commons-collections-2.7.0.jar
    aduna-commons-concurrent-2.6.0.jar
    aduna-commons-i18n-1.3.0.jar
    aduna-commons-io-2.8.0.jar
    aduna-commons-iteration-2.7.0.jar
    aduna-commons-lang-2.8.0.jar
    aduna-commons-net-2.6.0.jar
    aduna-commons-platform-info-2.8.0.jar
    aduna-commons-text-2.6.0.jar
    aduna-commons-webapp-core-2.8.0.jar
    aduna-commons-xml-2.6.0.jar
    aopalliance-1.0.jar
    asm-1.5.3.jar
    cglib-2.1_3.jar
    commons-codec-1.3.jar
    commons-dbcp-1.2.2.jar
    commons-httpclient-3.1.jar
    commons-pool-1.3.jar
    jcl-over-slf4j-1.5.10.jar
    jstl-1.1.2.jar
    logback-classic-0.9.18.jar
    logback-core-0.9.18.jar
    ojdbc6.jar
    sdordf.jar
    sdordfsesame.jar
    sesame-http-client-2.3.1.jar
    sesame-http-protocol-2.3.1.jar
    sesame-http-server-2.3.1.jar
    sesame-http-server-spring-2.3.1.jar
    sesame-model-2.3.1.jar
    sesame-query-2.3.1.jar
    sesame-queryalgebra-evaluation-2.3.1.jar
    sesame-queryalgebra-model-2.3.1.jar
    sesame-queryparser-api-2.3.1.jar
    sesame-queryparser-serql-2.3.1.jar
    sesame-queryparser-sparql-2.3.1.jar
    sesame-queryresultio-api-2.3.1.jar
    sesame-queryresultio-binary-2.3.1.jar
    sesame-queryresultio-sparqljson-2.3.1.jar
    sesame-queryresultio-sparqlxml-2.3.1.jar
    sesame-queryresultio-text-2.3.1.jar
    sesame-repository-api-2.3.1.jar
    sesame-repository-contextaware-2.3.1.jar
    sesame-repository-dataset-2.3.1.jar
    sesame-repository-event-2.3.1.jar
    sesame-repository-http-2.3.1.jar
    sesame-repository-manager-2.3.1.jar
    sesame-repository-sail-2.3.1.jar
    sesame-rio-api-2.3.1.jar
    sesame-rio-n3-2.3.1.jar
    sesame-rio-ntriples-2.3.1.jar
    sesame-rio-rdfxml-2.3.1.jar
    sesame-rio-trig-2.3.1.jar
    sesame-rio-trix-2.3.1.jar
    sesame-rio-turtle-2.3.1.jar
    sesame-runtime-2.3.1.jar
    sesame-sail-api-2.3.1.jar
    sesame-sail-inferencer-2.3.1.jar
    sesame-sail-memory-2.3.1.jar
    sesame-sail-nativerdf-2.3.1.jar
    sesame-sail-rdbms-2.3.1.jar
    slf4j-api-1.5.10.jar
    spring-aop-2.5.6.jar
    spring-beans-2.5.6.jar
    spring-context-2.5.6.jar
    spring-context-support-2.5.6.jar
    spring-core-2.5.6.jar
    spring-web-2.5.6.jar
    spring-webmvc-2.5.6.jar
    standard-1.1.2.jar
    urlrewritefilter-3.0.4.jar
     
    openrdf-sesame.war/WEB-INF/taglibs:
    navigation.tld
     
    openrdf-sesame.war/WEB-INF/tags:
    errors.tag
     
    openrdf-sesame.war/WEB-INF/views:
    home
    overview.jsp
    system
     
    openrdf-sesame.war/WEB-INF/views/home:
    overview.jsp
     
    openrdf-sesame.war/WEB-INF/views/system:
    info
    logging
    overview.jsp
     
    openrdf-sesame.war/WEB-INF/views/system/info:
    debug.jsp
    overview.jsp
     
    openrdf-sesame.war/WEB-INF/views/system/logging:
    overview.jsp
    
  12. Start or restart WebLogic Server.

  13. Verify your deployment by using your Web browser to connect to a URL in the following format (assume that the Web application is deployed at port 7001): http://<hostname>:7001/openrdf-sesame/querymgt?abortqid=0

    You should see an XML response, which indicates a successful deployment of the Sesame Server with the Sesame Adapter query management servlet.

You can also install the OpenRDF Workbench, which is a graphical user interface (GUI) for communicating with the Sesame Server. To deploy the OpenRDF Workbench into the Oracle WebLogic Server, you must have performed at least the first 7 steps in the preceding instructions for setting up the SPARQL endpoint (explained earlier in this section); then, follow these steps:

  1. Go to the autodeploy directory of WebLogic Server and copy files, as follows. (For information about auto-deploying applications in development domains, see: http://download.oracle.com/docs/cd/E11035_01/wls100/deployment/autodeploy.html)

    cd <domain_name>/autodeploy
    cp -rf  /tmp/openrdf-workbench.war  <domain_name>/autodeploy
    

    In the preceding example, <domain_name> is the name of a WebLogic Server domain.

  2. Check the files and the directory structure, as in the following example:

    autodeploy/% ls -1R ./openrdf-workbench.war/
     openrdf-workbench.war/:
    
    favicon.ico
    favicon.png
    images
    locale
    META-INF
    scripts
    styles
    transformations
    WEB-INF
     
    openrdf-workbench.war/images:
    affirmative.png
    edit.png
    logo.png
    negative.png
    productgroup.png
    server_network.png
    server.png
    view.png
     
    openrdf-workbench.war/locale:
    messages.xsl
     
    openrdf-workbench.war/META-INF:
    MANIFEST.MF
    maven
     
    openrdf-workbench.war/META-INF/maven:
    org.openrdf.sesame
     
    openrdf-workbench.war/META-INF/maven/org.openrdf.sesame:
    sesame-http-workbench
     
    openrdf-workbench.war/META-INF/maven/org.openrdf.sesame/sesame-http-workbench:
    pom.properties
    pom.xml
     
    openrdf-workbench.war/scripts:
    cookies.html
     
    openrdf-workbench.war/styles:
    basic
    default
    msie-png-alpha.css
    msie-png-alpha.png
    w3-html40-recommended.css
     
    openrdf-workbench.war/styles/basic:
    all.css
     
    openrdf-workbench.war/styles/default:
    images
    msie-minheight.css
    print.css
    screen.css
     
    openrdf-workbench.war/styles/default/images:
    bg_body.png
    bg_header.png
    bg_html.png
    hr_menu.png
     
    openrdf-workbench.war/transformations:
    add.xsl
    boolean.xsl
    clear.xsl
    contexts.xsl
    create-memory-rdfs-dt.xsl
    create-memory-rdfs.xsl
    create-memory.xsl
    create-mysql.xsl
    create-native-rdfs-dt.xsl
    create-native-rdfs.xsl
    create-native.xsl
    create-pgsql.xsl
    create-remote.xsl
    create.xsl
    delete.xsl
    explore.xsl
    export.xsl
    graph.xsl
    information.xsl
    list.xsl
    namespaces.xsl
    query.xsl
    remove.xsl
    repositories.xsl
    server.xsl
    summary.xsl
    table.xsl
    template.xsl
    tuple.xsl
    types.xsl
    url-encode.xsl
     
    openrdf-workbench.war/WEB-INF:
    classes
    lib
    web.xml
     
    openrdf-workbench.war/WEB-INF/classes:
    org
     
    openrdf-workbench.war/WEB-INF/classes/org:
    openrdf
     
    openrdf-workbench.war/WEB-INF/classes/org/openrdf:
    workbench
     
    openrdf-workbench.war/WEB-INF/classes/org/openrdf/workbench:
    base
    commands
    exceptions
    proxy
    RepositoryServlet.class
    util
     
    openrdf-workbench.war/WEB-INF/classes/org/openrdf/workbench/base:
    BaseRepositoryServlet.class
    BaseServlet.class
    TransformationServlet.class
    TupleServlet.class
     
    openrdf-workbench.war/WEB-INF/classes/org/openrdf/workbench/commands:
    AddServlet.class
    ClearServlet.class
    ContextsServlet.class
    CreateServlet.class
    DeleteServlet.class
    ExploreServlet.class
    ExportServlet.class
    InformationServlet.class
    InfoServlet.class
    NamespacesServlet.class
    QueryServlet.class
    RemoveServlet.class
    RepositoriesServlet.class
    SummaryServlet.class
    TypesServlet.class
     
    openrdf-workbench.war/WEB-INF/classes/org/openrdf/workbench/exceptions:
    BadRequestException.class
    MissingInitParameterException.class
     
    openrdf-workbench.war/WEB-INF/classes/org/openrdf/workbench/proxy:
    CookieCacheControlFilter$CacheAwareRequest.class
    CookieCacheControlFilter$CacheAwareResponse.class
    CookieCacheControlFilter.class
    ProxyRepositoryServlet.class
    RedirectFilter.class
    WorkbenchGateway.class
    WorkbenchServlet.class
     
    openrdf-workbench.war/WEB-INF/classes/org/openrdf/workbench/util:
    BasicServletConfig.class
    ConfigTemplate.class
    DynamicHttpRequest.class
    TupleResultBuilder.class
    WorkbenchRequest.class
     
    openrdf-workbench.war/WEB-INF/lib:
    aduna-appbase-core-3.9.0.jar
    aduna-appbase-logging-api-3.9.0.jar
    aduna-appbase-logging-file-3.9.0.jar
    aduna-commons-collections-2.7.0.jar
    aduna-commons-concurrent-2.6.0.jar
    aduna-commons-i18n-1.3.0.jar
    aduna-commons-io-2.8.0.jar
    aduna-commons-iteration-2.7.0.jar
    aduna-commons-lang-2.8.0.jar
    aduna-commons-net-2.6.0.jar
    aduna-commons-platform-info-2.8.0.jar
    aduna-commons-text-2.6.0.jar
    aduna-commons-xml-2.6.0.jar
    commons-cli-1.1.jar
    commons-codec-1.3.jar
    commons-dbcp-1.2.2.jar
    commons-fileupload-1.2.1.jar
    commons-httpclient-3.1.jar
    commons-io-1.3.2.jar
    commons-pool-1.3.jar
    jcl-over-slf4j-1.5.10.jar
    sesame-console-2.3.1.jar
    sesame-http-client-2.3.1.jar
    sesame-http-protocol-2.3.1.jar
    sesame-model-2.3.1.jar
    sesame-query-2.3.1.jar
    sesame-queryalgebra-evaluation-2.3.1.jar
    sesame-queryalgebra-model-2.3.1.jar
    sesame-queryparser-api-2.3.1.jar
    sesame-queryparser-serql-2.3.1.jar
    sesame-queryparser-sparql-2.3.1.jar
    sesame-queryresultio-api-2.3.1.jar
    sesame-queryresultio-binary-2.3.1.jar
    sesame-queryresultio-sparqljson-2.3.1.jar
    sesame-queryresultio-sparqlxml-2.3.1.jar
    sesame-queryresultio-text-2.3.1.jar
    sesame-repository-api-2.3.1.jar
    sesame-repository-contextaware-2.3.1.jar
    sesame-repository-dataset-2.3.1.jar
    sesame-repository-event-2.3.1.jar
    sesame-repository-http-2.3.1.jar
    sesame-repository-manager-2.3.1.jar
    sesame-repository-sail-2.3.1.jar
    sesame-rio-api-2.3.1.jar
    sesame-rio-n3-2.3.1.jar
    sesame-rio-ntriples-2.3.1.jar
    sesame-rio-rdfxml-2.3.1.jar
    sesame-rio-trig-2.3.1.jar
    sesame-rio-trix-2.3.1.jar
    sesame-rio-turtle-2.3.1.jar
    sesame-runtime-2.3.1.jar
    sesame-sail-api-2.3.1.jar
    sesame-sail-inferencer-2.3.1.jar
    sesame-sail-memory-2.3.1.jar
    sesame-sail-nativerdf-2.3.1.jar
    sesame-sail-rdbms-2.3.1.jar
    slf4j-api-1.5.10.jar
    slf4j-jdk14-1.5.10.jar
    
  3. Start or restart WebLogic Server.

  4. Verify your deployment by using your Web browser to connect to a URL in the following format (assume that the Web application is deployed at port 7001): http://<hostname>:7001/openrdf-workbench

    You should see a web page with the OpenRDF Workbench logo. Click the Repositories link on the left to see a list of repositories.

8.2.2.1 Creating the Required Data Source Using WebLogic Server

If you need help creating the required J2EE data source using the WebLogic Server admin console, you can follow these steps:

  1. Login to: http://<hostname>:7001/console

  2. In the Domain Structure panel, click Services.

  3. Click JDBC

  4. Click Data Sources.

  5. In the Summary of JDBC Data Sources panel, click New under the Data Sources table.

  6. In the Create a New JDBC Data Source panel, enter or select the following values.

    Name: OracleSemDS

    JNDI Name: OracleSemDS

    Database Type: Oracle

    Database Driver: Oracle's Driver (Thin) Versions: 9.0.1,9.2.0,10,11

  7. Click Next twice.

  8. In the Connection Properties panel, enter the appropriate values for the Database Name, Host Name, Port, Database User Name (schema that contains semantic data), Password fields.

  9. Click Next.

  10. Select (check) the target server or servers to which you want to deploy this OracleSemDS data source.

  11. Click Finish.

    You should see a message that all changes have been activated and no restart is necessary.

8.2.2.2 Configuring the SPARQL Service

You can use the OracleSailStore API and other relevant APIs to create semantic models, load data (incrementally or in bulk), and perform logical inference.

After the semantic models are in the Oracle database, you can use the Sesame Console (described in Section 8.9) to connect to the Sesame Server and create an Oracle semantic model-backed repository. For example, you might create a repository with the repository ID myOracleRepos that uses a semantic model named LUBM and the OWLPRIME rulebase.

This newly created repository provides a new service endpoint on the Sesame Server.

8.2.2.3 Terminating Long-Running SPARQL Queries

Because some applications need to be able to terminate long-running SPARQL queries, an abort framework has been introduced with the Sesame Adapter and the Sesame Server setup. Basically, for queries that may take a long time to run, you must stamp each with a unique query ID (qid) value.

For example, the following SPARQL query selects out the subject of all triples. A query ID (qid) is set so that this query can be terminated upon request.

PREFIX ORACLE_SEM_FS_NS:  <http://example.com/semtech#qid=8761>
SELECT ?subject WHERE {?subject ?property ?object }

The qid attribute value is of long integer type. You can choose a value for the qid for a particular query based on your own application needs.

To terminate a SPARQL query that has been submitted with a qid value, applications can send an abort request to a servlet in the following format and specify a matching QID value

http://<hostname>:7001/openrdf-sesame/querymgt?abortqid=8761

8.2.3 Setting Up the Semantic Technologies Environment

To use the Sesame Adapter to perform queries, you can connect as any user (with suitable privileges) and use any models in the semantic network. If your Semantic Technologies environment already meets the requirements, you can go directly to compiling and running Java code that uses the Sesame Adapter. If your Semantic Technologies environment is not yet set up to be able to use the Sesame Adapter, you can perform actions similar to the following example steps:

  1. Connect as SYS with the SYSDBA role:

    sqlplus sys/<password-for-sys> as sysdba
    
  2. Create a tablespace for the system tables. For example:

    CREATE TABLESPACE rdf_users datafile 'rdf_users01.dbf' 
        size 128M reuse autoextend on next 64M 
        maxsize unlimited segment space management auto;
    
  3. Create the semantic network. For example:

    EXECUTE sem_apis.create_sem_network('RDF_USERS');
    
  4. Create a database user (for connecting to the database to use the semantic network and the Sesame Adapter). For example:

    CREATE USER rdfusr IDENTIFIED BY <password-for-udfusr>
                       DEFAULT TABLESPACE rdf_users;
    
  5. Grant the necessary privileges to this database user. For example:

    GRANT connect, resource TO rdfusr;
    
  6. To use the Sesame Adapter with your own semantic data, perform the appropriate steps to store data, create a model, and create database indexes, as explained in Section 1.10, "Quick Start for Using Semantic Data". Then perform queries by compiling and running Java code; see Section 8.10 for information about example queries.

    To use the Sesame Adapter with supplied example data, see Section 8.10.

8.3 SEM_MATCH and Sesame Adapter Queries Compared

There are two ways to query semantic data stored in Oracle Database: SEM_MATCH-based SQL statements and SPARQL queries through the Sesame Adapter. Queries using each approach are similar in appearance, but there are important behavioral differences. To ensure consistent application behavior, you must understand the differences and use care when dealing with query results coming from SEM_MATCH queries and SPARQL queries.

The following simple examples show the two approaches.

Query 1 (SEM_MATCH-based)

select s, p, o
    from table(sem_match('{?s ?p ?o}', sem_models('Test_Model'), ....))

Query 2 (SPARQL query through the Sesame Adapter)

select ?s ?p ?o
where {?s ?p ?o}

These two queries perform the same kind of functions; however, there are some important differences. Query 1 (SEM_MATCH-based):

  • Reads all triples out of Test_Model.

  • Does not differentiate among URI, bNode, plain literals, and typed literals, and it does not handle long literals.

  • Does not unescape certain characters (such as '\n').

Query 2 (SPARQL query executed through the Sesame Adapter) also reads all triples out of Test_Model (assume it executed a call to ModelOracleSem referring to the same underlying Test_Model). However, Query 2:

  • Reads out additional columns (as opposed to just the s, p, and o columns with the SEM_MATCH table function), to differentiate URI, bNodes, plain literals, typed literals, and long literals. This is to ensure proper creation of Sesame Node objects.

  • Unescapes those characters that are escaped when stored in Oracle Database

Blank node handling is another difference between the two approaches:

  • In a SEM_MATCH-based query, blank nodes are always treated as constants.

  • In a SPARQL query, a blank node that is not wrapped inside < and > is treated as a variable when the query is executed through the Sesame Adapter. This matches the SPARQL standard semantics. However, a blank node that is wrapped inside < and > is treated as a constant when the query is executed, and the Sesame Adapter adds a proper prefix to the blank node label as required by the underlying data modeling.

The maximum length for the name of a semantic model created using the Sesame Adapter API is 22 characters.

8.4 Optimized Handling of SPARQL Queries

This section describes some performance-related features of the Sesame Adapter that can enhance SPARQL query processing. These features are performed automatically by default.

8.4.1 Compilation of SPARQL Queries to a Single SEM_MATCH Call

SPARQL queries involving DISTINCT, OPTIONAL, GRAPH, FILTER, UNION, ORDER BY, and LIMIT are converted to a single SQL query based on the SEM_MATCH table function. For SPARQL ASK, DESCRIBE, and CONSTRUCT queries, the Sesame Adapter employs additional query transformation logic and mid-tier processing.

If the SQL query based on SEM_MATCH fails to execute, the Sesame Adapter employs the Sesame evaluation strategy to execute the query.

8.5 Recommendations for Best Performance

The following are recommendations for enhancing the performance of queries using the Sesame Adapter:

  • Put only relevant triple patterns in the GRAPH clause of a SPARQL query, to reduce the number of additional joins with the application table. (For an explanation, see Section 8.5.1.)

  • For the triple patterns inside GRAPH clause of a named graph query, do not use variables with null bindings. A null binding will not match any row in the join with the relevant application table, and therefore this binding will be dropped.

  • Because of the preceding item, do not put an OPTIONAL clause in a GRAPH clause, or a GRAPH clause in an OPTIONAL clause.

  • Do not use blank nodes for the CONTEXT column in the application table (see Section 8.5.1), because blank nodes in named graphs from two different semantic models will be treated as the same resource if they have the same label. This is not the case for blank nodes in triples (they are stored separately if coming from different models).

8.5.1 Statement Storage and Implications for Sesame Adapter Queries

Because the MDSYS.SEMM_model-name view (described in Table 1-2 in Section 1.3.1) does not have columns for the graph name or context for each model, these two pieces of information are not stored inside that Oracle semantic model. Rather, this information is kept in the CONTEXT column in the user's application table.

When an OracleSailStore object is created for the first time using a model name that does not exist in the semantic network, a new application table is created automatically, and it includes these columns:

  • TRIPLE, of type SDO_RDF_TRIPLE_S

  • CONTEXT, of type VARCHAR2

A semantic model with the specified model name is then created. To enhance the performance of named graph queries and statement-oriented operations, a default index is created on the application table, and this index covers the CONTEXT column (leading column) as well as the three numeric ID attributes for triple's subject, predicate, and object field of the TRIPLE column.

When a SPARQL query runs against a semantic model, if the query does not use a named graph (that is, does not contain a GRAPH clause), the generated SQL statement will not use the model's application table. However, if the query has a GRAPH clause, the generated SQL statement will join a SEM_MATCH-based subquery with the model's application table to scope the results down to relevant contexts or named graphs. The number of joins with the application table depends on how many triple patterns are in the GRAPH clause, and therefore it is best to put only relevant triple patterns in the GRAPH clause, to reduce the number of additional joins with the application table.

8.6 Additions to the SPARQL Syntax to Support Other Features

The Sesame Adapter allows you to pass in hints and additional query options. It implements these capabilities by overloading the SPARQL namespace prefix syntax by using Oracle-specific namespaces that contain query options. The namespaces are in the form PREFIX ORACLE_SEM_xx_NS, where xx indicates the type of feature (such as HT for hint or AP for additional predicate)

8.6.1 SQL Hints

SQL hints can be passed to a SEM_MATCH query including a line in the following form:

PREFIX ORACLE_SEM_HT_NS: <http://oracle.com/semtech#hint>

Where hint can be any hint supported by SEM_MATCH. For example:

PREFIX ORACLE_SEM_HT_NS: <http://oracle.com/semtech#leading(t0,t1)> 
SELECT ?book ?title ?isbn     
WHERE { ?book <http://title> ?title. ?book <http://ISBN> ?isbn }

In this example, t0,t1 refers to the first and second patterns in the query.

Note the slight difference in specifying hints when compared to SEM_MATCH. Due to restrictions of namespace value syntax, a comma (,) must be used to separate t0 and t1 (or other hint components) instead of a space.

For more information about using SQL hints, see Section 1.6, "Using the SEM_MATCH Table Function to Query Semantic Data", specifically the material about the HINT0 keyword in the options attribute.

8.6.2 Additional WHERE Clause Predicates

The SEM_MATCH filter attribute can specify additional selection criteria as a string in the form of a WHERE clause without the WHERE keyword. Additional WHERE clause predicates can be passed to a SEM_MATCH query including a line in the following form:

PREFIX ORACLE_SEM_AP_NS: <http://oracle.com/semtech#pred>

Where pred reflects the WHERE clause content to be appended to the query. For example:

PREFIX rdfs: <http://www.w3.org/2000/01/rdf-schema#>
PREFIX ORACLE_SEM_AP_NS:<http://www.oracle.com/semtech#label$RDFLANG='fr'>  
SELECT DISTINCT ?inst ?label
  WHERE { ?inst a <http://someCLass>. ?inst rdfs:label ?label . }
  ORDER BY (?label) LIMIT 20

In this example, a restriction is added to the query that the language type of the label variable must be 'fr'.

8.6.3 Additional Query Options

Additional query options can be passed to a SEM_MATCH query including a line in the following form:

PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#option>

Where option reflects a query option (or multiple query options delimited by commas) to be appended to the query. For example:

PREFIX ORACLE_SEM_FS_NS:   
<http://oracle.com/semtech#timeout=3,dop=4,INF_ONLY,ORDERED,ALLOW_DUP=T>
SELECT * WHERE {?subject ?property ?object }

The following query options are supported:

  • ALLOW_DUP=t chooses a faster way to query multiple semantic models, although duplicate results may occur.

  • DOP=n specifies the degree of parallelism (n) for the query. With multi-core or multi-CPU processors, experimenting with different DOP values (such as 4 or 8) may improve performance.

  • INF_ONLY causes only the inferred model to be queried. (For this option, you should also specify the includeInferred=true option for the query.)

  • ORDERED is translated to a LEADING SQL hint for the query triple pattern joins, while performing the necessary RDF_VALUE$ joins last.

  • QID=n specifies a query ID number; this feature can be used to cancel the query if it is not responding.

  • REWRITE=F disables ODCI_Table_Rewrite for the SEM_MATCH table function.

  • SKIP_CLOB=T causes CLOB values not to be returned for the query.

  • TIMEOUT=n (query timeout) specifies the number of seconds (n) that the query will run until it is terminated.

  • USE_BIND_VAR(n) specifies that a bind variable be used during query parsing for the nth constant in triple patterns. (Do not apply this feature to constants inside an OPTIONAL clause.)

8.7 Support for Server-Side APIs

This section describes some of the Oracle Database Semantic Technologies features that are exposed by the Sesame Adapter. For comprehensive documentation of the API calls that support the available features, see the Sesame Adapter reference information (Javadoc). For additional information about the server-side features exposed by the Sesame Adapter, see the relevant chapters in this manual.

8.7.1 Virtual Models Support

Virtual models (explained in Section 1.3.8) are specified in the OracleSailStore constructor, and they are handled transparently. If a virtual model exists for the model-rulebase combination, it is used in query answering; if such a virtual model does not exist, it is created in the database.


Note:

Virtual model support through the Sesame Adapter is available only with Oracle Database Release 11.2 or later.

The following example reuses an existing virtual model.

public void vm(String jdbcUrl, 
                        String user, 
                        String password, 
                        String model,                              
                        PrintStream psOut) 
{
String m1 = "TEST_1";
String m2 = "TEST_2";
OraclePool op = new OraclePool(
    OraclePool.getOracleDataSource(jdbcUrl, user, password));
        
// create the attachment semantic model in case it doesn't exist 
// (constructor handles it)
OracleSailStore store = new OracleSailStore(op, m1);
store.shutDown();
                
store= new OracleSailStore(op, m2);
store.shutDown();
 
String vmName = "VM_" + m1;
 
// create a virtual model with name vmName that will be used in the 
// OracleSailStore
Oracle oracle = op.getOracle();
oracle.executeCall(
"begin sem_apis.create_virtual_model(?,sem_models(?,?), null); end;", 
 vmName, m1, m2);
op.returnOracle(oracle);
 
 String[] modelNames = {m2};
 String[] rulebaseNames = {};
 
 Attachment attachment = Attachment.createInstance(modelNames,
 rulebaseNames); 
 
    store = new OracleSailStore(op, m1, attachment, vmName);
 
OracleSailConnection conn = store.getConnection();
conn.clear();
conn.addStatement(
        store.getValueFactory().createURI("http://a1"), 
        store.getValueFactory().createURI("http://p1"), 
        store.getValueFactory().createURI("http://b1"));
   
CloseableIteration res = conn.getStatements(null, null, null, false);
while (res.hasNext()) {
  Statement stmt = (Statement) res.next();
  System.out.println(stmt);
}    
conn.close();
store.shutDown();
}

You can also use the OracleSailStore constructor to create a virtual model, as in the following example:

OracleSailStore store = new OracleSailStore(oracle, modelName, attachment, true);

In this example, the fourth parameter (true) specifies that a virtual model needs to be created for the specified modelName and attachment.

8.7.2 Connection Pooling Support

Oracle Database Connection Pooling is provided through the Sesame Adapter OraclePool class. Usually, OraclePool is initialized with a DataSource, using the OraclePool (DataSource ods) constructor. In this case, OraclePool acts as an extended wrapper for the DataSource, while using the connection pooling capabilities of the data source. When you create an OracleSailStore object, it is sufficient to specify the OraclePool object in the store constructor; the database connections will then be managed automatically by the Sesame Adapter.

If you need to retrieve Oracle connection objects (which are essentially database connection wrappers) explicitly, you can invoke the OraclePool.getOracle method. After finishing with the connection, you can invoke the OraclePool.returnOracle method to return the object to the connection pool.

OraclePool can also be initialized with an OracleConnection object. In this case, only a single database connection is used when referring to this OraclePool; and because multiple Sesame SailConnection objects will use the same database connection, you must be aware that "dirty" reads across different SailConnection objects are possible.

More information about using OraclePool can be found in the Sesame Adapter API reference information (Javadoc).

The following example sets up an OracleSailStore object using the OraclePool class, with two initial connections.

public void op(String jdbcUrl, 
                 String user, 
                 String password, 
                 String model,                              
                 PrintStream psOut)
  throws Exception
  {          
    // test with connection properties 
    java.util.Properties prop = new java.util.Properties();
    prop.setProperty("MinLimit", "2");     // the cache size is 2 at least 
    prop.setProperty("MaxLimit", "10");
    prop.setProperty("InitialLimit", "2"); // create 2 connections at startup
    prop.setProperty("InactivityTimeout", "1800");    //  seconds
    prop.setProperty("AbandonedConnectionTimeout", "900");  //  seconds
    prop.setProperty("MaxStatementsLimit", "10");
    prop.setProperty("PropertyCheckInterval", "60"); // seconds
 
    System.out.println("Creating Data Source");
    OracleDataSource ods = 
OraclePool.getOracleDataSource(jdbcUrl, user, password,
        prop, "OracleSemConnPool");
    System.out.println("Done creating Data Source");
    
    OraclePool op = new OraclePool(ods);
 
    // create an OracleSailStore with the OraclePool
    OracleSailStore store = new OracleSailStore(op, model);
    store.shutDown();
    op.close();
    ods.close();
  }

8.7.3 Semantic Model PL/SQL Interfaces

Several semantic PL/SQL subprograms are available through the Sesame Adapter. Table 8-1 lists the subprograms and their corresponding Java class and methods.

Table 8-1 PL/SQL Subprograms and Corresponding Sesame Adapter Java Class and Methods

PL/SQL SubprogramCorresponding Java Class and Methods

SEM_APIS.DROP_SEM_MODEL


OracleUtils.dropSemanticModel

SEM_APIS.MERGE_MODELS


OracleUtils.mergeModels

SEM_APIS.SWAP_NAMES


OracleUtils.swapNames

SEM_APIS.REMOVE_DUPLICATES


OracleUtils.removeDuplicates

SEM_APIS.RENAME_MODEL


OracleUtils.renameModels


For information about these PL/SQL utility subprograms, see the reference information in Chapter 9. For information about the corresponding Java class and methods, see the Sesame Adapter API Reference documentation (Javadoc).

8.7.4 Inference Options

You can add options to entailment calls by using the setInferenceOption(String options) procedure with the OracleSailConnection object. The following example enables incremental inference and specifies a parallelism value of 4 when creating an entailment.

conn.setInferenceOption("INC=T,DOP=4");

For information about inference options, see Section 2.2.

8.8 Oracle-Specific Extensions to Sesame APIs

This section describes the extensions that the Sesame Adapter for Oracle Database provides to the Sesame SailConnection and Sail APIs.

8.8.1 Statement Uniqueness

In Sesame 2.3.0, statement uniqueness is enforced at the context level: the same triple cannot occur twice in a single context. In the Oracle Sesame Adapter, such uniqueness is enforced through a unique index on the context, predicate, subject, and object. However, in the Sesame Adapter, statement uniqueness is not enabled by default. When constructing an OracleSailStore object, the default value for statement uniqueness is DONTCARE, meaning that if such a unique index does not already exist, it will not be created.

You can explicitly enable and disable the requirement for statement uniqueness either through the OracleSailStore constructor, or through the enableUniquenessConstraint and disableUniquenessConstraint methods.

8.8.2 Indexes and Interoperability with the Jena Adapter

Indexes on the application table are critical for good performance when performing DML operations (especially deletes) and named graph queries against an OracleSailStore object. When creating a new OracleSailStore object, a default CPSO index is created (where C,P,S,O stands for the context, predicate, subject, and object columns).

There are some storage differences between Oracle semantic models created using the Sesame Adapter and the Jena Adapter. For example, the default application table index is different in Jena Adapter. To access data in a Jena Adapter model from the Sesame Adapter (and vice versa), you can use the OracleUtils.migrateFromJena and OracleUtils.migrateToJena methods. These methods preserve the semantic data while making changes to the underlying storage structures. In particular, all triples along with their context information are preserved when migrating from Sesame Adapter to Jena Adapter and back.

Example 8-2 illustrates migration between models created using the Sesame Adapter and the Jena Adapter:

Example 8-2 Migration Between Jena Adapter and Sesame Adapter Models

public void migrate(String jdbcUrl, 
                         String user, 
                         String password, 
                         String model,                                
                         PrintStream psOut)
  throws Exception
  {
    OraclePool oraclePool= new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
       
    OracleSailStore store = new OracleSailStore(oraclePool, model);
    
    OracleSailConnection conn = store.getConnection();
    conn.clear();
    
    // create a statement and add it to the store
    URI subject = 
store.getValueFactory().createURI("http://example.org/foo");
    URI object = store.getValueFactory().createURI("http://example.org/bar");
    URI ngURI = store.getValueFactory().createURI("http://example.org/bar");
  
    conn.addStatement(subject, RDF.TYPE, object, ngURI);
    conn.commit();    
    
    // convert it to a jena model    
    OracleUtils.migrateToJena(oraclePool, model);    
 
    Oracle newOra = oraclePool.getOracle();
    oracle.spatial.rdf.client.jena.Oracle ora =
      new oracle.spatial.rdf.client.jena.Oracle(newOra.getConnection());
    
    GraphOracleSem graph = new GraphOracleSem(ora, model);
   
    // check size of jena graph
    System.out.print("GraphOracleSem size:" + graph.getSize());
    
    store.shutDown();
    
    // now convert it back to an Oracle sailstore.
    OracleUtils.migrateFromJena(oraclePool, model);
    conn = store.getConnection();   
 
    // check overall store size
    System.out.println("Size of store :" +conn.size());
    // check context size, should be 1
    System.out.println("Size of context " + ngURI + ":" + conn.size(ngURI));
    store.shutDown();
    ora.dispose();
    oraclePool.returnOracle(newOra);
  }

8.8.3 Inference

You can perform inference using the OracleSailConnection.performInference method. This method builds an entailment for the OracleSailStore object, given the models and rulebases that were specified in the attachment when constructing the store.

For information about creating entailments, see Section 1.3.7, "Entailments (Rules Indexes)".

8.8.4 Performing Analysis Operations

It is strongly recommended that you analyze the application table, semantic model, and inferred graph (if it exists) before performing inference and after loading a significant amount of semantic data into the database. Performing the analysis operations causes statistics to be gathered, which will help the Oracle optimizer select efficient execution plans when answering queries.

To gather relevant statistics, you can use the following methods in the OracleSailConnection and OracleSailStore classes:

  • OracleSailConnection.analyze

  • OracleSailConnection.analyzeApplicationTable

  • OracleSailConnection.analyzeInferredGraph

For information about these methods, including their parameters, see the Sesame Adapter reference information (Javadoc).

8.9 Using the Sesame Console with the Sesame Adapter

The Sesame Console is, as OpenRDF.org explains, "a command-line application for interacting with Sesame. For now, the best way to create and manage repositories in a SYSTEM repository is to use the Sesame Console. " For basic usage and reference information about the Sesame Console, see the documentation at http://www.openrdf.org/doc/sesame2/users/ch07.html.

This section describes how to use the Sesame Console with the Sesame Adapter for Oracle Database.

To create a new SAIL repository (OracleSailStore object) using the Sesame Console command-line tool, use the following command:

create oracle.

When you are prompted for parameters, you can specify the following:

Repository ID

ID string associated with the repository. Example: myOracleRepos

Repository Title

Title string associated with the repository. Example: My Oracle Repository

Model Name

Name of the Oracle semantic model that corresponds to this repository. All DML operations are performed against this model.

Additional Models

Comma-separated list of any additional semantic models to be included in this repository. Any additional models are used only for inference and query operations.

Rulebases

Comma-separated list of rulebases to be considered when building the entailment closure for this repository. The default is no rulebases, which means that no inference will be performed.

Use Virtual Model (TRUE|FALSE) [TRUE]

Specifies whether to use a virtual model when answering queries. The default is TRUE, which means to use a virtual model.

Virtual Model Name

If Use Virtual Model is TRUE, specify the name of the virtual model to be used to answer queries, if the database user that owns the repository has read permission on the specified virtual model.

Inference Maintenance Mode (NO_UPDATE|UPDATE_WHEN_COMMIT) [NO_UPDATE]

Specifies whether to update the entailment closure on each commit operation (UPDATE_WHEN_COMMIT) or only when the OracleSailConnection.performInference method is invoked (NO_UPDATE). The default is NO_UPDATE.

Allow Duplicates in Query (TRUE|FALSE) [TRUE]

Specifies whether, when querying a repository that has additional models, the query response can include duplicates across the different models. The default is TRUE, allows duplicates to be included, and which speeds query performance.

Allow Query with Non-Valid Inference status (INVALID|VALID|INCOMPLETE) [INVALID]

Specifies whether to allow queries when the entailment is not updated, that is, when the inference status is not valid. The default value is INVALID, which means that the entailment status can be INVALID when querying.

To allow queries only when entailments have a valid status, specify VALID. To allow queries when entailments have either a valid or incomplete status, specify INCOMPLETE.

Application Table Indexes [CPSO]

Specifies the custom index to be created on the application table for this repository. You can specify the columns to be included (and their order) by using a string of up to 4 of the following characters: C (context), P (property), S (subject), O (object). (You can use only those characters and cannot repeat any characters.) The default is CPSO, which creates an index on all four columns with context as the leading column.

Index Option: Degree of Parallelism [1]

Specifies the DOP (degree of parallelism) value for the application table indexes. With multi-core or multi-CPU processors, experimenting with different DOP values (such as 4 or 8) may improve performance. The default is 1 (no parallelism).

Index Option: Prefix Length to Compress [2]

Specifies whether to use key compression on the application table indexes. The default is 2, which specifies compression on the first two columns. A value of 0 (zero) means that no compression will be used.

Enforce Uniqueness (FALSE|TRUE) []

Specifies whether to enforce statement uniqueness across multiple contexts in the repository. Note that statement uniqueness is enforced by a unique index on the application table. The default is [] (that is, no value specified), which means that if the model exists, the uniqueness setting will be inherited from that model; and if the model does not already exist, a non-unique index will be created.

If you specify FALSE and if a unique index exists, that index is dropped and it is re-created as a non-unique index.

Data Source Name [OracleSemDS]

Name of the JDBC data source that provides the database connection setting for accessing the Oracle semantic models and application tables for this repository. The default is OracleSemDS.

Example 8-3 shows a sample scenario of using the Sesame Console.

Example 8-3 Using the Sesame Console

> connect http://localhost:8080/openrdf-sesame.
Disconnecting from default data directory
Connected to http://localhost:8080/openrdf-sesame
> create oracle.
Please specify values for the following variables:
Repository ID: model2Repos
Repository title: Repository for Model 2
Model Name: model2
Additional Models (comma separated):    
Rulebases (comma separated): OWLPRIME    -- OWLPRIME inference will be performed when this model is created
Use Virtual Model (TRUE|FALSE) [TRUE]:  TRUE
Virtual Model Name:     -- a new virtual model will be created if nonexistent
Inference Maintenance Mode (UPDATE_WHEN_COMMIT|NO_UPDATE) [UPDATE_WHEN_COMMIT]: NO_UPDATE
Allow duplicates in query (TRUE|FALSE) [TRUE]: TRUE
Allow query with non-valid inference status (NONE|INCOMPLETE|INVALID) [NONE]: NONE – inference must be up to date before query
Application Table Index [CPSO]: CPSO      –- default, recommended index
Index option: Degree of parallelism [1]: 1
Index option: Leading columns to compress [2]: 2
Enforce Uniqueness (|FALSE|TRUE) []: FALSE
DataSource Name [OracleSemDS]: OracleSemDS
Repository created
> open model2Repos.
Opened repository 'model2Repos'
model2Repos> show c.
--no contexts found--
model2Repos>

8.10 Example Queries Using the Sesame Adapter

This section includes example queries using the Sesame Adapter. Each example is self-contained: it typically starts an OraclePool object, creates an OracleSailStore object, adds and perhaps removes statements, performs a query that may involve inference, displays the result, shuts down the OracleSailStore object, and closes the OraclePool object.

For these examples, the following libraries must be included in the CLASSPATH definition:

sesame-console-2.3.1.jar
sesame-http-client-2.3.1.jar
sesame-http-protocol-2.3.1.jar
sesame-http-server-spring-2.3.1.jar
sesame-model-2.3.1.jar
sesame-query-2.3.1.jar
sesame-queryalgebra-evaluation-2.3.1.jar
sesame-queryalgebra-model-2.3.1.jar
sesame-queryparser-api-2.3.1.jar
sesame-queryparser-serql-2.3.1.jar
sesame-queryparser-sparql-2.3.1.jar
sesame-queryresultio-api-2.3.1.jar
sesame-queryresultio-binary-2.3.1.jar
sesame-queryresultio-sparqljson-2.3.1.jar
sesame-queryresultio-sparqlxml-2.3.1.jar
sesame-queryresultio-text-2.3.1.jar
sesame-repository-api-2.3.1.jar
sesame-repository-contextaware-2.3.1.jar
sesame-repository-dataset-2.3.1.jar
sesame-repository-event-2.3.1.jar
sesame-repository-http-2.3.1.jar
sesame-repository-manager-2.3.1.jar
sesame-repository-sail-2.3.1.jar
sesame-rio-api-2.3.1.jar
sesame-rio-n3-2.3.1.jar
sesame-rio-ntriples-2.3.1.jar
sesame-rio-rdfxml-2.3.1.jar
sesame-rio-trig-2.3.1.jar
sesame-rio-trix-2.3.1.jar
sesame-rio-turtle-2.3.1.jar
sesame-runtime-2.3.1.jar
sesame-sail-api-2.3.1.jar
sesame-sail-inferencer-2.3.1.jar
sesame-sail-memory-2.3.1.jar
sesame-sail-nativerdf-2.3.1.jar
sesame-sail-rdbms-2.3.1.jar
sdordf.jar
sdordfsesame.jar
ojdbc6.jar

To simplify the examples, an environment variable named CP was defined, as follows:

setenv CP ./:sesame-console-2.3.1.jar:sesame-http-client-2.3.1.jar:sesame-http-protocol-2.3.1.jar:sesame-http-server-spring-2.3.1.jar:sesame-model-2.3.1.jar:sesame-query-2.3.1.jar:sesame-queryalgebra-evaluation-2.3.1.jar:sesame-queryalgebra-model-2.3.1.jar:sesame-queryparser-api-2.3.1.jar:sesame-queryparser-serql-2.3.1.jar:sesame-queryparser-sparql-2.3.1.jar:sesame-queryresultio-api-2.3.1.jar:sesame-queryresultio-binary-2.3.1.jar:sesame-queryresultio-sparqljson-2.3.1.jar:sesame-queryresultio-sparqlxml-2.3.1.jar:sesame-queryresultio-text-2.3.1.jar:sesame-repository-api-2.3.1.jar:sesame-repository-contextaware-2.3.1.jar:sesame-repository-dataset-2.3.1.jar:sesame-repository-event-2.3.1.jar:sesame-repository-http-2.3.1.jar:sesame-repository-manager-2.3.1.jar:sesame-repository-sail-2.3.1.jar:sesame-rio-api-2.3.1.jar:sesame-rio-n3-2.3.1.jar:sesame-rio-ntriples-2.3.1.jar:sesame-rio-rdfxml-2.3.1.jar:sesame-rio-trig-2.3.1.jar:sesame-rio-trix-2.3.1.jar:sesame-rio-turtle-2.3.1.jar:sesame-runtime-2.3.1.jar:sesame-sail-api-2.3.1.jar:sesame-sail-inferencer-2.3.1.jar:sesame-sail-memory-2.3.1.jar:sesame-sail-nativerdf-2.3.1.jar:sesame-sail-rdbms-2.3.1.jar:sdordf.jar:sdordfsesame.jar:ojdbc6.jar

Note:

Enter the setenv command, as well as each javac and java command, on a single command line.

To run a query, you must do the following:

  1. Include the code in a Java source file. The examples used in this section are supplied in files in the examples directory of the Sesame Adapter download.

  2. Compile the Java source file. For example:

    > javac -classpath $CP Example1.java
    
  3. Run the compiled file. For example:

    > java -classpath $CP Test jdbc:oracle:thin:@localhost:1521:orcl scott <password-for-scott> M1
    

8.10.1 Example1.java: Basic Operations

Example 8-4 shows the Example1.java file, which performs some basic operations.

Example 8-4 Basic Operations

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
 
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
 
public class Example1
{
  public static void main(String[] args) throws SQLException, SailException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);       
 
    OracleSailConnection conn = store.getConnection();
 
    ValueFactory vf = store.getValueFactory();
    URI p = vf.createURI("http://p");   
    URI cls = vf.createURI("http://cls");
    URI a = vf.createURI("http://a");
    URI b = vf.createURI("http://b");
    URI ng1 = vf.createURI("http://ng1");
 
    conn.clear();
    conn.addStatement(p, RDFS.DOMAIN, cls);
    conn.addStatement(a, p, b, ng1);
 
    psOut.println("size of context " + ng1 + ":" + conn.size(ng1));
 
    // returns OracleStatements
    CloseableIteration<? extends Statement, SailException> it =
      conn.getStatements(null, null, null, false);
 
    while (it.hasNext()) {   
      Statement stmt = it.next();
      psOut.println("getStatements: stmt: " + stmt.toString());
    }
 
    conn.removeStatements(null, null, null, ng1);
 
    psOut.println("size of context " + ng1 + ":" + conn.size(ng1));
 
    conn.removeAll();   
 
    psOut.println("size of store: " + conn.size());
 
    conn.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

/usr/local/packages/jdk16/bin/javac -classpath $CP Example1.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example1 jdbc:oracle:thin:@localhost:1521:ORCL scott <password-for-scott> TestModel

The expected output of the java command might appear as follows:

size of context http://ng1:1
getStatements: stmt:  O: (http://a, http://p, http://b) [http://ng1]
getStatements: stmt:  O: (http://p, http://www.w3.org/2000/01/rdf-schema#domain, http://cls) [null]
size of context http://ng1:0
size of store: 0

8.10.2 Example2.java: Add a Data File (in TRIG format)

Example 8-5 shows the Example2.java file, which adds a data file in TRIG format.

Example 8-5 Add a Data File (in TRIG format)

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
 
public class Example2
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
    String trigFile = args[4];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);       
    SailRepository sr = new SailRepository(store);
    RepositoryConnection repConn = sr.getConnection();
 
    repConn.setAutoCommit(false);
    repConn.add(new File(trigFile), "http://my.com/", RDFFormat.TRIG);
    repConn.commit();
 
    psOut.println("size " + Long.toString(repConn.size()));
 
    repConn.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example2.java

For running this example, assume that a sample TRIG data file named test.trig was created as:

@prefix rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#> .
@prefix xsd: <http://www.w3.org/2001/XMLSchema#> .
@prefix swp: <http://www.w3.org/2004/03/trix/swp-1/> .
@prefix dc: <http://purl.org/dc/elements/1.1/> .
@prefix foaf: <http://xmlns.com/foaf/0.1/> .
@prefix ex: <http://example.org/> .
@prefix : <http://example.org/> .
 
# default graph
{
<http://example.org/bob>    dc:publisher  "Bob Hacker" .
<http://example.org/alice>  dc:publisher  "Alice Hacker" .
}
 
:bob {
_:a foaf:mbox <mailto:bob@oldcorp.example.org> .
}
 
:alice {
_:a foaf:name "Alice" .
_:a foaf:mbox <mailto:alice@work.example.org> .
}
 
:jack {
_:a foaf:name "Jack" .
_:a foaf:mbox <mailto:jack@oracle.example.org> .
}

To run this example using the test.trig data file, enter the following command:

/usr/local/packages/jdk16/bin/java -classpath $CP Example2 jdbc:oracle:thin:@localhost:1521:ORCL scott  <password>  TestModel ./test.trig

The expected output of the java command might appear as follows:

size 7

8.10.3 Example3.java: Simple Query

Example 8-6 shows the Example3.java file, which performs a simple query.

Example 8-6 Simple Query

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example3
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);   
    SailRepository myRepository = new SailRepository(store);
 
    ValueFactory f = myRepository.getValueFactory();
    RepositoryConnection conn = myRepository.getConnection();
 
    // create some resources and literals to make statements out of
    URI alice = f.createURI("http://example.org/people/alice");
    URI name = f.createURI("http://example.org/ontology/name");
    URI person = f.createURI("http://example.org/ontology/Person");
    Literal alicesName = f.createLiteral("Alice");
 
    conn.clear(); // to start from scratch
    conn.add(alice, RDF.TYPE, person);
    conn.add(alice, name, alicesName);
    conn.commit();
 
    store.analyze();
    store.analyzeApplicationTable();
 
    try {
      //run a query against the repository
      String queryString = " SELECT * WHERE {?x ?p ?y} LIMIT 1 ";
      TupleQuery tupleQuery = conn.prepareTupleQuery(QueryLanguage.SPARQL, queryString);
      TupleQueryResult result = tupleQuery.evaluate();
      try {
        while (result.hasNext()) {
          BindingSet bindingSet = result.next();
          psOut.println("value of x: " + bindingSet.getValue("x"));
        }
      }
      finally {
        result.close();
      }
    }
    finally {  
      conn.clear();
      if (conn != null && conn.isOpen()) {
        conn.close();
      }       
 
      myRepository.shutDown();
      op.close();
    }
  }
}

To compile this example, enter the following command:

/usr/local/packages/jdk16/bin/javac -classpath $CP Example3.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example3 jdbc:oracle:thin:@localhost:1521:ORCL scott <password-for-scott> TestModel

The expected output of the java command might appear as follows:

value of x: http://example.org/people/alice

8.10.4 Example4.java: Simple Bulk Load

Example 8-7 shows the Example4.java file, which performs a simple bulk load operation.

Example 8-7 Simple Bulk Load

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import org.openrdf.model.Resource;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example4
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
    String filename = args[4]; // N-TRIPLES file
 
    OraclePool op = new OraclePool(OraclePool.getOracleDataSource(jdbcUrl, user, password));
 
    psOut.println("testBulkLoad: start");
 
    OracleSailStore store = new OracleSailStore(op, model);
    OracleSailConnection osc = store.getConnection();
    FileInputStream fis = new FileInputStream(filename);
   
    long loadBegin = System.currentTimeMillis();
    osc.getBulkUpdateHandler().addInBulk(
        fis,
        "http://abc",                // baseURI
        RDFFormat.NTRIPLES,          // dataFormat
        null,                        // tablespaceName
        null,                        // flags
        null,                        // StatusListener
        (Resource[]) null            // Resource... for contexts
        );
 
    long loadEnd = System.currentTimeMillis();
    psOut.println("testBulkLoad: " + (loadEnd - loadBegin) + "ms.\n");
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example4.java

For running this example, assume that a sample ntriples data file named test.ntriples was created as:

<http://example.org/bob>    <http://purl.org/dc/elements/1.1/publisher>  "Bob Hacker" .
<http://example.org/alice>  <http://purl.org/dc/elements/1.1/publisher>  "Alice Hacker" .

To run this example using the test.ntriples file, enter the following command:

/usr/local/packages/jdk16/bin/java -classpath $CP Example4 jdbc:oracle:thin:@localhost:1521:ORCL scott  <password>  TestModel ./test.ntriples

The expected output of the java command might appear as follows:

testBulkLoad: start
testBulkLoad: 756ms

8.10.5 Example5.java: Bulk Load RDF/XML and Application Table Index Maintenance

Example 8-8 shows the Example5.java file, which disables indexes on the application table, performs a bulk load operation from an RDF file in XML format, and re-enables the indexes.

Example 8-8 Bulk Load RDF/XML and Application Table Index Maintenance

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import org.openrdf.model.Resource;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example5
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
    String filename = args[4]; // RDF/XML
 
    OraclePool op = new OraclePool(OraclePool.getOracleDataSource(jdbcUrl, user, password));
 
    psOut.println("testBulkLoad: start");
 
    OracleSailStore store = new OracleSailStore(op, model);
    OracleSailConnection osc = store.getConnection();
 
    // disable indexes on application table
    store.disableAllAppTabIndexes();
 
    // NOTE: can be a gzipped file!
    FileInputStream fis = new FileInputStream(filename);
   
    long loadBegin = System.currentTimeMillis();
    osc.getBulkUpdateHandler().addInBulk(
        fis,
        "http://abc",                // baseURI
        RDFFormat.RDFXML,            // dataFormat
        null,                        // tablespaceName
        null,                        // flags
        null,                        // StatusListener
        (Resource[]) null            // Resource... for contexts
        );
 
    long loadEnd = System.currentTimeMillis();
    psOut.println("testBulkLoad: " + (loadEnd - loadBegin) + "ms.\n");
   
    // enable indexes on application table
    // Note: one can also specify to rebuild indexes in parallel.
    store.enableAllAppTabIndexes();
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example5.java

For running this example, assume that a sample file named test.rdfxml was created as:

<?xml version="1.0"?>
<!DOCTYPE owl [
     <!ENTITY owl  "http://www.w3.org/2002/07/owl#" >
     <!ENTITY xsd  "http://www.w3.org/2001/XMLSchema#" >
   ]>
 
<rdf:RDF
  xmlns     = "http://a/b#" xml:base  = "http://a/b#" xmlns:my  = "http://a/b#"
  xmlns:owl = "http://www.w3.org/2002/07/owl#"
  xmlns:rdf = "http://www.w3.org/1999/02/22-rdf-syntax-ns#"
  xmlns:rdfs= "http://www.w3.org/2000/01/rdf-schema#"
  xmlns:xsd = "http://www.w3.org/2001/XMLSchema#">
  <owl:Class rdf:ID="Color">
    <owl:oneOf rdf:parseType="Collection">
      <owl:Thing rdf:ID="Red"/>
      <owl:Thing rdf:ID="Blue"/>
    </owl:oneOf>
  </owl:Class>
</rdf:RDF>

To run this example using the test.rdfxml file, enter the following command:

/usr/local/packages/jdk16/bin/java -classpath $CP Example5 jdbc:oracle:thin:@localhost:1521:ORCL scott  <password>  EX5 ./test.rdfxml

The expected output of the java command might appear as follows:

testBulkLoad: start
testBulkLoad: 825ms

8.10.6 Example6.java: Bulk Load With StatusListener to Handle Loading with Bad Data

Example 8-9 shows the Example6.java file, which performs the bulk load and index maintenance operations shown in Example 8-7, but also registers and implements a StatusListener class to check for and report any data errors.

Example 8-9 Bulk Load With StatusListener to Handle Loading with Bad Data

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import org.openrdf.model.Resource;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.model.Statement;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example6
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
    String filename = args[4]; // RDF/XML
 
    OraclePool op = new OraclePool(OraclePool.getOracleDataSource(jdbcUrl, user, password));
 
    psOut.println("testBulkLoad: start");
 
    OracleSailStore store = new OracleSailStore(op, model);
    OracleSailConnection osc = store.getConnection();
 
    // disable indexes on application table
    store.disableAllAppTabIndexes();
 
    // NOTE: can be a gzipped file!
    FileInputStream fis = new FileInputStream(filename);
   
    long loadBegin = System.currentTimeMillis();
    osc.getBulkUpdateHandler().addInBulk(
        fis,
        "http://abc",                // baseURI
        RDFFormat.NTRIPLES,          // dataFormat
        null,                        // tablespaceName
        null,                        // flags
        new MyListener(psOut),       // register a StatusListener
        (Resource[]) null            // Resource... for contexts
        );
 
    long loadEnd = System.currentTimeMillis();
    psOut.println("testBulkLoad: " + (loadEnd - loadBegin) + "ms.\n");
   
    // enable indexes on application table
    // Note: one can also specify to rebuild indexes in parallel.
    store.enableAllAppTabIndexes();
 
    osc.close();
    store.shutDown();
    op.close();
  }

  static class MyListener implements StatusListener {
    PrintStream m_ps = null;
    public MyListener(PrintStream ps) { m_ps = ps; }
 
    public void statusChanged(long count)
    {
      m_ps.println("process to " + Long.toString(count));
    }
 
    public int illegalStmtEncountered(Statement statement, long count)
    {
      m_ps.println("hit illegal statement with object " + statement.getObject().toString());
      return 0; // skip it
    }
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example6.java

For running this example, assume that a sample N-TRIPLES file named test-include-badtriple.ntriples was created as follows. In this file, the first and last triples have illegal typed literal values for objects.

<http://example.org/x>      <http://my.com/#age> "123.3456"^^<http://www.w3.org/2001/XMLSchema#integer> .
<http://example.org/bob>    <http://purl.org/dc/elements/1.1/publisher>  "Bob Hacker" .
<http://example.org/alice>  <http://purl.org/dc/elements/1.1/publisher>  "Alice Hacker" .
<http://example.org/y>      <http://my.com/#age> "hello"^^<http://www.w3.org/2001/XMLSchema#float> .

To run this example using the test-include-badtriple.ntriples file, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example6 jdbc:oracle:thin:@localhost:1521:ORCL scott  <password>  EX6 ./test-include-badtriple.ntriples

The expected output of the java command might appear as follows:

testBulkLoad: start
hit illegal statement with object "123.3456"^^<http://www.w3.org/2001/XMLSchema#integer>
process to 2
process to 3
hit illegal statement with object "hello"^^<http://www.w3.org/2001/XMLSchema#float>
testBulkLoad: 725ms.

8.10.7 Example7.java: Load Data from Sesame Store into Oracle Database

Example 8-10 shows the Example7.java file, disables indexes on the application table, loads an RDF file in XML format into an in-memory Sesame store, loads the data into the Oracle database, and re-enables the indexes.

Example 8-10 Load Data from Sesame Store into Oracle Database

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import org.openrdf.model.Resource;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.sail.memory.MemoryStore;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.model.Statement;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example7
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
    String filename = args[4]; // RDF/XML
 
    OraclePool op = new OraclePool(OraclePool.getOracleDataSource(jdbcUrl, user, password));
 
    psOut.println("testBulkLoad: start");
 
    SailRepository srInMem = null;
    RepositoryConnection repConnInMem = null;
    { // build an in memory Sesame store for testing purpose
      srInMem = new SailRepository(new MemoryStore());
      srInMem.initialize();
      repConnInMem = srInMem.getConnection();
      File file = new File(filename);
      repConnInMem.setAutoCommit(false);
      repConnInMem.add(file, "http://my.com", RDFFormat.RDFXML);
      repConnInMem.commit();
    }
 
    OracleSailStore store = new OracleSailStore(op, model);
    OracleSailConnection osc = store.getConnection();
 
    // disable indexes on application table
    store.disableAllAppTabIndexes();
 
    long loadBegin = System.currentTimeMillis();
    // load all statements from in memory store to Oracle
    osc.getBulkUpdateHandler().addInBulk(
        repConnInMem.getStatements(null, null, null, false),
        null                       // tablespaceName
        );
 
    long loadEnd = System.currentTimeMillis();
    psOut.println("testBulkLoad: " + (loadEnd - loadBegin) + "ms.\n");
   
    // enable indexes on application table
    // Note: one can also specify to rebuild indexes in parallel.
    store.enableAllAppTabIndexes();
 
    repConnInMem.close();
    srInMem.shutDown();
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example7.java

For running this example, assume that a file (to be loaded into memory) with the same format and content as the file described in Section 8.10.5, "Example5.java: Bulk Load RDF/XML and Application Table Index Maintenance" has been created.

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example7 jdbc:oracle:thin:@localhost:1521:ORCL scott  <password>  EX7 ./test.rdfxml

The expected output of the java command might appear as follows:

testBulkLoad: start
testBulkLoad: 720ms

8.10.8 Example8.java: SPARQL ASK Query

Example 8-11 shows the Example8.java file, which performs a SPARQL ASK query.

Example 8-11 SPARQL ASK Query

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.BooleanQuery;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example8
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);
 
    SailRepository sr = new SailRepository(store);
    RepositoryConnection repConn = sr.getConnection();
 
    ValueFactory vf = sr.getValueFactory();
    URI p   = vf.createURI("http://p");   
    URI cls = vf.createURI("http://cls");    
 
    repConn.clear();
    repConn.add(p, RDFS.DOMAIN, cls);
    repConn.commit();
 
    store.analyze();                 // analyze the semantic model
    store.analyzeApplicationTable(); // and then the application table
 
    BooleanQuery tq = null;
    tq = repConn.prepareBooleanQuery(QueryLanguage.SPARQL, "ASK { ?x ?p <http://cls> }" );
    boolean b = tq.evaluate();
    psOut.print("\nAnswer is " + Boolean.toString(b));
 
    repConn.close(); 
    sr.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example8.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example4 jdbc:oracle:thin:@localhost:1521:ORCL scott  <password>  EX8

The expected output of the java command might appear as follows:

Answer is true

8.10.9 Example9.java: SPARQL CONSTRUCT and DESCRIBE

Example 8-12 shows the Example9.java file, which performs SPARQL CONSTRUCT and DESCRIBE queries.

Example 8-12 SPARQL CONSTRUCT and DESCRIBE

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.BooleanQuery;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.GraphQuery;
import org.openrdf.query.GraphQueryResult;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example9
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);
 
    SailRepository sr = new SailRepository(store);
    RepositoryConnection repConn = sr.getConnection();
 
    ValueFactory vf = sr.getValueFactory();
    URI p   = vf.createURI("http://p");   
    URI cls = vf.createURI("http://cls");    
 
    repConn.clear();
    repConn.add(p, RDFS.DOMAIN, cls);
    repConn.commit();
 
    store.analyze();                 // analyze the semantic model
    store.analyzeApplicationTable(); // and then the application table
 
    GraphQuery tq = null;
    tq = repConn.prepareGraphQuery(QueryLanguage.SPARQL,
        "CONSTRUCT {?x <http://new_eq_p> ?o } WHERE { ?x ?p ?o }" );
    {
      psOut.println("Start construct query");
      GraphQueryResult result = tq.evaluate();
      while (result.hasNext()) {
        Statement stmt = (Statement) result.next();
        psOut.println(stmt.toString());
        // do something interesting with the values here...
      }
      result.close();
    }
 
    tq = repConn.prepareGraphQuery(QueryLanguage.SPARQL,
        "DESCRIBE <http://p> ");
    {
      psOut.println("Start describe query");
      GraphQueryResult result = tq.evaluate();
      while (result.hasNext()) {
        Statement stmt = (Statement) result.next();
        psOut.println(stmt.toString());
        // do something interesting with the values here...
      }
      result.close();
    }
 
    repConn.close(); 
    sr.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example9.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example9 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX9

The expected output of the java command might appear as follows:

Start construct query
(http://p, http://new_eq_p, http://cls)
Start describe query
(http://p, http://www.w3.org/2000/01/rdf-schema#domain, http://cls)

8.10.10 Example10.java: Inference

Example 8-13 shows the Example10.java file, which performs OWLPrime inference using the incremental (INC) and degree of parallelism (DOP) options.

Example 8-13 Inference

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.BooleanQuery;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.GraphQuery;
import org.openrdf.query.GraphQueryResult;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example10
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    String[] rulebaseNames = new String[1];
    rulebaseNames[0] = "owlprime";
 
    Attachment attachment = Attachment.createInstance(
        Attachment.NO_ADDITIONAL_MODELS, rulebaseNames,
        InferenceMaintenanceMode.NO_UPDATE,
        QueryOptions.DEFAULT);
 
    OracleSailStore store = new OracleSailStore(op, model, attachment);   
    OracleSailConnection osc = store.getConnection();
 
    ValueFactory vf = osc.getValueFactory();
 
    URI sub, pred, obj;
    {
      sub = vf.createURI("http://C1");
      pred = vf.createURI("http://www.w3.org/2000/01/rdf-schema#subClassOf");
      obj  = vf.createURI("http://C2");
      osc.addStatement(sub, pred, obj);
 
      sub = vf.createURI("http://C2");
      pred = vf.createURI("http://www.w3.org/2000/01/rdf-schema#subClassOf");
      obj  = vf.createURI("http://C3");
      osc.addStatement(sub, pred, obj);
    }
 
    osc.commit();
    osc.analyze();                 // analyze the semantic model
    osc.analyzeApplicationTable(); // and then the application table
 
    // parallel inference is certainly an overkill for this ontology
    osc.setInferenceOption("INC=T,DOP=4,RAW8=T");
    osc.performInference();
    osc.analyzeInferredGraph();
 
    TupleQuery tq = null;
    RepositoryConnection repConn = osc.asRepositoryConnection();
    tq = repConn.prepareTupleQuery(QueryLanguage.SPARQL,
        "SELECT ?s ?p ?o WHERE {?s ?p ?o } ");
    {
      TupleQueryResult result = tq.evaluate();
      int idx = 0;
      try {
        psOut.print("\nStart printing solution\n");
        while (result.hasNext()) {
          idx++;
          BindingSet bindingSet = result.next();
        }
      }
      finally {
        psOut.println("\ntotal # of solution " + Integer.toString(idx));
        result.close();
      }
    }
 
 
    {
      sub  = vf.createURI("http://C3");
      pred = vf.createURI("http://www.w3.org/2000/01/rdf-schema#subClassOf");
      obj  = vf.createURI("http://C4");
      osc.addStatement(sub, pred, obj);
    }
 
    // make a small change and then perform inference again
    osc.commit();
    osc.analyze();                 // analyze the semantic model
    osc.analyzeApplicationTable(); // and then the application table
    osc.performInference();
    osc.analyzeInferredGraph();
 
    {
      TupleQueryResult result = tq.evaluate();
      int idx = 0;
      try {
        psOut.print("\nStart printing solution\n");
        while (result.hasNext()) {
          idx++;
          BindingSet bindingSet = result.next();
        }
      }
      finally {
        psOut.println("\ntotal # of solution " + Integer.toString(idx));
        result.close();
      }
    }
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example10.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example10 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX10

The expected output of the java command might appear as follows:

Start printing solution
 
total # of solution 3
 
Start printing solution
 
total # of solution 6

8.10.11 Example11.java: Named Graph Query

Example 8-14 shows the Example9.java file, which performs a named graph query.

Example 8-14 Named Graph Query

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.BooleanQuery;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.GraphQuery;
import org.openrdf.query.GraphQueryResult;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example11
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
    String trigFile = args[4];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
 
    OracleSailStore store = new OracleSailStore(op, model);
    OracleSailConnection osc = store.getConnection();
    RepositoryConnection repConn = osc.asRepositoryConnection();
    repConn.setAutoCommit(false);
 
    // load the data incrementally since it is very small file
    repConn.add(new File(trigFile), "http://my.com/", RDFFormat.TRIG);
 
    osc.commit();
    osc.analyze();                 // analyze the semantic model
    osc.analyzeApplicationTable(); // and then the application table
 
    TupleQuery tq = null;
    tq = repConn.prepareTupleQuery(QueryLanguage.SPARQL,
        "SELECT ?g ?s ?p ?o WHERE {?g <http://purl.org/dc/elements/1.1/publisher> ?o1 . GRAPH ?g {?s ?p ?o}}");
    {
      TupleQueryResult result = tq.evaluate();
      int idx = 0;
      try {
        while (result.hasNext()) {
          idx++;
          BindingSet bindingSet = result.next();
          psOut.print("\nsolution " + bindingSet.toString());
        }
      }
      finally {
        psOut.println("\ntotal # of solution " + Integer.toString(idx));
        result.close();
      }
    }
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example9.java

For running this example, assume that the test.trig file described in Section 8.10.2, "Example2.java: Add a Data File (in TRIG format)" has been created.

To run this example using the test.trig data file, enter the following command:

/usr/local/packages/jdk16/bin/java -classpath $CP Example11 jdbc:oracle:thin:@localhost:1521:ORCL scott  <password>  EX11 ./test.trig

The expected output of the java command might appear as follows:

solution [g=http://example.org/alice;s=_:node14r2238h1x1;p=http://xmlns.com/foaf/0.1/mbox;o=mailto:alice@work.example.org]
solution [g=http://example.org/alice;s=_:node14r2238h1x1;p=http://xmlns.com/foaf/0.1/name;o="Alice"]
solution [g=http://example.org/bob;s=_:node14r2238h1x1;p=http://xmlns.com/foaf/0.1/mbox;o=mailto:bob@oldcorp.example.org]
total # of solution 3

8.10.12 Example12.java: Indexes on Application Table [Advanced]

Example 8-15 shows the Example12.java file, an advanced example that creates indexes on the application table.

Example 8-15 Indexes on Application Table [Advanced]

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.BooleanQuery;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.GraphQuery;
import org.openrdf.query.GraphQueryResult;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example12
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
 
    OracleSailStore store = new OracleSailStore(op, model);
 
    // create an index with compress 2, parallel 4
    store.createAppTabIndex("spoc", 2, 4);
 
    // create another index w/o parallel or compression
    store.createAppTabIndex("cspo", 0, 1);
 
    // will print out the default index (cpso) as well
    psOut.println("index signatures before dropping indexes:");
    for (String sig : store.getAppTabIndexSignatures()) {
      psOut.println("index signature:" + sig);     
    }
 
    store.dropAllAppTabIndexes();
    psOut.println("index signatures after dropping indexes...");
    for (String sig : store.getAppTabIndexSignatures()) {
      psOut.println("index signature:" + sig);     
    }
 
    store.shutDown();
 
    // clean up oracle tables and models
    OracleUtils.dropSemanticModelAndTables(op, model);
 
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example12.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example12 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX12

The expected output of the java command might appear as follows:

index signatures before dropping indexes:
index signature:spoc
index signature:cpso
index signature:cspo
index signatures after dropping indexes...

8.10.13 Example13.java: Uniqueness Constraint on Application Table [Advanced]

Example 8-16 shows the Example13.java file, an advanced example that enables and disables the uniqueness constraint on the application table.

Example 8-16 Uniqueness Constraint on Application Table [Advanced]

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.BooleanQuery;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.GraphQuery;
import org.openrdf.query.GraphQueryResult;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example13
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
 
    OracleSailStore store = new OracleSailStore(op, model);
    store.enableUniquenessConstraint();    ValueFactory f = store.getValueFactory();
 
    OracleSailConnection osc = store.getConnection();
 
    // create some resources and literals to make statements out of
    URI alice = f.createURI("http://example.org/people/alice");
    URI name = f.createURI("http://example.org/ontology/name");
    URI person = f.createURI("http://example.org/ontology/Person");   
 
    osc.addStatement(alice, name, person);
    osc.addStatement(alice, name, person);
    psOut.println("size of store after adding duplicate triple with uniqueness ON :" + osc.size());
 
    osc.clear();       
    store.disableUniquenessConstraint();    osc.addStatement(alice, name, person);
    osc.addStatement(alice, name, person);
    psOut.println("size of store after adding duplicate triple with uniqueness OFF:" + osc.size());
 
    osc.close();
 
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example13.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example13 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX13

The expected output of the java command might appear as follows:

size of store after adding duplicate triple with uniqueness ON :1
size of store after adding duplicate triple with uniqueness OFF:2
<O%!-- class="sect2" -->

8.10.14 Example14.java: Query Timeout and Parallel Execution [Advanced]

Example 8-17 shows the Example14.java file, an advanced example that specifies a query timeout and parallel execution (TIMEOUT and DOP options).

Example 8-17 Query Timeout and Parallel Execution [Advanced]

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example14
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);   
 
    OracleSailConnection osc = store.getConnection();
    ValueFactory f = osc.getValueFactory();
 
    // create some resources and literals to make statements out of
    URI alice = f.createURI("http://example.org/people/alice");
    URI name = f.createURI("http://example.org/ontology/name");
    URI person = f.createURI("http://example.org/ontology/Person");
    Literal alicesName = f.createLiteral("Alice");
 
    osc.addStatement(alice, RDF.TYPE, person);
    osc.addStatement(alice, name, alicesName);
    osc.commit();
 
    osc.analyze();
    osc.analyzeApplicationTable();
 
    // Run a query while specifying parallel execution (through DOP=2)
    // and max query execution time (through tmieout=3)
    String queryString =
      " PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#dop=2,timeout=3> " +
      " SELECT * WHERE {?x ?p ?y} ";
 
    TupleQuery tupleQuery = osc.asRepositoryConnection().prepareTupleQuery(
        QueryLanguage.SPARQL, queryString);
 
    TupleQueryResult result = tupleQuery.evaluate();
    try {
      while (result.hasNext()) {
        BindingSet bindingSet = result.next();
        psOut.println("value of x: " + bindingSet.getValue("x"));
      }
    }
    finally {
      result.close();
    }
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example14.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example14 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX14

The expected output of the java command might appear as follows:

value of x: http://example.org/people/alice
value of x: http://example.org/people/alice

8.10.15 Example15.java: Get COUNT of Matches [Advanced]

Example 8-18 shows the Example15.java file, an advanced example that returns the total number (COUNT) of matches (and only of matches).

Example 8-18 Get COUNT of Matches [Advanced]

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example15
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);   
 
    OracleSailConnection osc = store.getConnection();
    ValueFactory f = osc.getValueFactory();
 
    // create some resources and literals to make statements out of
    URI alice = f.createURI("http://example.org/people/alice");
    URI name = f.createURI("http://example.org/ontology/name");
    URI person = f.createURI("http://example.org/ontology/Person");
    Literal alicesName = f.createLiteral("Alice");
 
    osc.addStatement(alice, RDF.TYPE, person);
    osc.addStatement(alice, name, alicesName);
    osc.commit();
 
    osc.analyze();
    osc.analyzeApplicationTable();
 
    // Run a query and only return the number of matches (the count!)
    String queryString =
      " PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#count_only> " +
      " SELECT ?totalCount WHERE {?s ?p ?y} ";
 
    TupleQuery tupleQuery = osc.asRepositoryConnection().prepareTupleQuery(
        QueryLanguage.SPARQL, queryString);
 
    TupleQueryResult result = tupleQuery.evaluate();
    try {
      if (result.hasNext()) {
        BindingSet bindingSet = result.next();
        psOut.println("number of matches: " + bindingSet.getValue("totalCount"));
      }
    }
    finally {
      result.close();
    }
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example15.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example15 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX15

The expected output of the java command might appear as follows:

number of matches: "2"^^<http://www.w3.org/2001/XMLSchema#integer>

8.10.16 Example16.java: Specify Bind Variable for Constant in Query Pattern [Advanced]

Example 8-19 shows the Example16.java file, an advanced example that specifies a bind variable for a constant in the SPARQL query pattern.

Example 8-19 Specify Bind Variable for Constant in Query Pattern [Advanced]

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example16
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);   
 
    OracleSailConnection osc = store.getConnection();
    ValueFactory f = osc.getValueFactory();
 
    // create some resources and literals to make statements out of
    URI alice = f.createURI("http://example.org/people/alice");
    URI name = f.createURI("http://example.org/ontology/name");
    URI person = f.createURI("http://example.org/ontology/Person");
    Literal alicesName = f.createLiteral("Alice");
 
    osc.addStatement(alice, RDF.TYPE, person);
    osc.addStatement(alice, name, alicesName);
    osc.commit();
 
    osc.analyze();
    osc.analyzeApplicationTable();
 
    String queryString =
      " PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#use_bind_var(1)> " +
      " SELECT ?p WHERE { <http://example.org/people/alice> <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> ?p} ";
 
    TupleQuery tupleQuery = osc.asRepositoryConnection().prepareTupleQuery(
        QueryLanguage.SPARQL, queryString);
 
    TupleQueryResult result = tupleQuery.evaluate();
    try {
      if (result.hasNext()) {
        BindingSet bindingSet = result.next();
        psOut.println("solution " + bindingSet.toString());
      }
    }
    finally {
      result.close();
    }
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example16.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example16 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX16

The expected output of the java command might appear as follows:

solution [p=http://example.org/ontology/Person]

8.10.17 Example17.java: Specify Bind Variable for Constant in Different Position in Query Pattern [Advanced]

Example 8-20 shows the Example17.java file, an advanced example that specifies a bind variable for a constant in a different position in the SPARQL query pattern than in Example16.java (Example 8-19).

Example 8-20 Specify Bind Variable for Constant in Different Position in Query Pattern [Advanced]

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example17
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);   
 
    OracleSailConnection osc = store.getConnection();
    ValueFactory f = osc.getValueFactory();
 
    // create some resources and literals to make statements out of
    URI alice = f.createURI("http://example.org/people/alice");
    URI name = f.createURI("http://example.org/ontology/name");
    URI person = f.createURI("http://example.org/ontology/Person");
    Literal alicesName = f.createLiteral("Alice");
 
    osc.addStatement(alice, RDF.TYPE, person);
    osc.addStatement(alice, name, alicesName);
    osc.commit();
 
    osc.analyze();
    osc.analyzeApplicationTable();
 
    String queryString =
      " PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#use_bind_var(2)> " +
      " SELECT ?s WHERE { ?s <http://example.org/ontology/name> \"Alice\" } ";
 
    TupleQuery tupleQuery = osc.asRepositoryConnection().prepareTupleQuery(
        QueryLanguage.SPARQL, queryString);
 
    TupleQueryResult result = tupleQuery.evaluate();
    try {
      if (result.hasNext()) {
        BindingSet bindingSet = result.next();
        psOut.println("solution " + bindingSet.toString());
      }
    }
    finally {
      result.close();
    }
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example17.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example17 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX17

The expected output of the java command might appear as follows:

solution [s=http://example.org/people/alice]

8.10.18 Example18.java: Build URIs from Internal Numeric IDs [Advanced]

Example 8-21 shows the Example18.java file, an advanced example that builds URIs from internal numeric IDs.

Example 8-21 Build URIs from Internal Numeric IDs [Advanced]

import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.ValueFactory;
import info.aduna.iteration.CloseableIteration;
 
import java.io.*;
import java.sql.SQLException;
 
import oracle.spatial.rdf.client.sesame.*;
import org.openrdf.rio.RDFFormat;
import org.openrdf.sail.SailConnection;
import org.openrdf.sail.SailException;
import org.openrdf.model.vocabulary.RDFS;
import org.openrdf.repository.RepositoryConnection;
import org.openrdf.repository.sail.SailRepository;
import org.openrdf.repository.RepositoryException;
import org.openrdf.rio.RDFParseException;
import org.openrdf.model.Literal;
import org.openrdf.query.BindingSet;
import org.openrdf.query.QueryLanguage;
import org.openrdf.query.TupleQuery;
import org.openrdf.query.TupleQueryResult;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.query.QueryEvaluationException;
import org.openrdf.query.MalformedQueryException;
 
public class Example18
{
  public static void main(String[] args) throws SQLException, SailException,
    RepositoryException, IOException, RDFParseException, QueryEvaluationException,
    MalformedQueryException
  {
    PrintStream psOut = System.out;
    String jdbcUrl  = args[0];
    String user     = args[1];
    String password = args[2];
    String model    = args[3];
 
    OraclePool op = new OraclePool(
        OraclePool.getOracleDataSource(jdbcUrl, user, password));
    OracleSailStore store = new OracleSailStore(op, model);   
 
    OracleSailConnection osc = store.getConnection();
    ValueFactory f = osc.getValueFactory();
 
    // create some resources and literals to make statements out of
    URI alice = f.createURI("http://example.org/people/alice");
    URI name = f.createURI("http://example.org/ontology/name");
    URI person = f.createURI("http://example.org/ontology/Person");
    Literal alicesName = f.createLiteral("Alice");
 
    osc.addStatement(alice, RDF.TYPE, person);
    osc.addStatement(alice, name, alicesName);
    osc.commit();
 
    osc.analyze();
    osc.analyzeApplicationTable();
 
    String queryString =
      " PREFIX ORACLE_SEM_FS_NS: <http://oracle.com/semtech#build-uri-for-id(?p)> " +
      " SELECT ?p WHERE { <http://example.org/people/alice> <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> ?p} ";
 
    TupleQuery tupleQuery = osc.asRepositoryConnection().prepareTupleQuery(
        QueryLanguage.SPARQL, queryString);
 
    TupleQueryResult result = tupleQuery.evaluate();
    try {
      if (result.hasNext()) {
        BindingSet bindingSet = result.next();
        psOut.println("solution " + bindingSet.toString());
      }
    }
    finally {
      result.close();
    }
 
    osc.close();
    store.shutDown();
    op.close();
  }
}

To compile this example, enter the following command:

usr/local/packages/jdk16/bin/javac -classpath $CP Example18.java

To run this example, enter a command in the following format:

/usr/local/packages/jdk16/bin/java -classpath $CP Example18 jdbc:oracle:thin:@localhost:1521:ORCL scott <password> EX18

The expected output of the java command might appear as follows:

solution [p=rdfvid:428072448720587401]
PK|ƼOOPK .Aoa,mimetypePK.A^)b]:iTunesMetadata.plistPK.AYuMETA-INF/container.xmlPK.AX$ OEBPS/sem_apis_ref.htmPK.A[pTOqOEBPS/cover.htmPK.AǤ OEBPS/sem_prtref.htmPK.Az>.OEBPS/title.htmPK.Aph(/// AOEBPS/loe.htmPK.A ]OEBPS/sem_jena.htmPK.AARo['V'`OEBPS/sem_glossary.htmPK.AfLiGiOEBPS/sem_enable.htmPK.AwGMMcOEBPS/fine_grained_acc.htmPK.A{OEBPS/sem_perf_ref.htmPK.A }D-?-OEBPS/skos.htmPK.AŠQLOEBPS/preface.htmPK.A—*  OEBPS/index.htmPK.A aw r ~ OEBPS/sem_prtusage.htmPK.Ag 9 OEBPS/img/inferencing.gifPK.Ajmm!r OEBPS/img/spgateway_browse_1a.jpgPK.A5ٛ& OEBPS/img/obiee_impmeta_datasource.jpgPK.AHPP OEBPS/img/spgateway_index_2.jpgPK.A@P mOEBPS/img/shortest_a_to_d.jpgPK.As;ZsOEBPS/img/cancer_ontology.gifPK.AZzR#OEBPS/img/spgateway_sparql_mgmt.jpgPK.Ac\H `#OEBPS/img/spgateway_browse_1.jpgPK.ATC% OEBPS/img/obiee_impmeta_metatypes.jpgPK.AЮp OEBPS/img/wm_rdf_rules_index.gifPK.AWu;OEBPS/img/spgateway_xslt.jpgPK.Ac= ND'"OEBPS/img/obiee_impmeta_metaobjects.jpgPK.A/))vOEBPS/img/sem_overview.gifPK.AehD#ޠOEBPS/img/spgateway_index_1.jpgPK.APƥjRGR N2OEBPS/img/spgateway_browse_3.jpgPK.AC}"OEBPS/img/family.gifPK.AUqM>% !"OEBPS/img_text/spgateway_xslt.htmPK.Av$w"OEBPS/img_text/spgateway_index_2.htmPK.ALq$}x"њ"OEBPS/img_text/cancer_ontology.htmPK.A8$"OEBPS/img_text/spgateway_index_1.htmPK.AtL%Ţ"OEBPS/img_text/spgateway_browse_3.htmPK.AsK"OEBPS/img_text/family.htmPK.Aú_to%"OEBPS/img_text/spgateway_browse_1.htmPK.A7& ,'(ί"OEBPS/img_text/spgateway_sparql_mgmt.htmPK.AsYj"P"OEBPS/img_text/shortest_a_to_d.htmPK.Af2A<&r"OEBPS/img_text/spgateway_browse_1a.htmPK.A<ٿ"OEBPS/img_text/sem_overview.htmPK.Aתhc%0"OEBPS/img_text/wm_rdf_rules_index.htmPK.A"OEBPS/img_text/inferencing.htmPK.Aƒ(#, "OEBPS/img_text/obiee_impmeta_metaobjects.htmPK.A;,gb+"OEBPS/img_text/obiee_impmeta_datasource.htmPK.A"*K"OEBPS/img_text/obiee_impmeta_metatypes.htmPK.AJ>B9B"OEBPS/sem_rdfctx_ref.htmPK.ALC ##OEBPS/toc.ncxPK.AMN33d1#OEBPS/content.opfPK.Aj wx s Ge#OEBPS/lof.htmPK.A_ o#OEBPS/dcommon/prodbig.gifPK.AY@ Ov#OEBPS/dcommon/doclib.gifPK.APp yyw#OEBPS/dcommon/oracle-logo.jpgPK.A#OEBPS/dcommon/contbig.gifPK.A#OEBPS/dcommon/darbbook.cssPK.AMά""!3#OEBPS/dcommon/O_signature_clr.JPGPK.APz ^$OEBPS/dcommon/feedbck2.gifPK.A-$OEBPS/dcommon/feedback.gifPK.Aː5"$OEBPS/dcommon/booklist.gifPK.AN61.$$OEBPS/dcommon/cpyr.htmPK.A!:3.6$OEBPS/dcommon/masterix.gifPK.AeӺ1,#8$OEBPS/dcommon/doccd.cssPK.A7 :$OEBPS/dcommon/larrow.gifPK.A#<$OEBPS/dcommon/indxicon.gifPK.AS'"+?$OEBPS/dcommon/leftnav.gifPK.Ahu,@$OEBPS/dcommon/uarrow.gifPK.Al-OJC$OEBPS/dcommon/oracle.gifPK.A(IL$OEBPS/dcommon/index.gifPK.AGC M$OEBPS/dcommon/bookbig.gifPK.AJV^W$OEBPS/dcommon/rarrow.gifPK.A枰pkY$OEBPS/dcommon/mix.gifPK.Ao"nR M }\$OEBPS/dcommon/doccd_epub.jsPK.Av I g$OEBPS/dcommon/toc.gifPK.A r~$eh$OEBPS/dcommon/topnav.gifPK.A1FAi$OEBPS/dcommon/prodicon.gifPK.A3( # ]m$OEBPS/dcommon/bp_layout.cssPK.Ax[?:z$OEBPS/dcommon/bookicon.gifPK.Ap*c^U$OEBPS/dcommon/conticon.gifPK.Aʍ$OEBPS/dcommon/blafdoc.cssPK.A+&$OEBPS/dcommon/rightnav.gifPK.Aje88r$OEBPS/dcommon/oracle-small.JPGPK.Aއ{&!$OEBPS/dcommon/help.gifPK.A.$OEBPS/sdo_rdf_concepts.htmPK.Aa!!(OEBPS/indexing_for_docs.htmPK.A7, ]aXa)OEBPS/sdo_rdf_newfeat.htmPK.A)C *OEBPS/toc.htmPK.Ao_2-*OEBPS/sem_ols_ref.htmPK.A