... |
{code} |
http://localhost:8080/odata/northwind.1/NW.customers(1234)/NW.orders?$filter=orderdate > '12-31-2012' |
http://localhost:8080/odata/northwind.1/NW.customers(1234)/NW.orders?$filter=orderdate gt datetime'2012-12-31T21:23:38Z' |
{code} |
... |
{code:lang=SQL} |
SELECT * FROM NW.orders o join NW.customers c join o.customer_id = c.id where c.id=1234 and o.orderdate > '12-31-2012' {ts '2012-12-31 21:23:38'} |
{code} |
|
{tip:title=More Comprehensive Documentation about ODATA} |
For detailed protocol access you can read the specification at [http://odata.org]. You can also read this very useful web resource [for an example|http://msdn.microsoft.com/en-us/library/ff478141.aspx] access of an OData server. |
{tip} |
|
h3. How to update your data? Using OData protocol user can issue CREATE/UPDATE/DELETE operations along with READ operations as shown above. These operations use different HTTP methods to accomplish this INSERT/CREATE is accomplished through HTTP method "POST", below is an example request. {code} POST /service.svc/Customers HTTP/1.1 Host: host Content-Type: application/atom+xml Accept: application/atom+xml Content-Length: nnn <?xml version="1.0" encoding="utf-8"?> <entry xml:base="http://host/service.svc/" xmlns:d="http://schemas.microsoft.com/ado/2007/08/dataservices" xmlns:m="http://schemas.microsoft.com/ado/2007/08/dataservices/metadata" xmlns="http://www.w3.org/2005/Atom"> <id>http://host/service.svc/Customers('ASDFG')</id> <title type="text" /> <updated>2008-12-07T8:00:00Z</updated> <author> <name /> </author> <link rel="http://schemas.microsoft.com/ado/2007/08/dataservices/related/Orders" href="Orders(1)" /> <link rel="http://schemas.microsoft.com/ado/2007/08/dataservices/related/Orders" href="Orders(2)" /> <content type="application/xml"> <m:properties> <d:CustomerID>ASDFG</d:CustomerID> <d:CompanyName>Contoso Widgets</d:CompanyName> <d:Address> <d:Street>58 Contoso St</d:Street> <d:City>Seattle</d:City> </d:Address> </m:properties> </content> </entry> {code} To do a UPDATE, HTTP "PUT" method is used, a sample request to update Customer looks like {code} PUT /service.svc/Customers('ALFKI') HTTP/1.1 Host: host Content-Type: application/atom+xml Accept: application/atom+xml Content-Length: nnn DataServiceVersion: 1.0 MaxDataServiceVersion: 3.0 Prefer: return-content <?xml version="1.0" encoding="utf-8"?> <entry xmlns:d="http://schemas.microsoft.com/ado/2007/08/dataservices" xmlns:m="http://schemas.microsoft.com/ado/2007/08/dataservices/metadata" xmlns="http://www.w3.org/2005/Atom"> <content type="application/xml"> <m:properties> <d:CustomerID>ALFKI</d:CustomerID> <d:CompanyName>Updated Company Name</d:CompanyName> <d:Address> <d:Street>Updated Street</d:Street> </d:Address> </m:properties> </content> </entry> {code} For DELETE operation use HTTP "DELETE" method, a sample request looks like {code} DELETE /service.svc/Customers('ALFKI') HTTP/1.1 Host: host Content-Type: application/atom+xml Accept: application/atom+xml Content-Length: nnn DataServiceVersion: 1.0 {code} |
h3. Security |
... |
The Open Data Protocol (OData) is a Web protocol for querying and updating data that provides a way to unlock your data and free it from silos that exist in applications today. OData does this by applying and building upon Web technologies such as HTTP, Atom Publishing Protocol (AtomPub) and JSON to provide access to information from a variety of applications, services, and stores. The protocol emerged from experiences implementing AtomPub clients and servers in a variety of products over the past several years. OData is used to expose and access information from a variety of sources including, but not limited to, relational databases, file systems, content management systems and traditional Web sites.
OData is consistent with the way the Web works - it makes a deep commitment to URIs for resource identification and commits to an HTTP-based, uniform interface for interacting with those resources (just like the Web). This commitment to core Web principles allows OData to enable a new level of data integration and interoperability across a broad range of clients, servers, services, and tools.
copied from http://odata.org
When a user successfully deploys a VDB into a Teiid Server the OData protocol support is implicitly provided by the Teiid server without any further configuration. OData support is currently not available in the Teiid Embedded profile. Teiid makes use of JBoss AS and its already configured RestEasy and OData4J libraries to provide Rest based access to the VDB. The access would be similar to accessing to any web resources deployed on JBoss AS. OData support is implemented and deployed through a single WAR file across the Teiid system. A user can access this WAR file at <jboss-as>/modules/org/jboss/teiid/deployments/odata-8.6.0.Final.war.
For example, if you have a vdb by name northwind deployed and that vdb has a table customers in a model called NW then you can access that table as
http://localhost:8080/odata/northwind.1/NW.customers
this would be akin to, making a JDBC/ODBC connection and issuing a SQL call as
SELECT * FROM NW.customers
Note that you need to fully qualify the table name along with the model name. Also, use correct case (upper or lower) as used in the VDB. |
the returned results from OData query can be in Atom/AtomPub XML format or JSON format. By default AtomPub based XML result is returned.
Seeing "EdmEntitySet Not Found" error?When you issue the above query are you seeing a message similar to below? <error xmlns="http://schemas.microsoft.com/ado/2007/08/dataservices/metadata"> <code>NotFoundException</code> <message lang="en-US">EdmEntitySet NW.customer is not found</message> </error> Then, it means that either you supplied the model-name.table-name combination wrong, check spelling and case. Or your table did not have any PRIMARY KEY or UNIQUE KEY(s) on them. Since OData access is more key oriented, it is * MANDATORY* that every table Teiid exposes through OData interface must have a PK or at least one UNIQUE key. Note that Teiid does support composite PRIMARY KEY too. |
Not seeing all the rows? See the configuration section below for more details. Issue another call as http://localhost:8080/odata/northwind.1/NW.customers?$skiptoken=256 |
Users can also submit criteria with along their query to filter the results
http://localhost:8080/odata/northwind.1/NW.customers?$filter=name eq 'bob'
this would be akin to, making a JDBC/ODBC connection and issuing a SQL call as
SELECT * FROM NW.customers where name = 'bob'
The power of OData querying comes from navigations from one entity to another, similar the foreign key relationships in the relational databases. For example, if customers table had a association with orders table, where say customers table had primary key of id then, user can issue a query like
http://localhost:8080/odata/northwind.1/NW.customers(1234)/NW.orders?$filter=orderdate gt datetime'2012-12-31T21:23:38Z'
this would be akin to, making a JDBC/ODBC connection and issuing a SQL call as
SELECT * FROM NW.orders o join NW.customers c join o.customer_id = c.id where c.id=1234 and o.orderdate > {ts '2012-12-31 21:23:38'}
More Comprehensive Documentation about ODATA For detailed protocol access you can read the specification at http://odata.org. You can also read this very useful web resource for an example access of an OData server. |
Using OData protocol user can issue CREATE/UPDATE/DELETE operations along with READ operations as shown above. These operations use different HTTP methods to accomplish this
INSERT/CREATE is accomplished through HTTP method "POST", below is an example request.
POST /service.svc/Customers HTTP/1.1 Host: host Content-Type: application/atom+xml Accept: application/atom+xml Content-Length: nnn <?xml version="1.0" encoding="utf-8"?> <entry xml:base="http://host/service.svc/" xmlns:d="http://schemas.microsoft.com/ado/2007/08/dataservices" xmlns:m="http://schemas.microsoft.com/ado/2007/08/dataservices/metadata" xmlns="http://www.w3.org/2005/Atom"> <id>http://host/service.svc/Customers('ASDFG')</id> <title type="text" /> <updated>2008-12-07T8:00:00Z</updated> <author> <name /> </author> <link rel="http://schemas.microsoft.com/ado/2007/08/dataservices/related/Orders" href="Orders(1)" /> <link rel="http://schemas.microsoft.com/ado/2007/08/dataservices/related/Orders" href="Orders(2)" /> <content type="application/xml"> <m:properties> <d:CustomerID>ASDFG</d:CustomerID> <d:CompanyName>Contoso Widgets</d:CompanyName> <d:Address> <d:Street>58 Contoso St</d:Street> <d:City>Seattle</d:City> </d:Address> </m:properties> </content> </entry>
To do a UPDATE, HTTP "PUT" method is used, a sample request to update Customer looks like
PUT /service.svc/Customers('ALFKI') HTTP/1.1 Host: host Content-Type: application/atom+xml Accept: application/atom+xml Content-Length: nnn DataServiceVersion: 1.0 MaxDataServiceVersion: 3.0 Prefer: return-content <?xml version="1.0" encoding="utf-8"?> <entry xmlns:d="http://schemas.microsoft.com/ado/2007/08/dataservices" xmlns:m="http://schemas.microsoft.com/ado/2007/08/dataservices/metadata" xmlns="http://www.w3.org/2005/Atom"> <content type="application/xml"> <m:properties> <d:CustomerID>ALFKI</d:CustomerID> <d:CompanyName>Updated Company Name</d:CompanyName> <d:Address> <d:Street>Updated Street</d:Street> </d:Address> </m:properties> </content> </entry>
For DELETE operation use HTTP "DELETE" method, a sample request looks like
DELETE /service.svc/Customers('ALFKI') HTTP/1.1 Host: host Content-Type: application/atom+xml Accept: application/atom+xml Content-Length: nnn DataServiceVersion: 1.0
Currently by default OData access is secured using HTTPBasic authentication. The user will be authenticated against Teiid's default security domain "teiid-security" and makes use of the security role odata. However, if you wish to change the security domain, manually edit the web.xml file WAR file in <modules>/org/jboss/teiid/main/deployments directory. In future versions more WS-Security based configurations will be provided.
The OData WAR file can be configured with following properties in the web.xml file.
Property Name | Description | Default Value |
---|---|---|
batch-size | Number of rows to send back each time, -1 returns all rows | 256 |
skiptoken-cache-time | Time interval between the results being recycled/expired between $skiptoken requests | 300000 |
local-transport-name | Teiid Local transport name for connection | odata |
Teiid OData server, implements cursoring logic when the result rows exceed the configured batch size. On every request, only batch-size number of rows are returned. Each such request is considered an active cursor, with a specified amount of idle time specified by skip-token-cache-time. After the cursor is timed out, the cursor will be closed and remaining results will be cleaned up, and will no longer be available for further queries. Since there is no session based tracking of these cursors, if the request for skiptoken comes after the expired time, the original query will be executed again and tries to reposition the cursor to relative absolute potion, however the results are not guaranteed to be same as the underlying sources may have been updated with new information meanwhile.
The following feature limitations currently apply.
OData access is really where the user comes in, depending upon your programming model and needs there are various ways you write your access layer into OData. The following are some suggestions:
OData defines its schema using Conceptual Schema Definition Language (CSDL). Every VDB, that is deployed in an ACTIVE state in Teiid server exposes its metadata in CSDL format. For example if you want retrieve metadata for your vdb northwind, you need to issue a query like
http://localhost:8080/odata/northwind/$metadata
Since OData schema model is not a relational schema model, Teiid uses the following semantics to map its relational schema model to OData schema model.
Relational Entity | Mapped OData Entity |
---|---|
Model Name | Schema Namespace, EntityContainer Name |
Table/View | EntityType, EntitySet |
Table Columns | EntityType's Properties |
Primary Key | EntityType's Key Properties |
Foreign Key | Navigation Property on EntityType, Association, AssosiationSet |
Procedure | FunctionImport |
Procedure's Table Return | ComplexType |
Teiid by design does not define any "embedded" ComplexType in the EntityType.
Teiid does not define any one EntityContainer that resulted from different vdb models as a default container, so all entities must be accessed using full path to them. |