blob: 1210a3fd065ea4830aa4f3eb8c3de8db05ec2e80 [file] [log] [blame]
Ian Maxonb57cd732024-04-01 16:39:29 -07001<!DOCTYPE html>
2<!--
3 | Generated by Apache Maven Doxia Site Renderer 1.8.1 from target/generated-site/markdown/feeds.md at 2024-04-01
4 | Rendered using Apache Maven Fluido Skin 1.7
5-->
6<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
7 <head>
8 <meta charset="UTF-8" />
9 <meta name="viewport" content="width=device-width, initial-scale=1.0" />
10 <meta name="Date-Revision-yyyymmdd" content="20240401" />
11 <meta http-equiv="Content-Language" content="en" />
12 <title>AsterixDB &#x2013; Data Ingestion with Feeds</title>
13 <link rel="stylesheet" href="./css/apache-maven-fluido-1.7.min.css" />
14 <link rel="stylesheet" href="./css/site.css" />
15 <link rel="stylesheet" href="./css/print.css" media="print" />
16 <script type="text/javascript" src="./js/apache-maven-fluido-1.7.min.js"></script>
17
18 </head>
19 <body class="topBarDisabled">
20 <div class="container-fluid">
21 <div id="banner">
22 <div class="pull-left"><a href="./" id="bannerLeft"><img src="images/asterixlogo.png" alt="AsterixDB"/></a></div>
23 <div class="pull-right"></div>
24 <div class="clear"><hr/></div>
25 </div>
26
27 <div id="breadcrumbs">
28 <ul class="breadcrumb">
29 <li id="publishDate">Last Published: 2024-04-01</li>
30 <li id="projectVersion" class="pull-right">Version: 0.9.9</li>
31 <li class="pull-right"><a href="index.html" title="Documentation Home">Documentation Home</a></li>
32 </ul>
33 </div>
34 <div class="row-fluid">
35 <div id="leftColumn" class="span2">
36 <div class="well sidebar-nav">
37 <ul class="nav nav-list">
38 <li class="nav-header">Get Started - Installation</li>
39 <li><a href="ncservice.html" title="Option 1: using NCService"><span class="none"></span>Option 1: using NCService</a></li>
40 <li><a href="ansible.html" title="Option 2: using Ansible"><span class="none"></span>Option 2: using Ansible</a></li>
41 <li><a href="aws.html" title="Option 3: using Amazon Web Services"><span class="none"></span>Option 3: using Amazon Web Services</a></li>
42 <li class="nav-header">AsterixDB Primer</li>
43 <li><a href="sqlpp/primer-sqlpp.html" title="Using SQL++"><span class="none"></span>Using SQL++</a></li>
44 <li class="nav-header">Data Model</li>
45 <li><a href="datamodel.html" title="The Asterix Data Model"><span class="none"></span>The Asterix Data Model</a></li>
46 <li class="nav-header">Queries</li>
47 <li><a href="sqlpp/manual.html" title="The SQL++ Query Language"><span class="none"></span>The SQL++ Query Language</a></li>
48 <li><a href="SQLPP.html" title="Raw SQL++ Grammar"><span class="none"></span>Raw SQL++ Grammar</a></li>
49 <li><a href="sqlpp/builtins.html" title="Builtin Functions"><span class="none"></span>Builtin Functions</a></li>
50 <li class="nav-header">API/SDK</li>
51 <li><a href="api.html" title="HTTP API"><span class="none"></span>HTTP API</a></li>
52 <li><a href="csv.html" title="CSV Output"><span class="none"></span>CSV Output</a></li>
53 <li class="nav-header">Advanced Features</li>
54 <li><a href="aql/externaldata.html" title="Accessing External Data"><span class="none"></span>Accessing External Data</a></li>
55 <li class="active"><a href="#"><span class="none"></span>Data Ingestion with Feeds</a></li>
56 <li><a href="udf.html" title="User Defined Functions"><span class="none"></span>User Defined Functions</a></li>
57 <li><a href="sqlpp/filters.html" title="Filter-Based LSM Index Acceleration"><span class="none"></span>Filter-Based LSM Index Acceleration</a></li>
58 <li><a href="sqlpp/fulltext.html" title="Support of Full-text Queries"><span class="none"></span>Support of Full-text Queries</a></li>
59 <li><a href="sqlpp/similarity.html" title="Support of Similarity Queries"><span class="none"></span>Support of Similarity Queries</a></li>
60 <li><a href="geo/quickstart.html" title="GIS Support Overview"><span class="none"></span>GIS Support Overview</a></li>
61 <li><a href="geo/functions.html" title="GIS Functions"><span class="none"></span>GIS Functions</a></li>
62 <li><a href="interval_join.html" title="Support of Interval Joins"><span class="none"></span>Support of Interval Joins</a></li>
63 <li><a href="spatial_join.html" title="Support of Spatial Joins"><span class="none"></span>Support of Spatial Joins</a></li>
64 <li><a href="sqlpp/arrayindex.html" title="Support of Array Indexes"><span class="none"></span>Support of Array Indexes</a></li>
65 <li class="nav-header">Deprecated</li>
66 <li><a href="aql/primer.html" title="AsterixDB Primer: Using AQL"><span class="none"></span>AsterixDB Primer: Using AQL</a></li>
67 <li><a href="aql/manual.html" title="Queries: The Asterix Query Language (AQL)"><span class="none"></span>Queries: The Asterix Query Language (AQL)</a></li>
68 <li><a href="aql/builtins.html" title="Queries: Builtin Functions (AQL)"><span class="none"></span>Queries: Builtin Functions (AQL)</a></li>
69</ul>
70 <hr />
71 <div id="poweredBy">
72 <div class="clear"></div>
73 <div class="clear"></div>
74 <div class="clear"></div>
75 <div class="clear"></div>
76<a href="./" title="AsterixDB" class="builtBy"><img class="builtBy" alt="AsterixDB" src="images/asterixlogo.png" /></a>
77 </div>
78 </div>
79 </div>
80 <div id="bodyColumn" class="span10" >
81<!--
82 ! Licensed to the Apache Software Foundation (ASF) under one
83 ! or more contributor license agreements. See the NOTICE file
84 ! distributed with this work for additional information
85 ! regarding copyright ownership. The ASF licenses this file
86 ! to you under the Apache License, Version 2.0 (the
87 ! "License"); you may not use this file except in compliance
88 ! with the License. You may obtain a copy of the License at
89 !
90 ! http://www.apache.org/licenses/LICENSE-2.0
91 !
92 ! Unless required by applicable law or agreed to in writing,
93 ! software distributed under the License is distributed on an
94 ! "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
95 ! KIND, either express or implied. See the License for the
96 ! specific language governing permissions and limitations
97 ! under the License.
98 !-->
99<h1>Data Ingestion with Feeds</h1>
100<div class="section">
101<h2><a name="Table_of_Contents"></a><a name="atoc" id="#toc">Table of Contents</a></h2>
102<ul>
103
104<li><a href="#Introduction">Introduction</a></li>
105<li><a href="#FeedAdapters">Feed Adapters</a></li>
106<li><a href="#FeedPolicies">Feed Policies</a><!--
107! Licensed to the Apache Software Foundation (ASF) under one
108! or more contributor license agreements. See the NOTICE file
109! distributed with this work for additional information
110! regarding copyright ownership. The ASF licenses this file
111! to you under the Apache License, Version 2.0 (the
112! "License"); you may not use this file except in compliance
113! with the License. You may obtain a copy of the License at
114!
115! http://www.apache.org/licenses/LICENSE-2.0
116!
117! Unless required by applicable law or agreed to in writing,
118! software distributed under the License is distributed on an
119! "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
120! KIND, either express or implied. See the License for the
121! specific language governing permissions and limitations
122! under the License.
123!--></li>
124</ul></div>
125<div class="section">
126<h2><a name="Introduction">Introduction</a></h2>
127<p>In this document, we describe the support for data ingestion in AsterixDB. Data feeds are a new mechanism for having continuous data arrive into a BDMS from external sources and incrementally populate a persisted dataset and associated indexes. We add a new BDMS architectural component, called a data feed, that makes a Big Data system the caretaker for functionality that used to live outside, and we show how it improves users&#x2019; lives and system performance.</p></div>
128<div class="section">
129<h2><a name="Feed_Adapters"></a><a name="FeedAdapters">Feed Adapters</a></h2>
130<p>The functionality of establishing a connection with a data source and receiving, parsing and translating its data into ADM objects (for storage inside AsterixDB) is contained in a feed adapter. A feed adapter is an implementation of an interface and its details are specific to a given data source. An adapter may optionally be given parameters to configure its runtime behavior. Depending upon the data transfer protocol/APIs offered by the data source, a feed adapter may operate in a push or a pull mode. Push mode involves just one initial request by the adapter to the data source for setting up the connection. Once a connection is authorized, the data source &#x201c;pushes&#x201d; data to the adapter without any subsequent requests by the adapter. In contrast, when operating in a pull mode, the adapter makes a separate request each time to receive data. AsterixDB currently provides built-in adapters for several popular data sources such as Twitter and RSS feeds. AsterixDB additionally provides a generic socket-based adapter that can be used to ingest data that is directed at a prescribed socket.</p>
131<p>In this tutorial, we shall describe building two example data ingestion pipelines that cover the popular scenarios of ingesting data from (a) Twitter (b) RSS (c) Socket Feed source.</p>
132<div class="section">
133<div class="section">
134<h4><a name="Ingesting_Twitter_Stream"></a>Ingesting Twitter Stream</h4>
135<p>We shall use the built-in push-based Twitter adapter. As a pre-requisite, we must define a Tweet using the AsterixDB Data Model (ADM) and the query language SQL++. Given below are the type definitions in SQL++ that create a Tweet datatype which is representative of a real tweet as obtained from Twitter.</p>
136
137<div>
138<div>
139<pre class="source"> drop dataverse feeds if exists;
140
141 create dataverse feeds;
142 use feeds;
143
144 create type TwitterUser as closed {
145 screen_name: string,
146 lang: string,
147 friends_count: int32,
148 statuses_count: int32
149 };
150
151 create type Tweet as open {
152 id: int64,
153 user: TwitterUser
154 };
155
156 create dataset Tweets (Tweet) primary key id;
157</pre></div></div>
158
159<p>We also create a dataset that we shall use to persist the tweets in AsterixDB. Next we make use of the <tt>create feed</tt> SQL++ statement to define our example data feed.</p>
160<div class="section">
161<h5><a name="Using_the_.E2.80.9Cpush_twitter.E2.80.9D_feed_adapter"></a>Using the &#x201c;push_twitter&#x201d; feed adapter</h5>
162<p>The &#x201c;push_twitter&#x201d; adapter requires setting up an application account with Twitter. To retrieve tweets, Twitter requires registering an application. Registration involves providing a name and a brief description for the application. Each application has associated OAuth authentication credentials that include OAuth keys and tokens. Accessing the Twitter API requires providing the following.</p>
163<ol style="list-style-type: decimal">
164
165<li>Consumer Key (API Key)</li>
166<li>Consumer Secret (API Secret)</li>
167<li>Access Token</li>
168<li>Access Token Secret</li>
169</ol>
170<p>The &#x201c;push_twitter&#x201d; adapter takes as configuration the above mentioned parameters. End users are required to obtain the above authentication credentials prior to using the &#x201c;push_twitter&#x201d; adapter. For further information on obtaining OAuth keys and tokens and registering an application with Twitter, please visit <a class="externalLink" href="http://apps.twitter.com">http://apps.twitter.com</a>.</p>
171<p>Note that AsterixDB uses the Twitter4J API for getting data from Twitter. Due to a license conflict, Apache AsterixDB cannot ship the Twitter4J library. To use the Twitter adapter in AsterixDB, please download the necessary dependencies (<tt>twitter4j-core-4.0.x.jar</tt> and <tt>twitter4j-stream-4.0.x.jar</tt>) and drop them into the <tt>repo/</tt> directory before AsterixDB starts.</p>
172<p>Given below is an example SQL++ statement that creates a feed called &#x201c;TwitterFeed&#x201d; by using the &#x201c;push_twitter&#x201d; adapter.</p>
173
174<div>
175<div>
176<pre class="source"> use feeds;
177
178 create feed TwitterFeed with {
179 &quot;adapter-name&quot;: &quot;push_twitter&quot;,
180 &quot;type-name&quot;: &quot;Tweet&quot;,
181 &quot;format&quot;: &quot;twitter-status&quot;,
182 &quot;consumer.key&quot;: &quot;************&quot;,
183 &quot;consumer.secret&quot;: &quot;************&quot;,
184 &quot;access.token&quot;: &quot;**********&quot;,
185 &quot;access.token.secret&quot;: &quot;*************&quot;
186 };
187</pre></div></div>
188
189<p>It is required that the above authentication parameters are provided valid. Note that the <tt>create feed</tt> statement does not initiate the flow of data from Twitter into the AsterixDB instance. Instead, the <tt>create feed</tt> statement only results in registering the feed with the instance. The flow of data along a feed is initiated when it is connected to a target dataset using the connect feed statement and activated using the start feed statement.</p>
190<p>The Twitter adapter also supports several Twitter streaming APIs as follow:</p>
191<ol style="list-style-type: decimal">
192
193<li>Track filter <tt>&quot;keywords&quot;: &quot;AsterixDB, Apache&quot;</tt></li>
194<li>Locations filter <tt>&quot;locations&quot;: &quot;-29.7, 79.2, 36.7, 72.0; -124.848974,-66.885444, 24.396308, 49.384358&quot;</tt></li>
195<li>Language filter <tt>&quot;language&quot;: &quot;en&quot;</tt></li>
196<li>Filter level <tt>&quot;filter-level&quot;: &quot;low&quot;</tt></li>
197</ol>
198<p>An example of Twitter adapter tracking tweets with keyword &#x201c;news&#x201d; can be described using following ddl:</p>
199
200<div>
201<div>
202<pre class="source"> use feeds;
203
204 create feed TwitterFeed with {
205 &quot;adapter-name&quot;: &quot;push_twitter&quot;,
206 &quot;type-name&quot;: &quot;Tweet&quot;,
207 &quot;format&quot;: &quot;twitter-status&quot;,
208 &quot;consumer.key&quot;: &quot;************&quot;,
209 &quot;consumer.secret&quot;: &quot;************&quot;,
210 &quot;access.token&quot;: &quot;**********&quot;,
211 &quot;access.token.secret&quot;: &quot;*************&quot;,
212 &quot;keywords&quot;: &quot;news&quot;
213 };
214</pre></div></div>
215
216<p>For more details about these APIs, please visit <a class="externalLink" href="https://dev.twitter.com/streaming/overview/request-parameters">https://dev.twitter.com/streaming/overview/request-parameters</a></p></div></div>
217<div class="section">
218<h4><a name="Lifecycle_of_a_Feed"></a>Lifecycle of a Feed</h4>
219<p>A feed is a logical artifact that is brought to life (i.e., its data flow is initiated) only when it is activated using the <tt>start feed</tt> statement. Before we active a feed, we need to designate the dataset where the data to be persisted using <tt>connect feed</tt> statement. Subsequent to a <tt>connect feed</tt> statement, the feed is said to be in the connected state. After that, <tt>start feed</tt> statement will activate the feed, and start the dataflow from feed to its connected dataset. Multiple feeds can simultaneously be connected to a dataset such that the contents of the dataset represent the union of the connected feeds. Also one feed can be simultaneously connected to multiple target datasets.</p>
220
221<div>
222<div>
223<pre class="source"> use feeds;
224
225 connect feed TwitterFeed to dataset Tweets;
226
227 start feed TwitterFeed;
228</pre></div></div>
229
230<p>The <tt>connect feed</tt> statement above directs AsterixDB to persist the data from <tt>TwitterFeed</tt> feed into the <tt>Tweets</tt> dataset. The <tt>start feed</tt> statement will activate the feed and start the dataflow. If it is required (by the high-level application) to also retain the raw tweets obtained from Twitter, the end user may additionally choose to connect TwitterFeed to a different dataset.</p>
231<p>Let the feed run for a minute, then run the following query to see the latest tweets that are stored into the data set.</p>
232
233<div>
234<div>
235<pre class="source"> use feeds;
236
237 select * from Tweets limit 10;
238</pre></div></div>
239
240<p>The dataflow of data from a feed can be terminated explicitly by <tt>stop feed</tt> statement.</p>
241
242<div>
243<div>
244<pre class="source"> use feeds;
245
246 stop feed TwitterFeed;
247</pre></div></div>
248
249<p>The <tt>disconnnect statement</tt> can be used to disconnect the feed from certain dataset.</p>
250
251<div>
252<div>
253<pre class="source"> use feeds;
254
255 disconnect feed TwitterFeed from dataset Tweets;
256</pre></div></div>
257</div></div>
258<div class="section">
259<h3><a name="Ingesting_with_Other_Adapters"></a>Ingesting with Other Adapters</h3>
260<p>AsterixDB has several builtin feed adapters for data ingestion. User can also implement their own adapters and plug them into AsterixDB. Here we introduce <tt>socket_adapter</tt> and <tt>localfs</tt> feed adapter that cover most of the common application scenarios.</p>
261<div class="section">
262<div class="section">
263<h5><a name="Using_the_.E2.80.9Csocket_adapter.E2.80.9D_feed_adapter"></a>Using the &#x201c;socket_adapter&#x201d; feed adapter</h5>
264<p><tt>socket_adapter</tt> feed opens a web socket on the given node which allows user to push data into AsterixDB directly. Here is an example:</p>
265
266<div>
267<div>
268<pre class="source"> drop dataverse feeds if exists;
269 create dataverse feeds;
270 use feeds;
271
272 create type TestDataType as open {
273 screenName: string
274 };
275
276 create dataset TestDataset(TestDataType) primary key screenName;
277
278 create feed TestSocketFeed with {
279 &quot;adapter-name&quot;: &quot;socket_adapter&quot;,
280 &quot;sockets&quot;: &quot;127.0.0.1:10001&quot;,
281 &quot;address-type&quot;: &quot;IP&quot;,
282 &quot;type-name&quot;: &quot;TestDataType&quot;,
283 &quot;format&quot;: &quot;adm&quot;
284 };
285
286 connect feed TestSocketFeed to dataset TestDataset;
287
288 use feeds;
289 start feed TestSocketFeed;
290</pre></div></div>
291
292<p>The above statements create a socket feed which is listening to &#x201c;10001&#x201d; port of the host machine. This feed accepts data records in &#x201c;adm&#x201d; format. As an example, you can download the sample dataset <a href="../data/chu.adm">Chirp Users</a> and push them line by line into the socket feed using any socket client you like. Following is a socket client example in Python:</p>
293
294<div>
295<div>
296<pre class="source"> from socket import socket
297
298 ip = '127.0.0.1'
299 port1 = 10001
300 filePath = 'chu.adm'
301
302 sock1 = socket()
303 sock1.connect((ip, port1))
304
305 with open(filePath) as inputData:
306 for line in inputData:
307 sock1.sendall(line)
308 sock1.close()
309</pre></div></div>
310</div></div>
311<div class="section">
312<h4><a name="Using_the_.E2.80.9Clocalfs.E2.80.9D_feed_adapter"></a>Using the &#x201c;localfs&#x201d; feed adapter</h4>
313<p><tt>localfs</tt> adapter enables data ingestion from local file system. It allows user to feed data records on local disk into a dataset. A DDL example for creating a <tt>localfs</tt> feed is given as follow:</p>
314
315<div>
316<div>
317<pre class="source"> use feeds;
318
319 create type TestDataType as open {
320 screenName: string
321 };
322
323 create dataset TestDataset(TestDataType) primary key screenName;
324
325 create feed TestFileFeed with {
326 &quot;adapter-name&quot;: &quot;localfs&quot;,
327 &quot;type-name&quot;: &quot;TestDataType&quot;,
328 &quot;path&quot;: &quot;HOSTNAME://LOCAL_FILE_PATH&quot;,
329 &quot;format&quot;: &quot;adm&quot;
330 };
331
332 connect feed TestFileFeed to dataset TestDataset;
333
334 start feed TestFileFeed;
335</pre></div></div>
336
337<p>Similar to previous examples, we need to define the datatype and dataset this feed uses. The &#x201c;path&#x201d; parameter refers to the local data file that we want to ingest data from. <tt>HOSTNAME</tt> can either be the IP address or node name of the machine which holds the file. <tt>LOCAL_FILE_PATH</tt> indicates the absolute path to the file on that machine. Similarly to <tt>socket_adapter</tt>, this feed takes <tt>adm</tt> formatted data records.</p></div></div>
338<div class="section">
339<h3><a name="Datatype_for_feed_and_target_dataset"></a>Datatype for feed and target dataset</h3>
340<p>The &#x201c;type-name&#x201d; parameter in create feed statement defines the <tt>datatype</tt> of the datasource. In most use cases, feed will have the same <tt>datatype</tt> as the target dataset. However, if we want to perform certain preprocess before the data records gets into the target dataset (append autogenerated key, apply user defined functions, etc.), we will need to define the datatypes for feed and dataset separately.</p>
341<div class="section">
342<h4><a name="Ingestion_with_autogenerated_key"></a>Ingestion with autogenerated key</h4>
343<p>AsterixDB supports using autogenerated uuid as the primary key for dataset. When we use this feature, we will need to define a datatype with the primary key field, and specify that field to be autogenerated when creating the dataset. Use that same datatype in feed definition will cause a type discrepancy since there is no such field in the datasource. Thus, we will need to define two separate datatypes for feed and dataset:</p>
344
345<div>
346<div>
347<pre class="source"> use feeds;
348
349 create type DBLPFeedType as closed {
350 dblpid: string,
351 title: string,
352 authors: string,
353 misc: string
354 }
355
356 create type DBLPDataSetType as open {
357 id: uuid,
358 dblpid: string,
359 title: string,
360 authors: string,
361 misc: string
362 }
363 create dataset DBLPDataset(DBLPDataSetType) primary key id autogenerated;
364
365 create feed DBLPFeed with {
366 &quot;adapter-name&quot;: &quot;socket_adapter&quot;,
367 &quot;sockets&quot;: &quot;127.0.0.1:10001&quot;,
368 &quot;address-type&quot;: &quot;IP&quot;,
369 &quot;type-name&quot;: &quot;DBLPFeedType&quot;,
370 &quot;format&quot;: &quot;adm&quot;
371 };
372
373 connect feed DBLPFeed to dataset DBLPDataset;
374
375 start feed DBLPFeed;
376</pre></div></div>
377</div></div></div>
378<div class="section">
379<h2><a name="Policies_for_Feed_Ingestion"></a><a name="FeedPolicies">Policies for Feed Ingestion</a></h2>
380<p>Multiple feeds may be concurrently operational on an AsterixDB cluster, each competing for resources (CPU cycles, network bandwidth, disk IO) to maintain pace with their respective data sources. As a data management system, AsterixDB is able to manage a set of concurrent feeds and make dynamic decisions related to the allocation of resources, resolving resource bottlenecks and the handling of failures. Each feed has its own set of constraints, influenced largely by the nature of its data source and the applications that intend to consume and process the ingested data. Consider an application that intends to discover the trending topics on Twitter by analyzing tweets that are being processed. Losing a few tweets may be acceptable. In contrast, when ingesting from a data source that provides a click-stream of ad clicks, losing data would translate to a loss of revenue for an application that tracks revenue by charging advertisers per click.</p>
381<p>AsterixDB allows a data feed to have an associated ingestion policy that is expressed as a collection of parameters and associated values. An ingestion policy dictates the runtime behavior of the feed in response to resource bottlenecks and failures. AsterixDB provides a set of policies that help customize the system&#x2019;s runtime behavior when handling excess objects.</p>
382<div class="section">
383<div class="section">
384<h4><a name="Policies"></a>Policies</h4>
385<ul>
386
387<li>
388
389<p><i>Spill</i>: Objects that cannot be processed by an operator for lack of resources (referred to as excess objects hereafter) should be persisted to the local disk for deferred processing.</p>
390</li>
391<li>
392
393<p><i>Discard</i>: Excess objects should be discarded.</p>
394</li>
395</ul>
396<p>Note that the end user may choose to form a custom policy. For example, it is possible in AsterixDB to create a custom policy that spills excess objects to disk and subsequently resorts to throttling if the spillage crosses a configured threshold. In all cases, the desired ingestion policy is specified as part of the <tt>connect feed</tt> statement or else the &#x201c;Basic&#x201d; policy will be chosen as the default.</p>
397
398<div>
399<div>
400<pre class="source"> use feeds;
401
402 connect feed TwitterFeed to dataset Tweets using policy Basic;
403</pre></div></div></div></div></div>
404 </div>
405 </div>
406 </div>
407 <hr/>
408 <footer>
409 <div class="container-fluid">
410 <div class="row-fluid">
411<div class="row-fluid">Apache AsterixDB, AsterixDB, Apache, the Apache
412 feather logo, and the Apache AsterixDB project logo are either
413 registered trademarks or trademarks of The Apache Software
414 Foundation in the United States and other countries.
415 All other marks mentioned may be trademarks or registered
416 trademarks of their respective owners.
417 </div>
418 </div>
419 </div>
420 </footer>
421 </body>
422</html>