Cloudera Apache Impala Guide
Cloudera Apache Impala Guide
Important Notice
© 2010-2018 Cloudera, Inc. All rights reserved.
Hadoop and the Hadoop elephant logo are trademarks of the Apache Software
Foundation. All other trademarks, registered trademarks, product names and company
names or logos mentioned in this document are the property of their respective owners.
Reference to any products, services, processes or other information, by trade name,
trademark, manufacturer, supplier or otherwise does not constitute or imply
endorsement, sponsorship or recommendation thereof by us.
Complying with all applicable copyright laws is the responsibility of the user. Without
limiting the rights under copyright, no part of this document may be reproduced, stored
in or introduced into a retrieval system, or transmitted in any form or by any means
(electronic, mechanical, photocopying, recording, or otherwise), or for any purpose,
without the express written permission of Cloudera.
The information in this document is subject to change without notice. Cloudera shall
not be liable for any damages resulting from technical errors or omissions which may
be present in this document, or from use of this document.
Cloudera, Inc.
395 Page Mill Road
Palo Alto, CA 94306
info@cloudera.com
US: 1-888-789-1488
Intl: 1-650-362-0488
www.cloudera.com
Release Information
Installing Impala.....................................................................................................27
What is Included in an Impala Installation.........................................................................................................27
Installing Impala with Cloudera Manager..........................................................................................................27
Installing Impala without Cloudera Manager.....................................................................................................28
Managing Impala...................................................................................................30
Post-Installation Configuration for Impala.........................................................................................................30
Configuring Impala to Work with ODBC.............................................................................................................31
Downloading the ODBC Driver.............................................................................................................................................32
Configuring the ODBC Port..................................................................................................................................................32
Example of Setting Up an ODBC Application for Impala......................................................................................................32
Notes about JDBC and ODBC Interaction with Impala SQL Features...................................................................................33
Configuring Impala to Work with JDBC..............................................................................................................34
Configuring the JDBC Port....................................................................................................................................................34
Choosing the JDBC Driver.....................................................................................................................................................34
Enabling Impala JDBC Support on Client Systems................................................................................................................35
Establishing JDBC Connections.............................................................................................................................................36
Notes about JDBC and ODBC Interaction with Impala SQL Features...................................................................................37
Kudu Considerations for DML Statements...........................................................................................................................37
Upgrading Impala...................................................................................................38
Upgrading Impala through Cloudera Manager - Parcels....................................................................................38
Upgrading Impala through Cloudera Manager - Packages.................................................................................39
Upgrading Impala without Cloudera Manager...................................................................................................40
Starting Impala......................................................................................................42
Starting Impala through Cloudera Manager.......................................................................................................42
Starting Impala from the Command Line...........................................................................................................42
Modifying Impala Startup Options.....................................................................................................................43
Configuring Impala Startup Options through Cloudera Manager.......................................................................................43
Configuring Impala Startup Options through the Command Line.......................................................................................43
Checking the Values of Impala Configuration Options........................................................................................................46
Startup Options for impalad Daemon..................................................................................................................................46
Startup Options for statestored Daemon.............................................................................................................................46
Startup Options for catalogd Daemon.................................................................................................................................46
Impala Tutorials.....................................................................................................47
Tutorials for Getting Started...............................................................................................................................47
Explore a New Impala Instance............................................................................................................................................47
Load CSV Data from Local Files............................................................................................................................................52
Point an Impala Table at Existing Data Files........................................................................................................................53
Describe the Impala Table....................................................................................................................................................55
Query the Impala Table........................................................................................................................................................55
Data Loading and Querying Examples.................................................................................................................................56
Advanced Tutorials.............................................................................................................................................58
Attaching an External Partitioned Table to an HDFS Directory Structure............................................................................58
Switching Back and Forth Between Impala and Hive...........................................................................................................60
Cross Joins and Cartesian Products with the CROSS JOIN Operator....................................................................................61
Dealing with Parquet Files with Unknown Schema............................................................................................62
Impala Administration...........................................................................................75
Admission Control and Query Queuing..............................................................................................................75
Overview of Impala Admission Control................................................................................................................................75
How Impala Admission Control Relates to YARN.................................................................................................................76
How Impala Schedules and Enforces Limits on Concurrent Queries....................................................................................76
How Admission Control works with Impala Clients (JDBC, ODBC, HiveServer2)..................................................................77
Configuring Admission Control............................................................................................................................................77
Guidelines for Using Admission Control...............................................................................................................................82
Integrated Resource Management with YARN...................................................................................................83
The Llama Daemon..............................................................................................................................................................83
How Resource Limits Are Enforced.......................................................................................................................................84
impala-shell Query Options for Resource Management......................................................................................................84
Limitations of Resource Management for Impala...............................................................................................................84
Setting Timeout Periods for Daemons, Queries, and Sessions...........................................................................84
Increasing the Statestore Timeout.......................................................................................................................................84
Setting the Idle Query and Idle Session Timeouts for impalad.............................................................................................84
Cancelling a Query...............................................................................................................................................................85
Using Impala through a Proxy for High Availability............................................................................................85
Overview of Proxy Usage and Load Balancing for Impala...................................................................................................86
Special Proxy Considerations for Clusters Using Kerberos...................................................................................................87
Example of Configuring HAProxy Load Balancer for Impala................................................................................................88
Managing Disk Space for Impala Data................................................................................................................89
Impala Security......................................................................................................91
Security Guidelines for Impala...........................................................................................................................91
Securing Impala Data and Log Files....................................................................................................................92
Installation Considerations for Impala Security..................................................................................................93
Securing the Hive Metastore Database..............................................................................................................93
Securing the Impala Web User Interface............................................................................................................93
Configuring TLS/SSL for Impala..........................................................................................................................94
Using Cloudera Manager.....................................................................................................................................................94
Using the Command Line.....................................................................................................................................................95
Using TLS/SSL with Business Intelligence Tools....................................................................................................................95
Enabling Sentry Authorization for Impala..........................................................................................................95
The Sentry Privilege Model..................................................................................................................................................96
Starting the impalad Daemon with Sentry Authorization Enabled......................................................................................96
Using Impala with the Sentry Service (CDH 5.1 or higher only)...........................................................................................97
Using Impala with the Sentry Policy File..............................................................................................................................98
Setting Up Schema Objects for a Secure Impala Deployment............................................................................................102
Privilege Model and Object Hierarchy...............................................................................................................................102
Debugging Failed Sentry Authorization Requests..............................................................................................................105
Managing Sentry for Impala through Cloudera Manager.................................................................................................106
The DEFAULT Database in a Secure Deployment...............................................................................................................106
Impala Authentication......................................................................................................................................106
Enabling Kerberos Authentication for Impala....................................................................................................................106
Enabling LDAP Authentication for Impala..........................................................................................................................109
Using Multiple Authentication Methods with Impala........................................................................................................111
Configuring Impala Delegation for Hue and BI Tools.........................................................................................................112
Auditing Impala Operations.............................................................................................................................112
Durability and Performance Considerations for Impala Auditing......................................................................................112
Format of the Audit Log Files.............................................................................................................................................113
Which Operations Are Audited..........................................................................................................................................113
Reviewing the Audit Logs...................................................................................................................................................114
Viewing Lineage Information for Impala Data..................................................................................................114
Troubleshooting Impala.......................................................................................578
Troubleshooting Impala SQL Syntax Issues......................................................................................................578
Troubleshooting I/O Capacity Problems...........................................................................................................578
Impala Troubleshooting Quick Reference........................................................................................................579
Impala Web User Interface for Debugging.......................................................................................................580
Debug Web UI for impalad................................................................................................................................................581
Note: Impala graduated from the Apache Incubator on November 15, 2017. In places where the
documentation formerly referred to “Cloudera Impala”, now the official name is “Apache Impala”.
Impala Benefits
Impala provides:
• Familiar SQL interface that data scientists and analysts already know.
• Ability to query high volumes of data (“big data”) in Apache Hadoop.
• Distributed queries in a cluster environment, for convenient scaling and to make use of cost-effective commodity
hardware.
• Ability to share data files between different components with no copy or export/import step; for example, to
write with Pig, transform with Hive and query with Impala. Impala can read from and write to Hive tables, enabling
simple data interchange using Impala for analytics on Hive-produced data.
• Single system for big data processing and analytics, so customers can avoid costly modeling and ETL just for
analytics.
• Hive Metastore - Stores information about the data available to Impala. For example, the metastore lets Impala
know what databases are available and what the structure of those databases is. As you create, drop, and alter
schema objects, load data into tables, and so on through Impala SQL statements, the relevant metadata changes
are automatically broadcast to all Impala nodes by the dedicated catalog service introduced in Impala 1.2.
• Impala - This process, which runs on DataNodes, coordinates and executes queries. Each instance of Impala can
receive, plan, and coordinate queries from Impala clients. Queries are distributed among Impala nodes, and these
nodes then act as workers, executing parallel query fragments.
• HBase and HDFS - Storage for data to be queried.
Queries executed using Impala are handled as follows:
1. User applications send SQL queries to Impala through ODBC or JDBC, which provide standardized querying
interfaces. The user application may connect to any impalad in the cluster. This impalad becomes the coordinator
for the query.
2. Impala parses the query and analyzes it to determine what tasks need to be performed by impalad instances
across the cluster. Execution is planned for optimal efficiency.
3. Services such as HDFS and HBase are accessed by local impalad instances to provide data.
4. Each impalad returns data to the coordinating impalad, which sends these results to the client.
Most considerations for load balancing and high availability apply to the impalad daemon. The statestored and
catalogd daemons do not have special requirements for high availability, because problems with those daemons do
not result in data loss. If those daemons become unavailable due to an outage on a particular host, you can stop the
Impala service, delete the Impala StateStore and Impala Catalog Server roles, add the roles on a different host, and
restart the Impala service.
Related information:
Scalability Considerations for the Impala Statestore on page 516, Modifying Impala Startup Options on page 43, Starting
Impala on page 42, Increasing the Statestore Timeout on page 84, Ports Used by Impala on page 583
By default, the metadata loading and caching on startup happens asynchronously, so Impala can begin accepting
requests promptly. To enable the original behavior, where Impala waited until all metadata was loaded before accepting
any requests, set the catalogd configuration option --load_catalog_in_background=false.
Most considerations for load balancing and high availability apply to the impalad daemon. The statestored and
catalogd daemons do not have special requirements for high availability, because problems with those daemons do
not result in data loss. If those daemons become unavailable due to an outage on a particular host, you can stop the
Impala service, delete the Impala StateStore and Impala Catalog Server roles, add the roles on a different host, and
restart the Impala service.
Note:
In Impala 1.2.4 and higher, you can specify a table name with INVALIDATE METADATA after the table
is created in Hive, allowing you to make individual tables visible to Impala without doing a full reload
of the catalog metadata. Impala 1.2.4 also includes other changes to make the metadata broadcast
mechanism faster and more responsive, especially during Impala startup. See New Features in Impala
Version 1.2.4 on page 617 for details.
Related information: Modifying Impala Startup Options on page 43, Starting Impala on page 42, Ports Used by Impala
on page 583
• In Impala 1.2 and higher, UDFs let you perform custom comparisons and transformation logic during SELECT and
INSERT...SELECT statements.
For users coming to Impala from traditional database or data warehousing backgrounds, the following aspects of the
SQL dialect might require some learning and practice for you to become proficient in the Hadoop environment:
• Impala SQL is focused on queries and includes relatively little DML. There is no UPDATE or DELETE statement.
Stale data is typically discarded (by DROP TABLE or ALTER TABLE ... DROP PARTITION statements) or
replaced (by INSERT OVERWRITE statements).
• All data creation is done by INSERT statements, which typically insert data in bulk by querying from other tables.
There are two variations, INSERT INTO which appends to the existing data, and INSERT OVERWRITE which
replaces the entire contents of a table or partition (similar to TRUNCATE TABLE followed by a new INSERT).
Although there is an INSERT ... VALUES syntax to create a small number of values in a single statement, it is
far more efficient to use the INSERT ... SELECT to copy and transform large amounts of data from one table
to another in a single operation.
• You often construct Impala table definitions and data files in some other environment, and then attach Impala so
that it can run real-time queries. The same data files and table metadata are shared with other components of
the Hadoop ecosystem. In particular, Impala can access tables created by Hive or data inserted by Hive, and Hive
can access tables and data produced by Impala. Many other Hadoop components can write files in formats such
as Parquet and Avro, that can then be queried by Impala.
• Because Hadoop and Impala are focused on data warehouse-style operations on large data sets, Impala SQL
includes some idioms that you might find in the import utilities for traditional database systems. For example,
you can create a table that reads comma-separated or tab-separated text files, specifying the separator in the
CREATE TABLE statement. You can create external tables that read existing data files but do not move or transform
them.
• Because Impala reads large quantities of data that might not be perfectly tidy and predictable, it does not require
length constraints on string data types. For example, you can define a database column as STRING with unlimited
length, rather than CHAR(1) or VARCHAR(64). (Although in Impala 2.0 and later, you can also use
length-constrained CHAR and VARCHAR types.)
Related information: Impala SQL Language Reference on page 116, especially Impala SQL Statements on page 214 and
Impala Built-In Functions on page 338
METADATA by itself retrieves metadata for all the tables tracked by the metastore. If you know that only specific tables
have been changed outside of Impala, you can issue REFRESH table_name for each affected table to only retrieve
the latest metadata for those tables.
Impala Requirements
To perform as expected, Impala depends on the availability of the software, hardware, and configurations described
in the following sections.
Note:
Installing and configuring a Hive metastore is an Impala requirement. Impala does not work
without the metastore database. For the process of installing and configuring the metastore, see
Installing Impala on page 27.
Always configure a Hive metastore service rather than connecting directly to the metastore
database. The Hive metastore service is required to interoperate between possibly different levels
of metastore APIs used by CDH and Impala, and avoids known issues with connecting directly to
the metastore database. The Hive metastore service is set up for you by default if you install
through Cloudera Manager 4.5 or higher.
A summary of the metastore installation process is as follows:
• Install a MySQL or PostgreSQL database. Start the database if it is not started after installation.
• Download the MySQL connector or the PostgreSQL connector and place it in the
/usr/share/java/ directory.
• Use the appropriate command line tool for your database to create the metastore database.
• Use the appropriate command line tool for your database to grant privileges for the metastore
database to the hive user.
• Modify hive-site.xml to include information matching your particular database: its URL,
user name, and password. You will copy the hive-site.xml file to the Impala Configuration
Directory later in the Impala installation process.
• Optional: Hive. Although only the Hive metastore database is required for Impala to function, you might install
Hive on some client machines to create and load data into tables that use certain file formats. See How Impala
Works with Hadoop File Formats on page 527 for details. Hive does not need to be installed on the same DataNodes
as Impala; it just needs access to the same metastore database.
Java Dependencies
Although Impala is primarily written in C++, it does use Java to communicate with various Hadoop components:
• The officially supported JVM for Impala is the Oracle JVM. Other JVMs might cause issues, typically resulting in a
failure at impalad startup. In particular, the JamVM used by default on certain levels of Ubuntu systems can
cause impalad to fail to start.
• Internally, the impalad daemon relies on the JAVA_HOME environment variable to locate the system Java libraries.
Make sure the impalad service is not run from an environment with an incorrect setting for this variable.
• All Java dependencies are packaged in the impala-dependencies.jar file, which is located at
/usr/lib/impala/lib/. These map to everything that is built under fe/target/dependency.
In the majority of cases, this automatic detection works correctly. If you need to explicitly set the hostname, do so by
setting the --hostname flag.
Hardware Requirements
During join operations, portions of data from each joined table are loaded into memory. Data sets can be very large,
so ensure your hardware has sufficient memory to accommodate the joins you anticipate completing.
While requirements vary according to data set size, the following is generally recommended:
• CPU - Impala version 2.2 and higher uses the SSSE3 instruction set, which is included in newer processors.
Note: This required level of processor is the same as in Impala version 1.x. The Impala 2.0 and
2.1 releases had a stricter requirement for the SSE4.1 instruction set, which has now been relaxed.
• Memory - 128 GB or more recommended, ideally 256 GB or more. If the intermediate results during query
processing on a particular node exceed the amount of memory available to Impala on that node, the query writes
temporary work data to disk, which can lead to long query times. Note that because the work is parallelized, and
intermediate results for aggregate queries are typically smaller than the original data, Impala can query and join
tables that are much larger than the memory available on an individual node.
• Storage - DataNodes with 12 or more disks each. I/O speeds are often the limiting factor for disk performance
with Impala. Ensure that you have sufficient disk space to store the data Impala will be querying.
For correct file deletion during DROP TABLE operations, Impala must be able to move files to the HDFS trashcan. You
might need to create an HDFS directory /user/impala, writeable by the impala user, so that the trashcan can be
created. Otherwise, data files might remain behind after a DROP TABLE statement.
Impala should not run as root. Best Impala performance is achieved using direct reads, but root is not permitted to
use direct reads. Therefore, running Impala as root negatively affects performance.
By default, any user can connect to Impala and access all the associated databases and tables. You can enable
authorization and authentication based on the Linux OS user who connects to the Impala server, and the associated
groups for that user. Impala Security on page 91 for details. These security features do not change the underlying file
permission requirements; the impala user still needs to be able to access the data files.
example, if you have only a few tens of megabytes of data per day, partitioning by YEAR, MONTH, and DAY columns
might be too granular. Most of your cluster might be sitting idle during queries that target a single day, or each node
might have very little work to do. Consider reducing the number of partition key columns so that each partition directory
contains several gigabytes worth of data.
For example, consider a Parquet table where each data file is 1 HDFS block, with a maximum block size of 1 GB. (In
Impala 2.0 and later, the default Parquet block size is reduced to 256 MB. For this exercise, let's assume you have
bumped the size back up to 1 GB by setting the query option PARQUET_FILE_SIZE=1g.) if you have a 10-node cluster,
you need 10 data files (up to 10 GB) to give each node some work to do for a query. But each core on each machine
can process a separate data block in parallel. With 16-core machines on a 10-node cluster, a query could process up
to 160 GB fully in parallel. If there are only a few data files per partition, not only are most cluster nodes sitting idle
during queries, so are most cores on those machines.
You can reduce the Parquet block size to as low as 128 MB or 64 MB to increase the number of files per partition and
improve parallelism. But also consider reducing the level of partitioning so that analytic queries have enough data to
work with.
Installing Impala
Impala is an open-source add-on to the Cloudera Enterprise Core that returns rapid responses to queries.
Note:
Under CDH 5, Impala is included as part of the CDH installation and no separate steps are needed.
Therefore, the instruction steps in this section apply to CDH 4 only.
Before doing the installation, ensure that you have all necessary prerequisites. See Impala Requirements on page 23
for details.
Note:
To install the latest Impala under CDH 4, upgrade Cloudera Manager to 4.8 or higher. Cloudera Manager
4.8 is the first release that can manage the Impala catalog service introduced in Impala 1.2. Cloudera
Manager 4.8 requires this service to be present, so if you upgrade to Cloudera Manager 4.8, also
upgrade Impala to the most recent version at the same time.
For information on installing Impala in a Cloudera Manager-managed environment, see Installing Impala.
Managing your Impala installation through Cloudera Manager has a number of advantages. For example, when you
make configuration changes to CDH components using Cloudera Manager, it automatically applies changes to the
copies of configuration files, such as hive-site.xml, that Impala keeps under /etc/impala/conf. It also sets up
the Hive Metastore service that is required for Impala running under CDH 4.1.
In some cases, depending on the level of Impala, CDH, and Cloudera Manager, you might need to add particular
component configuration details in some of the free-form option fields on the Impala configuration pages within
Cloudera Manager. In Cloudera Manager 4, these fields are labelled Safety Valve; in Cloudera Manager 5, they are
called Advanced Configuration Snippet.
<property>
<name>hive.metastore.uris</name>
<value>thrift://metastore_server_host:9083</value>
</property>
<property>
<name>hive.metastore.client.socket.timeout</name>
<value>3600</value>
<description>MetaStore Client socket timeout in seconds</description>
</property>
3. (Optional) If you installed the full Hive component on any host, you can verify that the metastore is configured
properly by starting the Hive console and querying for the list of available tables. Once you confirm that the console
starts, exit the console to continue the installation:
$ hive
Hive history file=/tmp/root/hive_job_log_root_201207272011_678722950.txt
hive> show tables;
table1
table2
hive> quit;
$
4. Confirm that your package management command is aware of the Impala repository settings, as described in
Impala Requirements on page 23. (For CDH 4, this is a different repository than for CDH.) You might need to
download a repo or list file into a system directory underneath /etc.
5. Use one of the following sets of commands to install the Impala package:
Note: Cloudera recommends that you not install Impala on any HDFS NameNode. Installing
Impala on NameNodes provides no additional data locality, and executing queries with such a
configuration might cause memory contention and negatively impact the HDFS NameNode.
6. Copy the client hive-site.xml, core-site.xml, hdfs-site.xml, and hbase-site.xml configuration files
to the Impala configuration directory, which defaults to /etc/impala/conf. Create this directory if it does not
already exist.
7. Use one of the following commands to install impala-shell on the machines from which you want to issue
queries. You can install impala-shell on any supported machine that can connect to DataNodes that are running
impalad.
8. Complete any required or recommended configuration, as described in Post-Installation Configuration for Impala
on page 30. Some of these configuration changes are mandatory. (They are applied automatically when you install
using Cloudera Manager.)
Once installation and configuration are complete, see Starting Impala on page 42 for how to activate the software on
the appropriate nodes in your cluster.
If this is your first time setting up and using Impala in this cluster, run through some of the exercises in Impala Tutorials
on page 47 to verify that you can do basic operations such as creating tables and querying them.
Managing Impala
This section explains how to configure Impala to accept connections from applications that use popular programming
APIs:
• Post-Installation Configuration for Impala on page 30
• Configuring Impala to Work with ODBC on page 31
• Configuring Impala to Work with JDBC on page 34
This type of configuration is especially useful when using Impala in combination with Business Intelligence tools, which
use these standard interfaces to query different kinds of database and Big Data systems.
You can also configure these other aspects of Impala:
• Impala Security on page 91
• Modifying Impala Startup Options on page 43
Note: If you use Cloudera Manager, you can enable short-circuit reads through a checkbox in the
user interface and that setting takes effect for Impala as well.
<property>
<name>dfs.client.read.shortcircuit</name>
<value>true</value>
</property>
<property>
<name>dfs.domain.socket.path</name>
<value>/var/run/hdfs-sockets/dn</value>
</property>
<property>
<name>dfs.client.file-block-storage-locations.timeout.millis</name>
<value>10000</value>
</property>
Note: If you are also going to enable block location tracking, you can skip copying configuration
files and restarting DataNodes and go straight to Optional: Block Location Tracking. Configuring
short-circuit reads and block location tracking require the same process of copying files and
restarting services, so you can complete that process once when you have completed all
configuration changes. Whether you copy files and restart services now or during configuring
block location tracking, short-circuit reads are not enabled until you complete those final steps.
<property>
<name>dfs.datanode.hdfs-blocks-metadata.enabled</name>
<value>true</value>
</property>
2. Copy the client core-site.xml and hdfs-site.xml configuration files from the Hadoop configuration directory
to the Impala configuration directory. The default Impala configuration location is /etc/impala/conf.
3. After applying these changes, restart all DataNodes.
ODBC, the operating system, and the third-party product have all been approved for use together. Before configuring
your systems to use ODBC, download a connector. You may need to sign in and accept license agreements before
accessing the pages required for downloading ODBC connectors.
Important: As of late 2015, most business intelligence applications are certified with the 2.x ODBC
drivers. Although the instructions on this page cover both the 2.x and 1.x drivers, expect to use the
2.x drivers exclusively for most ODBC applications connecting to Impala.
$ ls -1
Cloudera-ODBC-Driver-for-Impala-Install-Guide.pdf
BI_Tool_Installer.dmg
iodbc-sdk-3.52.7-macosx-10.5.dmg
ClouderaImpalaODBC.dmg
$ open iodbc-sdk-3.52.7-macosx-10.dmg
Install the IODBC driver using its installer
$ open ClouderaImpalaODBC.dmg
Install the Cloudera ODBC Connector using its installer
$ installer_dir=$(pwd)
$ cd /opt/cloudera/impalaodbc
$ ls -1
Cloudera ODBC Driver for Impala Install Guide.pdf
Readme.txt
Setup
lib
ErrorMessages
Release Notes.txt
Tools
$ cd Setup
$ ls
odbc.ini odbcinst.ini
$ cp odbc.ini ~/.odbc.ini
$ vi ~/.odbc.ini
$ cat ~/.odbc.ini
[ODBC]
# Specify any global ODBC configuration here such as ODBC tracing.
# Values for HOST, PORT, KrbFQDN, and KrbServiceName should be set here.
# They can also be specified on the connection string.
HOST=hostname.sample.example.com
PORT=21050
Schema=default
# General settings
TSaslTransportBufSize=1000
RowsFetchedPerBlock=10000
SocketTimeout=0
StringColumnLength=32767
UseNativeQuery=0
$ pwd
/opt/cloudera/impalaodbc/Setup
$ cd $installer_dir
$ open BI_Tool_Installer.dmg
Install the BI tool using its installer
$ ls /Applications | grep BI_Tool
BI_Tool.app
$ open -a BI_Tool.app
In the BI tool, connect to a data source using port 21050
Notes about JDBC and ODBC Interaction with Impala SQL Features
Most Impala SQL features work equivalently through the impala-shell interpreter of the JDBC or ODBC APIs. The
following are some exceptions to keep in mind when switching between the interactive shell and applications using
the APIs:
Note: If your JDBC or ODBC application connects to Impala through a load balancer such as haproxy,
be cautious about reusing the connections. If the load balancer has set up connection timeout values,
either check the connection frequently so that it never sits idle longer than the load balancer timeout
value, or check the connection validity before using it and create a new one if the connection has
been closed.
• The Impala complex types (STRUCT, ARRAY, or MAP) are available in CDH 5.5 / Impala 2.3 and higher. To use these
types with JDBC requires version 2.5.28 or higher of the Cloudera JDBC Connector for Impala. To use these types
with ODBC requires version 2.5.30 or higher of the Cloudera ODBC Connector for Impala. Consider upgrading all
JDBC and ODBC drivers at the same time you upgrade from CDH 5.5 or higher.
• Although the result sets from queries involving complex types consist of all scalar values, the queries involve join
notation and column references that might not be understood by a particular JDBC or ODBC connector. Consider
defining a view that represents the flattened version of a table containing complex type columns, and pointing
the JDBC or ODBC application at the view. See Complex Types (CDH 5.5 or higher only) on page 156 for details.
Note: The latest JDBC driver, corresponding to Hive 0.13, provides substantial performance
improvements for Impala queries that return large result sets. Impala 2.0 and later are compatible
with the Hive 0.13 driver. If you already have an older JDBC driver installed, and are running Impala
2.0 or higher, consider upgrading to the latest Hive JDBC driver for best performance with JDBC
applications.
If you are using JDBC-enabled applications on hosts outside the CDH cluster, you cannot use the CDH install procedure
on the non-CDH hosts. Install the JDBC driver on at least one CDH host using the preceding procedure. Then download
the JAR files to each client machine that will use JDBC with Impala:
commons-logging-X.X.X.jar
hadoop-common.jar
hive-common-X.XX.X-cdhX.X.X.jar
hive-jdbc-X.XX.X-cdhX.X.X.jar
hive-metastore-X.XX.X-cdhX.X.X.jar
hive-service-X.XX.X-cdhX.X.X.jar
httpclient-X.X.X.jar
httpcore-X.X.X.jar
libfb303-X.X.X.jar
libthrift-X.X.X.jar
log4j-X.X.XX.jar
slf4j-api-X.X.X.jar
slf4j-logXjXX-X.X.X.jar
To enable JDBC support for Impala on the system where you run the JDBC application:
1. Download the JAR files listed above to each client machine.
Note: For Maven users, see this sample github page for an example of the dependencies you
could add to a pom file instead of downloading the individual JARs.
2. Store the JAR files in a location of your choosing, ideally a directory already referenced in your CLASSPATH setting.
For example:
• On Linux, you might use a location such as /opt/jars/.
• On Windows, you might use a subdirectory underneath C:\Program Files.
3. To successfully load the Impala JDBC driver, client programs must be able to locate the associated JAR files. This
often means setting the CLASSPATH for the client process to include the JARs. Consult the documentation for
your JDBC client for more details on how to install new JDBC drivers, but some examples of how to set CLASSPATH
variables include:
• On Linux, if you extracted the JARs to /opt/jars/, you might issue the following command to prepend the
JAR files path to an existing classpath:
export CLASSPATH=/opt/jars/*.jar:$CLASSPATH
• On Windows, use the System Properties control panel item to modify the Environment Variables for your
system. Modify the environment variables to include the path to which you extracted the files.
Note: If the existing CLASSPATH on your client machine refers to some older version of the
Hive JARs, ensure that the new JARs are the first ones listed. Either put the new JAR files
earlier in the listings, or delete the other references to Hive JAR files.
Note: If your JDBC or ODBC application connects to Impala through a load balancer such as haproxy,
be cautious about reusing the connections. If the load balancer has set up connection timeout values,
either check the connection frequently so that it never sits idle longer than the load balancer timeout
value, or check the connection validity before using it and create a new one if the connection has
been closed.
jdbc:impala://Host:Port[/Schema];Property1=Value;Property2=Value;...
jdbc:hive2://myhost.example.com:21050/;auth=noSasl
To connect to an instance of Impala that requires Kerberos authentication, use a connection string of the form
jdbc:hive2://host:port/;principal=principal_name. The principal must be the same user principal you
used when starting Impala. For example, you might use:
jdbc:hive2://myhost.example.com:21050/;principal=impala/myhost.example.com@H2.EXAMPLE.COM
To connect to an instance of Impala that requires LDAP authentication, use a connection string of the form
jdbc:hive2://host:port/db_name;user=ldap_userid;password=ldap_password. For example, you might
use:
jdbc:hive2://myhost.example.com:21050/test_db;user=fred;password=xyz123
Note:
Currently, the Hive JDBC driver does not support connections that use both Kerberos authentication
and SSL encryption. To use both of these security features with Impala through a JDBC application,
use the Cloudera JDBC Connector as the JDBC driver.
Notes about JDBC and ODBC Interaction with Impala SQL Features
Most Impala SQL features work equivalently through the impala-shell interpreter of the JDBC or ODBC APIs. The
following are some exceptions to keep in mind when switching between the interactive shell and applications using
the APIs:
• Complex type considerations:
– Queries involving the complex types (ARRAY, STRUCT, and MAP) require notation that might not be available
in all levels of JDBC and ODBC drivers. If you have trouble querying such a table due to the driver level or
inability to edit the queries used by the application, you can create a view that exposes a “flattened” version
of the complex columns and point the application at the view. See Complex Types (CDH 5.5 or higher only)
on page 156 for details.
– The complex types available in CDH 5.5 / Impala 2.3 and higher are supported by the JDBC getColumns()
API. Both MAP and ARRAY are reported as the JDBC SQL Type ARRAY, because this is the closest matching Java
SQL type. This behavior is consistent with Hive. STRUCT types are reported as the JDBC SQL Type STRUCT.
To be consistent with Hive's behavior, the TYPE_NAME field is populated with the primitive type name for
scalar types, and with the full toSql() for complex types. The resulting type names are somewhat inconsistent,
because nested types are printed differently than top-level types. For example, the following list shows how
toSQL() for Impala types are translated to TYPE_NAME values:
Upgrading Impala
Upgrading Impala involves stopping Impala services, using your operating system's package management tool to
upgrade Impala to the latest version, and then restarting Impala services.
Note:
• Each version of CDH 5 has an associated version of Impala, When you upgrade from CDH 4 to
CDH 5, you get whichever version of Impala comes with the associated level of CDH. Depending
on the version of Impala you were running on CDH 4, this could install a lower level of Impala on
CDH 5. For example, if you upgrade to CDH 5.0 from CDH 4 plus Impala 1.4, the CDH 5.0 installation
comes with Impala 1.3. Always check the associated level of Impala before upgrading to a specific
version of CDH 5. Where practical, upgrade from CDH 4 to the latest CDH 5, which also has the
latest Impala.
• When you upgrade Impala, also upgrade Cloudera Manager if necessary:
– Users running Impala on CDH 5 must upgrade to Cloudera Manager 5.0.0 or higher.
– Users running Impala on CDH 4 must upgrade to Cloudera Manager 4.8 or higher. Cloudera
Manager 4.8 includes management support for the Impala catalog service, and is the minimum
Cloudera Manager version you can use.
– Cloudera Manager is continually updated with configuration settings for features introduced
in the latest Impala releases.
• If you are upgrading from CDH 5 beta to CDH 5.0 production, make sure you are using the
appropriate CDH 5 repositories shown on the CDH version and packaging page, then follow the
procedures throughout the rest of this section.
• Every time you upgrade to a new major or minor Impala release, see Incompatible Changes and
Limitations in Apache Impala on page 625 in the Release Notes for any changes needed in your
source code, startup scripts, and so on.
• Also check Known Issues and Workarounds in Impala on page 635 in the Release Notes for any
issues or limitations that require workarounds.
Important: In CDH 5, there is not a separate Impala parcel; Impala is part of the main CDH 5 parcel.
Each level of CDH 5 has a corresponding version of Impala, and you upgrade Impala by upgrading CDH.
See the CDH 5 upgrade instructions and choose the instructions for parcels. The remainder of this
section only covers parcel upgrades for Impala under CDH 4.
and then remove the packages using one of the following commands:
5. Use one of the following sets of commands to update Impala shell on each node on which it is installed:
For RHEL, Oracle Linux, or CentOS systems:
2. Check if there are new recommended or required configuration settings to put into place in the configuration
files, typically under /etc/impala/conf. See Post-Installation Configuration for Impala on page 30 for settings
related to performance and scalability.
3. Use one of the following sets of commands to update Impala on each Impala node in your cluster:
For RHEL, Oracle Linux, or CentOS systems:
4. Use one of the following sets of commands to update Impala shell on each node on which it is installed:
For RHEL, Oracle Linux, or CentOS systems:
5. Depending on which release of Impala you are upgrading from, you might find that the symbolic links
/etc/impala/conf and /usr/lib/impala/sbin are missing. If so, see Known Issues and Workarounds in
Impala on page 635 for the procedure to work around this problem.
6. Restart Impala services:
a. Restart the Impala state store service on the desired nodes in your cluster. Expect to see a process named
statestored if the service started successfully.
Restart the state store service before the Impala server service to avoid “Not connected” errors when you
run impala-shell.
b. Restart the Impala catalog service on whichever host it runs on in your cluster. Expect to see a process named
catalogd if the service started successfully.
c. Restart the Impala daemon service on each node in your cluster. Expect to see a process named impalad if
the service started successfully.
Note:
If the services did not start successfully (even though the sudo service command might display
[OK]), check for errors in the Impala log file, typically in /var/log/impala.
Starting Impala
To activate Impala if it is installed but not yet started:
1. Set any necessary configuration options for the Impala services. See Modifying Impala Startup Options on page
43 for details.
2. Start one instance of the Impala statestore. The statestore helps Impala to distribute work efficiently, and to
continue running in the event of availability problems for other Impala nodes. If the statestore becomes unavailable,
Impala continues to function.
3. Start one instance of the Impala catalog service.
4. Start the main Impala service on one or more DataNodes, ideally on all DataNodes to maximize local processing
and avoid network traffic due to remote reads.
Once Impala is running, you can conduct interactive experiments using the instructions in Impala Tutorials on page 47
and try Using the Impala Shell (impala-shell Command) on page 470.
Note:
Currently, Impala UDFs and UDAs are not persisted in the metastore database. Information about
these functions is held in the memory of the catalogd daemon. You must reload them by running
the CREATE FUNCTION statements again each time you restart the catalogd daemon.
Start the Impala service on each DataNode using a command similar to the following:
Note:
Currently, Impala UDFs and UDAs are not persisted in the metastore database. Information about
these functions is held in the memory of the catalogd daemon. You must reload them by running
the CREATE FUNCTION statements again each time you restart the catalogd daemon.
IMPALA_STATE_STORE_HOST=127.0.0.1
IMPALA_STATE_STORE_PORT=24000
IMPALA_BACKEND_PORT=22000
IMPALA_LOG_DIR=/var/log/impala
IMPALA_CATALOG_SERVICE_HOST=...
IMPALA_STATE_STORE_HOST=...
export IMPALA_STATE_STORE_ARGS=${IMPALA_STATE_STORE_ARGS:- \
-log_dir=${IMPALA_LOG_DIR} -state_store_port=${IMPALA_STATE_STORE_PORT}}
IMPALA_SERVER_ARGS=" \
-log_dir=${IMPALA_LOG_DIR} \
-catalog_service_host=${IMPALA_CATALOG_SERVICE_HOST} \
-state_store_port=${IMPALA_STATE_STORE_PORT} \
-use_statestore \
-state_store_host=${IMPALA_STATE_STORE_HOST} \
-be_port=${IMPALA_BACKEND_PORT}"
export ENABLE_CORE_DUMPS=${ENABLE_COREDUMPS:-false}
To use alternate values, edit the defaults file, then restart all the Impala-related services so that the changes take
effect. Restart the Impala server using the following commands:
IMPALA_STATE_STORE_HOST=127.0.0.1
to:
IMPALA_STATE_STORE_HOST=192.168.0.27
• Catalog server address (including both the hostname and the port number). Update the value of the
IMPALA_CATALOG_SERVICE_HOST variable. Cloudera recommends the catalog server be on the same host as
the statestore. In that recommended configuration, the impalad daemon cannot refer to the catalog server using
the loopback address. If the catalog service is hosted on a machine with an IP address of 192.168.0.27, add the
following line:
IMPALA_CATALOG_SERVICE_HOST=192.168.0.27:26000
The /etc/default/impala defaults file currently does not define an IMPALA_CATALOG_ARGS environment
variable, but if you add one it will be recognized by the service startup/shutdown script. Add a definition for this
variable to /etc/default/impala and add the option -catalog_service_host=hostname. If the port is
different than the default 26000, also add the option -catalog_service_port=port.
• Memory limits. You can limit the amount of memory available to Impala. For example, to allow Impala to use no
more than 70% of system memory, change:
export IMPALA_SERVER_ARGS=${IMPALA_SERVER_ARGS:- \
-log_dir=${IMPALA_LOG_DIR} \
-state_store_port=${IMPALA_STATE_STORE_PORT} \
-use_statestore -state_store_host=${IMPALA_STATE_STORE_HOST} \
-be_port=${IMPALA_BACKEND_PORT}}
to:
export IMPALA_SERVER_ARGS=${IMPALA_SERVER_ARGS:- \
-log_dir=${IMPALA_LOG_DIR} -state_store_port=${IMPALA_STATE_STORE_PORT} \
-use_statestore -state_store_host=${IMPALA_STATE_STORE_HOST} \
-be_port=${IMPALA_BACKEND_PORT} -mem_limit=70%}
You can specify the memory limit using absolute notation such as 500m or 2G, or as a percentage of physical
memory such as 60%.
Note: Queries that exceed the specified memory limit are aborted. Percentage limits are based
on the physical memory of the machine and do not consider cgroups.
• Core dump enablement. To enable core dumps on systems not managed by Cloudera Manager, change:
export ENABLE_CORE_DUMPS=${ENABLE_COREDUMPS:-false}
to:
export ENABLE_CORE_DUMPS=${ENABLE_COREDUMPS:-true}
On systems managed by Cloudera Manager, enable the Enable Core Dump setting for the Impala service.
Note:
• The location of core dump files may vary according to your operating system configuration.
• Other security settings may prevent Impala from writing core dumps even when this option
is enabled.
• On systems managed by Cloudera Manager, the default location for core dumps is on a
temporary filesystem, which can lead to out-of-space issues if the core dumps are large,
frequent, or not removed promptly. To specify an alternative location for the core dumps,
filter the Impala configuration settings to find the core_dump_dir option, which is available
in Cloudera Manager 5.4.3 and higher. This option lets you specify a different directory for
core dumps for each of the Impala-related daemons.
• Authorization using the open source Sentry plugin. Specify the -server_name and
-authorization_policy_file options as part of the IMPALA_SERVER_ARGS and IMPALA_STATE_STORE_ARGS
settings to enable the core Impala support for authentication. See Starting the impalad Daemon with Sentry
Authorization Enabled on page 96 for details.
• Auditing for successful or blocked Impala queries, another aspect of security. Specify the
-audit_event_log_dir=directory_path option and optionally the
-max_audit_event_log_file_size=number_of_queries and -abort_on_failed_audit_event options
as part of the IMPALA_SERVER_ARGS settings, for each Impala node, to enable and customize auditing. See
Auditing Impala Operations on page 112 for details.
• Password protection for the Impala web UI, which listens on port 25000 by default. This feature involves adding
some or all of the --webserver_password_file, --webserver_authentication_domain, and
--webserver_certificate_file options to the IMPALA_SERVER_ARGS and IMPALA_STATE_STORE_ARGS
settings. See Security Guidelines for Impala on page 91 for details.
• Another setting you might add to IMPALA_SERVER_ARGS is a comma-separated list of query options and values:
-default_query_options='option=value,option=value,...'
These options control the behavior of queries performed by this impalad instance. The option values you specify
here override the default values for Impala query options, as shown by the SET statement in impala-shell.
• During troubleshooting, Cloudera Support might direct you to change other values, particularly for
IMPALA_SERVER_ARGS, to work around issues or gather debugging information.
Note:
These startup options for the impalad daemon are different from the command-line options for the
impala-shell command. For the impala-shell options, see impala-shell Configuration Options
on page 470.
Impala Tutorials
This section includes tutorial scenarios that demonstrate how to begin using Impala once the software is installed. It
focuses on techniques for loading data, because once you have some data in tables and can query that data, you can
quickly progress to more advanced Impala features.
Note:
Where practical, the tutorials take you from “ground zero” to having the desired Impala tables and
data. In some cases, you might need to download additional files from outside sources, set up additional
software components, modify commands or scripts to fit your own configuration, or substitute your
own sample data.
Before trying these tutorial lessons, install Impala using one of these procedures:
• If you already have a CDH environment set up and just need to add Impala to it, follow the installation process
described in Installing Impala on page 27. Make sure to also install the Hive metastore service if you do not already
have Hive configured.
• To set up Impala and all its prerequisites at once, in a minimal configuration that you can use for small-scale
experiments, set up the Cloudera QuickStart VM, which includes CDH and Impala on CentOS. Use this single-node
VM to try out basic SQL functionality, not anything related to performance and scalability. For more information,
see the Cloudera QuickStart VM.
+-------------------------------------------
[localhost:21000] > show databases;
+--------------------------+
| name |
+--------------------------+
| _impala_builtins |
| ctas |
| d1 |
| d2 |
| d3 |
| default |
| explain_plans |
| external_table |
| file_formats |
| tpc |
+--------------------------+
[localhost:21000] > select current_database();
+--------------------+
| current_database() |
+--------------------+
| default |
+--------------------+
[localhost:21000] > show tables;
+-------+
| name |
+-------+
| ex_t |
| t1 |
+-------+
[localhost:21000] > show tables in d3;
Once you know what tables and databases are available, you descend into a database with the USE statement. To
understand the structure of each table, you use the DESCRIBE command. Once inside a database, you can issue
statements such as INSERT and SELECT that operate on particular tables.
The following example explores a database named TPC whose name we learned in the previous example. It shows
how to filter the table names within a database based on a search string, examine the columns of a table, and run
queries to examine the characteristics of the table data. For example, for an unfamiliar table you might want to know
the number of rows, the number of different values for a column, and other properties such as whether the column
contains any NULL values. When sampling the actual data values from a table, use a LIMIT clause to avoid excessive
output if the table contains more rows or distinct values than you expect.
| Dr. |
| |
| Miss |
| Sir |
| Mrs. |
+--------------+
When you graduate from read-only exploration, you use statements such as CREATE DATABASE and CREATE TABLE
to set up your own database objects.
The following example demonstrates creating a new database holding a new table. Although the last example ended
inside the TPC database, the new EXPERIMENTS database is not nested inside TPC; all databases are arranged in a
single top-level list.
The following example creates a new table, T1. To illustrate a common mistake, it creates this table inside the wrong
database, the TPC database where the previous example ended. The ALTER TABLE statement lets you move the table
to the intended database, EXPERIMENTS, as part of a rename operation. The USE statement is always needed to switch
to a new database, and the current_database() function confirms which database the session is in, to avoid these
kinds of mistakes.
+--------------------+
| tpc |
+--------------------+
[localhost:21000] > alter table t1 rename to experiments.t1;
[localhost:21000] > use experiments;
[localhost:21000] > show tables;
+------+
| name |
+------+
| t1 |
+------+
[localhost:21000] > select current_database();
+--------------------+
| current_database() |
+--------------------+
| experiments |
+--------------------+
For your initial experiments with tables, you can use ones with just a few columns and a few rows, and text-format
data files.
Note: As you graduate to more realistic scenarios, you will use more elaborate tables with many
columns, features such as partitioning, and file formats such as Parquet. When dealing with realistic
data volumes, you will bring in data using LOAD DATA or INSERT ... SELECT statements to operate
on millions or billions of rows at once.
The following example sets up a couple of simple tables with a few rows, and performs queries involving sorting,
aggregate functions and joins.
$ whoami
cloudera
$ hdfs dfs -ls /user
Found 3 items
drwxr-xr-x - cloudera cloudera 0 2013-04-22 18:54 /user/cloudera
drwxrwx--- - mapred mapred 0 2013-03-15 20:11 /user/history
drwxr-xr-x - hue supergroup 0 2013-03-15 20:10 /user/hive
Here is some sample data, for two tables named TAB1 and TAB2.
Copy the following content to .csv files in your local filesystem:
tab1.csv:
1,true,123.123,2012-10-24 08:55:00
2,false,1243.5,2012-10-25 13:40:00
3,false,24453.325,2008-08-22 09:33:21.123
4,false,243423.325,2007-05-12 22:32:21.33454
5,true,243.325,1953-04-22 09:11:33
tab2.csv:
1,true,12789.123
2,false,1243.5
3,false,24453.325
4,false,2423.3254
5,true,243.325
60,false,243565423.325
70,true,243.325
80,false,243423.325
90,true,243.325
Put each .csv file into a separate HDFS directory using commands like the following, which use paths available in the
Impala Demo VM:
The name of each data file is not significant. In fact, when Impala examines the contents of the data directory for the
first time, it considers all files in the directory to make up the data of the table, regardless of how many files there are
or what the files are named.
To understand what paths are available within your own HDFS filesystem and what the permissions are for the various
directories and files, issue hdfs dfs -ls / and work your way down the tree doing -ls operations for the various
directories.
Use the impala-shell command to create tables, either interactively or through a SQL script.
The following example shows creating three tables. For each table, the example shows creating columns with various
attributes such as Boolean or integer types. The example also includes commands that provide information about how
the data is formatted, such as rows terminating with commas, which makes sense in the case of importing data from
a .csv file. Where we already have .csv files containing data in the HDFS directory tree, we specify the location of
the directory containing the appropriate .csv file. Impala considers all the data from all the files in that directory to
represent the data for the table.
Note: Getting through these CREATE TABLE statements successfully is an important validation step
to confirm everything is configured correctly with the Hive metastore and HDFS permissions. If you
receive any errors during the CREATE TABLE statements:
• Make sure you followed the installation instructions closely, in Installing Impala on page 27.
• Make sure the hive.metastore.warehouse.dir property points to a directory that Impala
can write to. The ownership should be hive:hive, and the impala user should also be a member
of the hive group.
• If the value of hive.metastore.warehouse.dir is different in the Cloudera Manager dialogs
and in the Hive shell, you might need to designate the hosts running impalad with the “gateway”
role for Hive, and deploy the client configuration files to those hosts.
impala-shell a *.sql file with contents similar to the following, to create an Impala table that accesses an existing
data file used by Hive.
The following examples set up 2 tables, referencing the paths and sample data supplied with the Cloudera QuickStart
VM. For historical reasons, the data physically resides in an HDFS directory tree under /user/hive, although this
particular data is entirely managed by Impala rather than Hive. When we create an external table, we specify the
directory containing one or more data files, and Impala queries the combined content of all the files inside that directory.
Here is how we examine the directories and files within the HDFS filesystem:
$ cd ~/cloudera/datasets
$ ./tpcds-setup.sh
... Downloads and unzips the kit, builds the data and loads it into HDFS ...
$ hdfs dfs -ls /user/hive/tpcds/customer
Found 1 items
-rw-r--r-- 1 cloudera supergroup 13209372 2013-03-22 18:09
/user/hive/tpcds/customer/customer.dat
$ hdfs dfs -cat /user/hive/tpcds/customer/customer.dat | more
1|AAAAAAAABAAAAAAA|980124|7135|32946|2452238|2452208|Mr.|Javier|Lewis|Y|9|12|1936|CHILE||Javie
r.Lewis@VFAxlnZEvOx.org|2452508|
2|AAAAAAAACAAAAAAA|819667|1461|31655|2452318|2452288|Dr.|Amy|Moses|Y|9|4|1966|TOGO||Amy.Moses@
Ovk9KjHH.com|2452318|
3|AAAAAAAADAAAAAAA|1473522|6247|48572|2449130|2449100|Miss|Latisha|Hamilton|N|18|9|1979|NIUE||
Latisha.Hamilton@V.com|2452313|
4|AAAAAAAAEAAAAAAA|1703214|3986|39558|2450030|2450000|Dr.|Michael|White|N|7|6|1983|MEXICO||Mic
hael.White@i.org|2452361|
5|AAAAAAAAFAAAAAAA|953372|4470|36368|2449438|2449408|Sir|Robert|Moran|N|8|5|1956|FIJI||Robert.
Moran@Hh.edu|2452469|
...
Here is a SQL script to set up Impala tables pointing to some of these data files in HDFS. (The script in the VM sets up
tables like this through Hive; ignore those tables for purposes of this demonstration.) Save the following as
customer_setup.sql:
--
-- store_sales fact table and surrounding dimension tables only
--
create database tpcds;
use tpcds;
ca_street_name string,
ca_street_type string,
ca_suite_number string,
ca_city string,
ca_county string,
ca_state string,
ca_zip string,
ca_country string,
ca_gmt_offset float,
ca_location_type string
)
row format delimited fields terminated by '|'
location '/user/hive/tpcds/customer_address';
$ impala-shell -i impala-host
Connected to localhost:21000
• Passing a single command to the impala-shell command. The query is executed, the results are returned, and
the shell exits. Make sure to quote the command, preferably with single quotation marks to avoid shell expansion
of characters such as *.
Loading Data
Loading data involves:
• Establishing a data set. The example below uses .csv files.
• Creating tables to which to load data.
• Loading the data into the tables you created.
Sample Queries
To run these sample queries, create a SQL query file query.sql, copy and paste each query into the query file, and
then run the query file using the shell. For example, to run query.sql on impala-host, you might use the command:
The examples and results below assume you have loaded the sample data into the tables as described above.
Example: Examining Contents of Tables
Let's start by verifying that the tables do contain the data we expect. Because Impala often deals with tables containing
millions or billions of rows, when examining tables of unknown size, include the LIMIT clause to avoid huge amounts
of unnecessary output, as in the final query. (If your interactive query starts displaying an unexpected volume of data,
press Ctrl-C in impala-shell to cancel the query.)
Results:
+----+-------+------------+-------------------------------+
| id | col_1 | col_2 | col_3 |
+----+-------+------------+-------------------------------+
| 1 | true | 123.123 | 2012-10-24 08:55:00 |
| 2 | false | 1243.5 | 2012-10-25 13:40:00 |
| 3 | false | 24453.325 | 2008-08-22 09:33:21.123000000 |
| 4 | false | 243423.325 | 2007-05-12 22:32:21.334540000 |
+----+-------+---------------+
| id | col_1 | col_2 |
+----+-------+---------------+
| 1 | true | 12789.123 |
| 2 | false | 1243.5 |
| 3 | false | 24453.325 |
| 4 | false | 2423.3254 |
| 5 | true | 243.325 |
| 60 | false | 243565423.325 |
| 70 | true | 243.325 |
| 80 | false | 243423.325 |
| 90 | true | 243.325 |
+----+-------+---------------+
+----+-------+-----------+
| id | col_1 | col_2 |
+----+-------+-----------+
| 1 | true | 12789.123 |
| 2 | false | 1243.5 |
| 3 | false | 24453.325 |
| 4 | false | 2423.3254 |
| 5 | true | 243.325 |
+----+-------+-----------+
Results:
+-------+-----------------+-----------------+
| col_1 | max(tab2.col_2) | min(tab2.col_2) |
+-------+-----------------+-----------------+
| false | 24453.325 | 1243.5 |
| true | 12789.123 | 243.325 |
+-------+-----------------+-----------------+
SELECT tab2.*
FROM tab2,
(SELECT tab1.col_1, MAX(tab2.col_2) AS max_col2
FROM tab2, tab1
WHERE tab1.id = tab2.id
GROUP BY col_1) subquery1
WHERE subquery1.max_col2 = tab2.col_2;
Results:
+----+-------+-----------+
| id | col_1 | col_2 |
+----+-------+-----------+
| 1 | true | 12789.123 |
| 3 | false | 24453.325 |
+----+-------+-----------+
Results:
+----+-------+---------+-------+-----+
| id | col_1 | col_2 | month | day |
+----+-------+---------+-------+-----+
| 1 | true | 123.123 | 10 | 24 |
| 2 | false | 1243.5 | 10 | 25 |
+----+-------+---------+-------+-----+
Advanced Tutorials
These tutorials walk you through advanced scenarios or specialized features.
Back in the Linux shell, we examine the HDFS directory structure. (Your Impala data directory might be in a different
location; for historical reasons, it is sometimes under the HDFS path /user/hive/warehouse.) We use the hdfs
dfs -ls command to examine the nested subdirectories corresponding to each partitioning column, with separate
subdirectories at each level (with = in their names) representing the different values for each partitioning column.
When we get to the lowest level of subdirectory, we use the hdfs dfs -cat command to examine the data file and
see CSV-formatted data produced by the INSERT statement in Impala.
Found 2 items
drwxr-xr-x - impala hive 0 2013-08-07 12:23
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=07
drwxr-xr-x - impala hive 0 2013-08-07 12:24
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=08
$ hdfs dfs -ls /user/impala/warehouse/external_partitions.db/logs/year=2013/month=07
Found 2 items
drwxr-xr-x - impala hive 0 2013-08-07 12:22
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=07/day=28
drwxr-xr-x - impala hive 0 2013-08-07 12:23
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=07/day=29
$ hdfs dfs -ls
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=07/day=28
Found 2 items
drwxr-xr-x - impala hive 0 2013-08-07 12:21
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=07/day=28/host=host1
drwxr-xr-x - impala hive 0 2013-08-07 12:22
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=07/day=28/host=host2
$ hdfs dfs -ls
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=07/day=28/host=host1
Found 1 items
-rw-r--r-- 3 impala hive 12 2013-08-07 12:21
/user/impala/warehouse/external_partiti
ons.db/logs/year=2013/month=07/day=28/host=host1/3981726974111751120--8907184999369517436_822630111_data.0
$ hdfs dfs -cat
/user/impala/warehouse/external_partitions.db/logs/year=2013/month=07/day=28/\
host=host1/3981726974111751120--8 907184999369517436_822630111_data.0
foo,foo,foo
Still in the Linux shell, we use hdfs dfs -mkdir to create several data directories outside the HDFS directory tree
that Impala controls (/user/impala/warehouse in this example, maybe different in your case). Depending on your
configuration, you might need to log in as a user with permission to write into this HDFS directory tree; for example,
the commands shown here were run while logged in as the hdfs user.
We make a tiny CSV file, with values different than in the INSERT statements used earlier, and put a copy within each
subdirectory that we will use as an Impala partition.
$ cat >dummy_log_data
bar,baz,bletch
$ hdfs dfs -mkdir -p
/user/impala/data/external_partitions/year=2013/month=08/day=01/host=host1
$ hdfs dfs -mkdir -p
/user/impala/data/external_partitions/year=2013/month=07/day=28/host=host1
$ hdfs dfs -mkdir -p
/user/impala/data/external_partitions/year=2013/month=07/day=28/host=host2
$ hdfs dfs -mkdir -p
/user/impala/data/external_partitions/year=2013/month=07/day=29/host=host1
$ hdfs dfs -put dummy_log_data /user/impala/data/logs/year=2013/month=07/day=28/host=host1
$ hdfs dfs -put dummy_log_data /user/impala/data/logs/year=2013/month=07/day=28/host=host2
$ hdfs dfs -put dummy_log_data /user/impala/data/logs/year=2013/month=07/day=29/host=host1
$ hdfs dfs -put dummy_log_data /user/impala/data/logs/year=2013/month=08/day=01/host=host1
Back in the impala-shell interpreter, we move the original Impala-managed table aside, and create a new external
table with a LOCATION clause pointing to the directory under which we have set up all the partition subdirectories
and data files.
use external_partitions;
alter table logs rename to logs_original;
create external table logs (field1 string, field2 string, field3 string)
partitioned by (year string, month string, day string, host string)
Because partition subdirectories and data files come and go during the data lifecycle, you must identify each of the
partitions through an ALTER TABLE statement before Impala recognizes the data files they contain.
We issue a REFRESH statement for the table, always a safe practice when data files have been manually added, removed,
or changed. Then the data is ready to be queried. The SELECT * statement illustrates that the data from our trivial
CSV file was recognized in each of the partitions where we copied it. Although in this case there are only a few rows,
we include a LIMIT clause on this test query just in case there is more data than we expect.
refresh log_type;
select * from log_type limit 100;
+--------+--------+--------+------+-------+-----+-------+
| field1 | field2 | field3 | year | month | day | host |
+--------+--------+--------+------+-------+-----+-------+
| bar | baz | bletch | 2013 | 07 | 28 | host1 |
| bar | baz | bletch | 2013 | 08 | 01 | host1 |
| bar | baz | bletch | 2013 | 07 | 29 | host1 |
| bar | baz | bletch | 2013 | 07 | 28 | host2 |
+--------+--------+--------+------+-------+-----+-------+
For examples showing how this process works for the REFRESH statement, look at the examples of creating RCFile and
SequenceFile tables in Impala, loading data through Hive, and then querying the data through Impala. See Using the
RCFile File Format with Impala Tables on page 552 and Using the SequenceFile File Format with Impala Tables on page
554 for those examples.
For examples showing how this process works for the INVALIDATE METADATA statement, look at the example of
creating and loading an Avro table in Hive, and then querying the data through Impala. See Using the Avro File Format
with Impala Tables on page 547 for that example.
Note:
Originally, Impala did not support UDFs, but this feature is available in Impala starting in Impala 1.2.
Some INSERT ... SELECT transformations that you originally did through Hive can now be done
through Impala. See Impala User-Defined Functions (UDFs) on page 447 for details.
Prior to Impala 1.2, the REFRESH and INVALIDATE METADATA statements needed to be issued on
each Impala node to which you connected and issued queries. In Impala 1.2 and higher, when you
issue either of those statements on any Impala node, the results are broadcast to all the Impala nodes
in the cluster, making it truly a one-step operation after each round of DDL or ETL operations in Hive.
Cross Joins and Cartesian Products with the CROSS JOIN Operator
Originally, Impala restricted join queries so that they had to include at least one equality comparison between the
columns of the tables on each side of the join operator. With the huge tables typically processed by Impala, any
miscoded query that produced a full Cartesian product as a result set could consume a huge amount of cluster resources.
In Impala 1.2.2 and higher, this restriction is lifted when you use the CROSS JOIN operator in the query. You still
cannot remove all WHERE clauses from a query like SELECT * FROM t1 JOIN t2 to produce all combinations of
rows from both tables. But you can use the CROSS JOIN operator to explicitly request such a Cartesian product.
Typically, this operation is applicable for smaller tables, where the result set still fits within the memory of a single
Impala node.
The following example sets up data for use in a series of comic books where characters battle each other. At first, we
use an equijoin query, which only allows characters from the same time period and the same planet to meet.
[localhost:21000] > create table heroes (name string, era string, planet string);
[localhost:21000] > create table villains (name string, era string, planet string);
[localhost:21000] > insert into heroes values
> ('Tesla','20th century','Earth'),
> ('Pythagoras','Antiquity','Earth'),
> ('Zopzar','Far Future','Mars');
Inserted 3 rows in 2.28s
[localhost:21000] > insert into villains values
> ('Caligula','Antiquity','Earth'),
> ('John Dillinger','20th century','Earth'),
> ('Xibulor','Far Future','Venus');
Inserted 3 rows in 1.93s
[localhost:21000] > select concat(heroes.name,' vs. ',villains.name) as battle
> from heroes join villains
> where heroes.era = villains.era and heroes.planet = villains.planet;
+--------------------------+
| battle |
+--------------------------+
| Tesla vs. John Dillinger |
| Pythagoras vs. Caligula |
+--------------------------+
Returned 2 row(s) in 0.47s
Readers demanded more action, so we added elements of time travel and space travel so that any hero could face
any villain. Prior to Impala 1.2.2, this type of query was impossible because all joins had to reference matching values
between the two tables:
With Impala 1.2.2, we rewrite the query slightly to use CROSS JOIN rather than JOIN, and now the result set includes
all combinations:
[localhost:21000] > -- Cartesian product available in Impala 1.2.2 with the CROSS JOIN
syntax.
> select concat(heroes.name,' vs. ',villains.name) as battle from
heroes cross join villains;
+-------------------------------+
| battle |
+-------------------------------+
| Tesla vs. Caligula |
| Tesla vs. John Dillinger |
| Tesla vs. Xibulor |
| Pythagoras vs. Caligula |
| Pythagoras vs. John Dillinger |
| Pythagoras vs. Xibulor |
| Zopzar vs. Caligula |
| Zopzar vs. John Dillinger |
| Zopzar vs. Xibulor |
+-------------------------------+
Returned 9 row(s) in 0.33s
The full combination of rows from both tables is known as the Cartesian product. This type of result set is often used
for creating grid data structures. You can also filter the result set by including WHERE clauses that do not explicitly
compare columns between the two tables. The following example shows how you might produce a list of combinations
of year and quarter for use in a chart, and then a shorter list with only selected quarters.
The data used in this tutorial represents airline on-time arrival statistics, from October 1987 through April 2008. See
the details on the 2009 ASA Data Expo web site. You can also see the explanations of the columns; for purposes of this
exercise, wait until after following the tutorial before examining the schema, to better simulate a real-life situation
where you cannot rely on assumptions and assertions about the ranges and representations of data values.
We will download Parquet files containing this data from the Ibis blog. First, we download and unpack the data files.
There are 8 files totalling 1.4 GB. Each file is less than 256 MB.
$ wget -O airlines_parquet.tar.gz
https://www.dropbox.com/s/ol9x51tqp6cv4yc/airlines_parquet.tar.gz?dl=0
...
Length: 1245204740 (1.2G) [application/octet-stream]
Saving to: “airlines_parquet.tar.gz”
Next, we put the Parquet data files in HDFS, all together in a single directory, with permissions on the directory and
the files so that the impala user will be able to read them.
Note: After unpacking, we saw the largest Parquet file was 253 MB. When copying Parquet files into
HDFS for Impala to use, for maximum query performance, make sure that each file resides in a single
HDFS data block. Therefore, we pick a size larger than any single file and specify that as the block size,
using the argument -Ddfs.block.size=256m on the hdfs dfs -put command.
With the files in an accessible location in HDFS, we create a database table that uses the data in those files. The CREATE
EXTERNAL syntax and the LOCATION attribute point Impala at the appropriate HDFS directory. The LIKE PARQUET
'path_to_any_parquet_file' clause means we skip the list of column names and types; Impala automatically
gets the column names and data types straight from the data files. (Currently, this technique only works for Parquet
files.) We ignore the warning about lack of READ_WRITE access to the files in HDFS; the impala user can read the
files, which will be sufficient for us to experiment with queries and perform some copy and transform operations into
other tables.
$ impala-shell -i localhost
Starting Impala Shell without Kerberos authentication
Connected to localhost:21000
Server version: impalad version 2.2.0-cdh5 RELEASE (build
2ffd73a4255cefd521362ffe1cfb37463f67f75c)
Welcome to the Impala shell. Press TAB twice to see a list of available commands.
(Shell build version: Impala Shell v2.1.2-cdh5 (92438b7) built on Tue Feb 24 12:36:33
PST 2015)
[localhost:21000] > create database airline_data;
[localhost:21000] > use airline_data;
[localhost:21000] > create external table airlines_external
> like parquet
'hdfs://demo_host.example.com:8020/user/impala/staging/airlines/93459d994898a9ba-77674173b331fa96_2118228804_data.0.parq'
With the table created, we examine its physical and logical characteristics to confirm that the data is really there and
in a format and shape that we can work with. The SHOW TABLE STATS statement gives a very high-level summary of
the table, showing how many files and how much total data it contains. Also, it confirms that the table is expecting all
the associated data files to be in Parquet format. (The ability to work with all kinds of HDFS data files in different
formats means that it is possible to have a mismatch between the format of the data files, and the format that the
table expects the data files to be in.) The SHOW FILES statement confirms that the data in the table has the expected
number, names, and sizes of the original Parquet files. The DESCRIBE statement (or its abbreviation DESC) confirms
the names and types of the columns that Impala automatically created after reading that metadata from the Parquet
file. The DESCRIBE FORMATTED statement prints out some extra detail along with the column definitions; the pieces
we care about for this exercise are the containing database for the table, the location of the associated data files in
HDFS, the fact that it's an external table so Impala will not delete the HDFS files when we finish the experiments and
drop the table, and the fact that the table is set up to work exclusively with files in the Parquet format.
| 239.95MB | |
| /user/impala/staging/airlines/93459d994898a9ba-77674173b331fa99_1555718317_data.0.parq
| 252.90MB | |
| /user/impala/staging/airlines/93459d994898a9ba-77674173b331fa99_1555718317_data.1.parq
| 15.89MB | |
| /user/impala/staging/airlines/93459d994898a9ba-77674173b331fa9a_2073981944_data.0.parq
| 176.92MB | |
| /user/impala/staging/airlines/93459d994898a9ba-77674173b331fa9b_1413430552_data.0.parq
| 212.47MB | |
+----------------------------------------------------------------------------------------+----------+-----------+
[localhost:21000] > describe airlines_external;
+---------------------+--------+---------------------------------------------------+
| name | type | comment |
+---------------------+--------+---------------------------------------------------+
| year | int | inferred from: optional int32 year |
| month | int | inferred from: optional int32 month |
| day | int | inferred from: optional int32 day |
| dayofweek | int | inferred from: optional int32 dayofweek |
| dep_time | int | inferred from: optional int32 dep_time |
| crs_dep_time | int | inferred from: optional int32 crs_dep_time |
| arr_time | int | inferred from: optional int32 arr_time |
| crs_arr_time | int | inferred from: optional int32 crs_arr_time |
| carrier | string | inferred from: optional binary carrier |
| flight_num | int | inferred from: optional int32 flight_num |
| tail_num | int | inferred from: optional int32 tail_num |
| actual_elapsed_time | int | inferred from: optional int32 actual_elapsed_time |
| crs_elapsed_time | int | inferred from: optional int32 crs_elapsed_time |
| airtime | int | inferred from: optional int32 airtime |
| arrdelay | int | inferred from: optional int32 arrdelay |
| depdelay | int | inferred from: optional int32 depdelay |
| origin | string | inferred from: optional binary origin |
| dest | string | inferred from: optional binary dest |
| distance | int | inferred from: optional int32 distance |
| taxi_in | int | inferred from: optional int32 taxi_in |
| taxi_out | int | inferred from: optional int32 taxi_out |
| cancelled | int | inferred from: optional int32 cancelled |
| cancellation_code | string | inferred from: optional binary cancellation_code |
| diverted | int | inferred from: optional int32 diverted |
| carrier_delay | int | inferred from: optional int32 carrier_delay |
| weather_delay | int | inferred from: optional int32 weather_delay |
| nas_delay | int | inferred from: optional int32 nas_delay |
| security_delay | int | inferred from: optional int32 security_delay |
| late_aircraft_delay | int | inferred from: optional int32 late_aircraft_delay |
+---------------------+--------+---------------------------------------------------+
[localhost:21000] > desc formatted airlines_external;
+------------------------------+-------------------------------
| name | type
+------------------------------+-------------------------------
...
| # Detailed Table Information | NULL
| Database: | airline_data
| Owner: | jrussell
...
| Location: | /user/impala/staging/airlines
| Table Type: | EXTERNAL_TABLE
...
| # Storage Information | NULL
| SerDe Library: | parquet.hive.serde.ParquetHiveSerDe
| InputFormat: | parquet.hive.DeprecatedParquetInputFormat
| OutputFormat: | parquet.hive.DeprecatedParquetOutputFormat
...
Now that we are confident that the connections are solid between the Impala table and the underlying Parquet files,
we run some initial queries to understand the characteristics of the data: the overall number of rows, and the ranges
and how many different values are in certain columns. For convenience in understanding the magnitude of the COUNT(*)
result, we run another query dividing the number of rows by 1 million, demonstrating that there are 123 million rows
in the table.
+-----------+
| 123534969 |
+-----------+
Fetched 1 row(s) in 1.32s
[localhost:21000] > select count(*) / 1e6 as 'millions of rows' from airlines_external;
+------------------+
| millions of rows |
+------------------+
| 123.534969 |
+------------------+
Fetched 1 row(s) in 1.24s
The NDV() function stands for “number of distinct values”, which for performance reasons is an estimate when there
are lots of different values in the column, but is precise when the cardinality is less than 16 K. Use NDV() calls for this
kind of exploration rather than COUNT(DISTINCT colname), because Impala can evaluate multiple NDV() functions
in a single query, but only a single instance of COUNT DISTINCT. Here we see that there are modest numbers of
different airlines, flight numbers, and origin and destination airports. Two things jump out from this query: the number
of tail_num values is much smaller than we might have expected, and there are more destination airports than origin
airports. Let's dig further. What we find is that most tail_num values are NULL. It looks like this was an experimental
column that wasn't filled in accurately. We make a mental note that if we use this data as a starting point, we'll ignore
this column. We also find that certain airports are represented in the ORIGIN column but not the DEST column; now
we know that we cannot rely on the assumption that those sets of airport codes are identical.
Note: A slight digression for some performance tuning. Notice how the first SELECT DISTINCT
DEST query takes almost 40 seconds. We expect all queries on such a small data set, less than 2 GB,
to take a few seconds at most. The reason is because the expression NOT IN (SELECT origin
FROM airlines_external) produces an intermediate result set of 123 million rows, then runs 123
million comparisons on each data node against the tiny set of destination airports. The way the NOT
IN operator works internally means that this intermediate result set with 123 million rows might be
transmitted across the network to each data node in the cluster. Applying another DISTINCT inside
the NOT IN subquery means that the intermediate result set is only 340 items, resulting in much less
network traffic and fewer comparison operations. The more efficient query with the added DISTINCT
is approximately 7 times as fast.
Next, we try doing a simple calculation, with results broken down by year. This reveals that some years have no data
in the AIRTIME column. That means we might be able to use that column in queries involving certain date ranges, but
we cannot count on it to always be reliable. The question of whether a column contains any NULL values, and if so
what is their number, proportion, and distribution, comes up again and again when doing initial exploration of a data
set.
With the notion of NULL values in mind, let's come back to the TAILNUM column that we discovered had a lot of NULLs.
Let's quantify the NULL and non-NULL values in that column for better understanding. First, we just count the overall
number of rows versus the non-NULL values in that column. That initial result gives the appearance of relatively few
non-NULL values, but we can break it down more clearly in a single query. Once we have the COUNT(*) and the
COUNT(colname) numbers, we can encode that initial query in a WITH clause, then run a followon query that performs
multiple arithmetic operations on those values. Seeing that only one-third of one percent of all rows have non-NULL
values for the TAILNUM column clearly illustrates that that column won't be of much use.
By examining other columns using these techniques, we can form a mental picture of the way data is distributed
throughout the table, and which columns are most significant for query purposes. For this tutorial, we focus mostly
on the fields likely to hold discrete values, rather than columns such as ACTUAL_ELAPSED_TIME whose names suggest
they hold measurements. We would dig deeper into those columns once we had a clear picture of which questions
were worthwhile to ask, and what kinds of trends we might look for. For the final piece of initial exploration, let's look
at the YEAR column. A simple GROUP BY query shows that it has a well-defined range, a manageable number of distinct
values, and relatively even distribution of rows across the different years.
We could go quite far with the data in this initial raw format, just as we downloaded it from the web. If the data set
proved to be useful and worth persisting in Impala for extensive queries, we might want to copy it to an internal table,
letting Impala manage the data files and perhaps reorganizing a little for higher efficiency. In this next stage of the
tutorial, we copy the original data into a partitioned table, still in Parquet format. Partitioning based on the YEAR
column lets us run queries with clauses such as WHERE year = 2001 or WHERE year BETWEEN 1989 AND 1999,
which can dramatically cut down on I/O by ignoring all the data from years outside the desired range. Rather than
reading all the data and then deciding which rows are in the matching years, Impala can zero in on only the data files
from specific YEAR partitions. To do this, Impala physically reorganizes the data files, putting the rows from each year
into data files in a separate HDFS directory for each YEAR value. Along the way, we'll also get rid of the TAIL_NUM
column that proved to be almost entirely NULL.
The first step is to create a new table with a layout very similar to the original AIRLINES_EXTERNAL table. We'll do
that by reverse-engineering a CREATE TABLE statement for the first table, then tweaking it slightly to include a
PARTITION BY clause for YEAR, and excluding the TAIL_NUM column. The SHOW CREATE TABLE statement gives us
the starting point.
Although we could edit that output into a new SQL statement, all the ASCII box characters make such editing
inconvenient. To get a more stripped-down CREATE TABLE to start with, we restart the impala-shell command
with the -B option, which turns off the box-drawing behavior.
(Shell build version: Impala Shell v2.1.2-cdh5 (92438b7) built on Tue Feb 24 12:36:33
PST 2015)
[localhost:21000] > show create table airlines_external;
"CREATE EXTERNAL TABLE airline_data.airlines_external (
year INT COMMENT 'inferred from: optional int32 year',
month INT COMMENT 'inferred from: optional int32 month',
day INT COMMENT 'inferred from: optional int32 day',
After copying and pasting the CREATE TABLE statement into a text editor for fine-tuning, we quit and restart
impala-shell without the -B option, to switch back to regular output.
Next we run the CREATE TABLE statement that we adapted from the SHOW CREATE TABLE output. We kept the
STORED AS PARQUET clause because we want to rearrange the data somewhat but still keep it in the high-performance
Parquet format. The LOCATION and TBLPROPERTIES clauses are not relevant for this new table, so we edit those out.
Because we are going to partition the new table based on the YEAR column, we move that column name (and its type)
into a new PARTITIONED BY clause.
Next, we copy all the rows from the original table into this new one with an INSERT statement. (We edited the CREATE
TABLE statement to make an INSERT statement with the column names in the same order.) The only change is to add
a PARTITION(year) clause, and move the YEAR column to the very end of the SELECT list of the INSERT statement.
Specifying PARTITION(year), rather than a fixed value such as PARTITION(year=2000), means that Impala figures
out the partition value for each row based on the value of the very last column in the SELECT list. This is the first SQL
statement that legitimately takes any substantial time, because the rows from different years are shuffled around the
cluster; the rows that go into each partition are collected on one node, before being written to one or more new data
files.
Once partitioning or join queries come into play, it's important to have statistics that Impala can use to optimize queries
on the corresponding tables. The COMPUTE INCREMENTAL STATS statement is the way to collect statistics for
partitioned tables. Then the SHOW TABLE STATS statement confirms that the statistics are in place for each partition,
and also illustrates how many files and how much raw data is in each partition.
At this point, we go through a quick thought process to sanity check the partitioning we did. All the partitions have
exactly one file, which is on the low side. A query that includes a clause WHERE year=2004 will only read a single data
block; that data block will be read and processed by a single data node; therefore, for a query targeting a single year,
all the other nodes in the cluster will sit idle while all the work happens on a single machine. It's even possible that by
chance (depending on HDFS replication factor and the way data blocks are distributed across the cluster), that multiple
year partitions selected by a filter such as WHERE year BETWEEN 1999 AND 2001 could all be read and processed
by the same data node. The more data files each partition has, the more parallelism you can get and the less probability
of “hotspots” occurring on particular nodes, therefore a bigger performance boost by having a big CDH cluster.
However, the more data files, the less data goes in each one. The overhead of dividing the work in a parallel query
might not be worth it if each node is only reading a few megabytes. 50 or 100 megabytes is a decent size for a Parquet
data block; 9 or 37 megabytes is on the small side. Which is to say, the data distribution we ended up with based on
this partitioning scheme is on the borderline between sensible (reasonably large files) and suboptimal (few files in
each partition). The way to see how well it works in practice is to run the same queries against the original flat table
and the new partitioned table, and compare times.
Spoiler: in this case, with my particular 4-node cluster with its specific distribution of data blocks and my particular
exploratory queries, queries against the partitioned table do consistently run faster than the same queries against the
unpartitioned table. But I could not be sure that would be the case without some real measurements. Here are some
queries I ran to draw that conclusion, first against AIRLINES_EXTERNAL (no partitioning), then against AIRLINES
(partitioned by year). The AIRLINES queries are consistently faster. Changing the volume of data, changing the size
of the cluster, running queries that did or didn't refer to the partition key columns, or other factors could change the
results to favor one table layout or the other.
Note: If you find the volume of each partition is only in the low tens of megabytes, consider lowering
the granularity of partitioning. For example, instead of partitioning by year, month, and day, partition
by year and month or even just by year. The ideal layout to distribute work efficiently in a parallel
query is many tens or even hundreds of megabytes per Parquet file, and the number of Parquet files
in each partition somewhat higher than the number of data nodes.
| 8662859484 |
+--------------+
Fetched 1 row(s) in 1.21s
Now we can finally do some serious analysis with this data set that, remember, a few minutes ago all we had were
some raw data files and we didn't even know what columns they contained. Let's see whether the “air time” of a flight
tends to be different depending on the day of the week. We can see that the average is a little higher on day number
6; perhaps Saturday is a busy flying day and planes have to circle for longer at the destination airport before landing.
To see if the apparent trend holds up over time, let's do the same breakdown by day of week, but also split up by year.
Now we can see that day number 6 consistently has a higher average air time in each year. We can also see that the
average air time increased over time across the board. And the presence of NULL for this column in years 1987 to 1994
shows that queries involving this column need to be restricted to a date range of 1995 and higher.
| 2008 | 6 | 107.4006306562128 |
| 2008 | 7 | 104.8648851041755 |
| 2007 | 1 | 102.2196114337825 |
| 2007 | 2 | 101.9317791906348 |
| 2007 | 3 | 102.0964767689043 |
| 2007 | 4 | 102.6215927201686 |
| 2007 | 5 | 102.4289399000661 |
| 2007 | 6 | 105.1477448215756 |
| 2007 | 7 | 103.6305945644095 |
...
| 1996 | 1 | 99.33860750862108 |
| 1996 | 2 | 99.54225446396656 |
| 1996 | 3 | 99.41129336113134 |
| 1996 | 4 | 99.5110373340348 |
| 1996 | 5 | 99.22120745027595 |
| 1996 | 6 | 101.1717447111921 |
| 1996 | 7 | 99.95410136133704 |
| 1995 | 1 | 96.93779698300494 |
| 1995 | 2 | 96.93458674589712 |
| 1995 | 3 | 97.00972311337051 |
| 1995 | 4 | 96.90843832024412 |
| 1995 | 5 | 96.78382115425562 |
| 1995 | 6 | 98.70872826057003 |
| 1995 | 7 | 97.85570478374616 |
| 1994 | 1 | NULL |
| 1994 | 2 | NULL |
| 1994 | 3 | NULL |
...
| 1987 | 5 | NULL |
| 1987 | 6 | NULL |
| 1987 | 7 | NULL |
+------+-----------+-------------------+
Impala Administration
As an administrator, you monitor Impala's use of resources and take action when necessary to keep Impala running
smoothly and avoid conflicts with other Hadoop components running on the same cluster. When you detect that an
issue has happened or could happen in the future, you reconfigure Impala or other components such as HDFS or even
the hardware of the cluster itself to resolve or avoid problems.
Related tasks:
As an administrator, you can expect to perform installation, upgrade, and configuration tasks for Impala on all machines
in a cluster. See Installing Impala on page 27, Upgrading Impala on page 38, and Managing Impala on page 30 for
details.
For security tasks typically performed by administrators, see Impala Security on page 91.
Administrators also decide how to allocate cluster resources so that all Hadoop components can run smoothly together.
For Impala, this task primarily involves:
• Deciding how many Impala queries can run concurrently and with how much memory, through the admission
control feature. See Admission Control and Query Queuing on page 75 for details.
• Dividing cluster resources such as memory between Impala and other components, using YARN for overall resource
management, and Llama to mediate resource requests from Impala to YARN. See Integrated Resource Management
with YARN on page 83 for details.
Note: Though Impala can be used together with YARN via simple configuration of Static Service Pools
in Cloudera Manager, the use of the general-purpose component Llama for integrated resource
management within YARN is no longer supported with CDH 5.5 / Impala 2.3 and higher.
being cancelled or running slowly and causing contention. As other queries finish, the queued queries are allowed to
proceed.
For details on the internal workings of admission control, see How Impala Schedules and Enforces Limits on Concurrent
Queries on page 76.
How Admission Control works with Impala Clients (JDBC, ODBC, HiveServer2)
Most aspects of admission control work transparently with client interfaces such as JDBC and ODBC:
• If a SQL statement is put into a queue rather than running immediately, the API call blocks until the statement is
dequeued and begins execution. At that point, the client program can request to fetch results, which might also
block until results become available.
• If a SQL statement is cancelled because it has been queued for too long or because it exceeded the memory limit
during execution, the error is returned to the client program with a descriptive error message.
If you want to submit queries to different resource pools through the REQUEST_POOL query option, as described in
REQUEST_POOL Query Option on page 320, In Impala 2.0 and higher you can change that query option through a SQL
SET statement that you submit from the client application, in the same session. Prior to Impala 2.0, that option was
only settable for a session through the impala-shell SET command, or cluster-wide through an impalad startup
option.
Admission control has the following limitations or special behavior when used with JDBC or ODBC applications:
• The MEM_LIMIT query option, sometimes useful to work around problems caused by inaccurate memory estimates
for complicated queries, is only settable through the impala-shell interpreter and cannot be used directly
through JDBC or ODBC applications.
• Admission control does not use the other resource-related query options, RESERVATION_REQUEST_TIMEOUT or
V_CPU_CORES. Those query options only apply to the YARN resource management framework.
Type: int64
Default: 200
default_pool_max_requests
Purpose: Maximum number of concurrent outstanding requests allowed to run before incoming requests are
queued. Because this limit applies cluster-wide, but each Impala node makes independent decisions to run queries
immediately or queue them, it is a soft limit; the overall number of concurrent queries might be slightly higher
during times of heavy load. A negative value indicates no limit. Ignored if fair_scheduler_config_path and
llama_site_path are set.
Type: int64
Default: 200
default_pool_mem_limit
Purpose: Maximum amount of memory (across the entire cluster) that all outstanding requests in this pool can use
before new requests to this pool are queued. Specified in bytes, megabytes, or gigabytes by a number followed by
the suffix b (optional), m, or g, either uppercase or lowercase. You can specify floating-point values for megabytes
and gigabytes, to represent fractional numbers such as 1.5. You can also specify it as a percentage of the physical
memory by specifying the suffix %. 0 or no setting indicates no limit. Defaults to bytes if no unit is given. Because
this limit applies cluster-wide, but each Impala node makes independent decisions to run queries immediately or
queue them, it is a soft limit; the overall memory used by concurrent queries might be slightly higher during times
of heavy load. Ignored if fair_scheduler_config_path and llama_site_path are set.
Note: Impala relies on the statistics produced by the COMPUTE STATS statement to estimate
memory usage for each query. See COMPUTE STATS Statement on page 226 for guidelines about
how and when to use this statement.
Type: string
Default: "" (empty string, meaning unlimited)
disable_admission_control
Purpose: Turns off the admission control feature entirely, regardless of other configuration option settings.
Type: Boolean
Default: true
disable_pool_max_requests
Purpose: Disables all per-pool limits on the maximum number of running requests.
Type: Boolean
Default: false
disable_pool_mem_limits
Purpose: Disables all per-pool mem limits.
Type: Boolean
Default: false
fair_scheduler_allocation_path
Purpose: Path to the fair scheduler allocation file (fair-scheduler.xml).
Type: string
Default: "" (empty string)
Usage notes: Admission control only uses a small subset of the settings that can go in this file, as described below.
For details about all the Fair Scheduler configuration settings, see the Apache wiki.
llama_site_path
Purpose: Path to the Llama configuration file (llama-site.xml). If set, fair_scheduler_allocation_path
must also be set.
Type: string
Default: "" (empty string)
Usage notes: Admission control only uses a small subset of the settings that can go in this file, as described below.
For details about all the Llama configuration settings, see the documentation on Github.
queue_wait_timeout_ms
Purpose: Maximum amount of time (in milliseconds) that a request waits to be admitted before timing out.
Type: int64
Default: 60000
llama.am.throttling.maximum.placed.reservations.queue_name
llama.am.throttling.maximum.queued.reservations.queue_name
For details about all the Llama configuration settings, see Llama Default Configuration.
See Example Admission Control Configurations Using Configuration Files on page 81 for sample configuration files for
admission control using multiple resource pools, without Cloudera Manager.
Figure 1: Sample Settings for Cloudera Manager Dynamic Resource Pools Page
The following figure shows a sample of the Placement Rules page in Cloudera Manager, accessed through the Clusters >
Cluster name > Resource Management > Dynamic Resource Pools menu choice and then the Configuration > Placement
Rules tabs. The settings demonstrate a reasonable configuration of a pool named default to service all requests
where the specified resource pool does not exist, is not explicitly set, or the user or group is not authorized for the
specified pool.
<allocations>
<queue name="root">
<aclSubmitApps> </aclSubmitApps>
<queue name="default">
<maxResources>50000 mb, 0 vcores</maxResources>
<aclSubmitApps>*</aclSubmitApps>
</queue>
<queue name="development">
<maxResources>200000 mb, 0 vcores</maxResources>
<aclSubmitApps>user1,user2 dev,ops,admin</aclSubmitApps>
</queue>
<queue name="production">
<maxResources>1000000 mb, 0 vcores</maxResources>
<aclSubmitApps> ops,admin</aclSubmitApps>
</queue>
</queue>
<queuePlacementPolicy>
<rule name="specified" create="false"/>
<rule name="default" />
</queuePlacementPolicy>
</allocations>
llama-site.xml:
The statements affected by the admission control feature are primarily queries, but also include statements that write
data such as INSERT and CREATE TABLE AS SELECT. Most write operations in Impala are not resource-intensive,
but inserting into a Parquet table can require substantial memory due to buffering 1 GB of data before writing out
each Parquet data block. See Loading Data into Parquet Tables on page 536 for instructions about inserting data
efficiently into Parquet tables.
Although admission control does not scrutinize memory usage for other kinds of DDL statements, if a query is queued
due to a limit on concurrent queries or memory usage, subsequent statements in the same session are also queued
so that they are processed in the correct order:
If you set up different resource pools for different users and groups, consider reusing any classifications and hierarchy
you developed for use with Sentry security. See Enabling Sentry Authorization for Impala on page 95 for details.
For details about all the Fair Scheduler configuration settings, see Fair Scheduler Configuration, in particular the tags
such as <queue> and <aclSubmitApps> to map users and groups to particular resource pools (queues).
You can limit the CPU and memory resources used by Impala, to manage and prioritize workloads on clusters that run
jobs from many Hadoop components.
Requests from Impala to YARN go through an intermediary service called Llama. When the resource requests are
granted, Impala starts the query and places all relevant execution threads into the cgroup containers and sets up the
memory limit on each host. If sufficient resources are not available, the Impala query waits until other jobs complete
and the resources are freed. During query processing, as the need for additional resources arises, Llama can “expand”
already-requested resources, to avoid over-allocating at the start of the query.
After a query is finished, Llama caches the resources (for example, leaving memory allocated) in case they are needed
for subsequent Impala queries. This caching mechanism avoids the latency involved in making a whole new set of
resource requests for each query. If the resources are needed by YARN for other types of jobs, Llama returns them.
While the delays to wait for resources might make individual queries seem less responsive on a heavily loaded cluster,
the resource management feature makes the overall performance of the cluster smoother and more predictable,
without sudden spikes in utilization due to memory paging, CPUs pegged at 100%, and so on.
See Scalability Considerations for the Impala Statestore on page 516 for more details about statestore operation and
settings on clusters with a large number of Impala-related objects such as tables and partitions.
Setting the Idle Query and Idle Session Timeouts for impalad
To keep long-running queries or idle sessions from tying up cluster resources, you can set timeout intervals for both
individual queries, and entire sessions.
Note:
The timeout clock for queries and sessions only starts ticking when the query or session is idle. For
queries, this means the query has results ready but is waiting for a client to fetch the data. A query
can run for an arbitrary time without triggering a timeout, because the query is computing results
rather than sitting idle waiting for the results to be fetched. The timeout period is intended to prevent
unclosed queries from consuming resources and taking up slots in the admission count of running
queries, potentially preventing other queries from starting.
For sessions, this means that no query has been submitted for some period of time.
For instructions on changing impalad startup options, see Modifying Impala Startup Options on page 43.
Note:
To avoid excessive polling, Impala checks periodically for idle sessions and queries to cancel. The actual
idle time before cancellation might be up to 50% greater than the specified configuration setting. For
example, if the timeout setting was 60, the session or query might be cancelled after being idle between
60 and 90 seconds.
Cancelling a Query
Sometimes, an Impala query might run for an unexpectedly long time, tying up resources in the cluster. You can cancel
the query explicitly, independent of the timeout period, by going into the web UI for the impalad host (on port 25000
by default), and using the link on the /queries tab to cancel the running query. Various client applications let you
interactively cancel queries submitted or monitored through those applications. For example, by pressing ^C in
impala-shell, clicking the Cancel button from the Watch page in Hue, clicking Actions > Cancel from the Queries
list in Cloudera Manager, and so on.
Note:
Most considerations for load balancing and high availability apply to the impalad daemon. The
statestored and catalogd daemons do not have special requirements for high availability, because
problems with those daemons do not result in data loss. If those daemons become unavailable due
to an outage on a particular host, you can stop the Impala service, delete the Impala StateStore and
Impala Catalog Server roles, add the roles on a different host, and restart the Impala service.
3. Specify the host and port settings for each Impala node. These are the hosts that the load balancer will choose
from when relaying each Impala query. See Ports Used by Impala on page 583 for when to use port 21000, 21050,
or another value depending on what type of connections you are load balancing.
Note:
In particular, if you are using Hue or JDBC-based applications, you typically set up load balancing
for both ports 21000 and 21050, because these client applications connect through port 21050
while the impala-shell command connects through port 21000.
4. Run the load-balancing proxy server, pointing it at the configuration file that you set up.
5. On systems managed by Cloudera Manager, on the page Impala > Configuration > Impala Daemon Default Group,
specify a value for the Impala Daemons Load Balancer field. Specify the address of the load balancer in host:port
format. This setting lets Cloudera Manager route all appropriate Impala-related operations through the proxy
server.
6. For any scripts, jobs, or configuration settings for applications that formerly connected to a specific datanode to
run Impala SQL statements, change the connection information (such as the -i option in impala-shell) to point
to the load balancer instead.
Note: The following sections use the HAProxy software as a representative example of a load balancer
that you can use with Impala. For information specifically about using Impala with the F5 BIG-IP load
balancer, see Impala HA with F5 BIG-IP.
$ ktutil
ktutil: read_kt proxy.keytab
ktutil: read_kt impala.keytab
ktutil: write_kt proxy_impala.keytab
ktutil: quit
Note: On systems managed by Cloudera Manager 5.1.0 and later, the keytab merging happens
automatically. To verify that Cloudera Manager has merged the keytabs, run the command:
klist -k keytabfile
which lists the credentials for both principal and be_principal on all nodes.
6. Make sure that the impala user has permission to read this merged keytab file.
7. Change some configuration settings for each host in the cluster that participates in the load balancing. Follow the
appropriate steps depending on whether you use Cloudera Manager or not:
• In the impalad option definition, or the Cloudera Manager safety valve (Cloudera Manager 4) or advanced
configuration snippet (Cloudera Manager 5), add:
--principal=impala/proxy_host@realm
--be_principal=impala/actual_host@realm
--keytab_file=path_to_merged_keytab
Note:
Every host has a different --be_principal because the actual hostname is different on
each host.
Specify the fully qualified domain name (FQDN) for the proxy host, not the IP address. Use
the exact FQDN as returned by a reverse DNS lookup for the associated IP address.
• On a cluster managed by Cloudera Manager, create a role group to set the configuration values from the
preceding step on a per-host basis.
• On a cluster not managed by Cloudera Manager, see Modifying Impala Startup Options on page 43 for the
procedure to modify the startup options.
8. Restart Impala to make the changes take effect. Follow the appropriate steps depending on whether you use
Cloudera Manager or not:
• On a cluster managed by Cloudera Manager, restart the Impala service.
• On a cluster not managed by Cloudera Manager, restart the impalad daemons on all hosts in the cluster, as
well as the statestored and catalogd daemons.
/usr/sbin/haproxy –f /etc/haproxy/haproxy.cfg
• In impala-shell, JDBC applications, or ODBC applications, connect to the listener port of the proxy host, rather
than port 21000 or 21050 on a host actually running impalad. The sample configuration file sets haproxy to listen
on port 25003, therefore you would send all requests to haproxy_host:25003.
global
# To have these messages end up in /var/log/haproxy.log you will
# need to:
#
# 1) configure syslog to accept network log events. This is done
# by adding the '-r' option to the SYSLOGD_OPTIONS in
# /etc/sysconfig/syslog
#
# 2) configure local2 events to go to the /var/log/haproxy.log
# file. A line like the following can be added to
# /etc/sysconfig/syslog
#
# local2.* /var/log/haproxy.log
#
log 127.0.0.1 local0
log 127.0.0.1 local1 notice
chroot /var/lib/haproxy
pidfile /var/run/haproxy.pid
maxconn 4000
user haproxy
group haproxy
daemon
#---------------------------------------------------------------------
# common defaults that all the 'listen' and 'backend' sections will
# use if not designated in their block
#
# You might need to adjust timing values to prevent timeouts.
#---------------------------------------------------------------------
defaults
mode http
log global
option httplog
option dontlognull
option http-server-close
option forwardfor except 127.0.0.0/8
option redispatch
retries 3
maxconn 3000
contimeout 5000
clitimeout 50000
srvtimeout 50000
#
# This sets up the admin page for HA Proxy at port 25002.
#
listen stats :25002
balance
mode http
stats enable
stats auth username:password
Note: If your JDBC or ODBC application connects to Impala through a load balancer such as haproxy,
be cautious about reusing the connections. If the load balancer has set up connection timeout values,
either check the connection frequently so that it never sits idle longer than the load balancer timeout
value, or check the connection validity before using it and create a new one if the connection has
been closed.
How Impala Works with Hadoop File Formats on page 527 for details about file formats, especially Using the Parquet
File Format with Impala Tables on page 535. See CREATE TABLE Statement on page 235 and ALTER TABLE Statement
on page 215 for syntax details.
• You manage underlying data files differently depending on whether the corresponding Impala table is defined as
an internal or external table:
– Use the DESCRIBE FORMATTED statement to check if a particular table is internal (managed by Impala) or
external, and to see the physical location of the data files in HDFS. See DESCRIBE Statement on page 245 for
details.
– For Impala-managed (“internal”) tables, use DROP TABLE statements to remove data files. See DROP TABLE
Statement on page 258 for details.
– For tables not managed by Impala (“external” tables), use appropriate HDFS-related commands such as
hadoop fs, hdfs dfs, or distcp, to create, move, copy, or delete files within HDFS directories that are
accessible by the impala user. Issue a REFRESH table_name statement after adding or removing any files
from the data directory of an external table. See REFRESH Statement on page 277 for details.
– Use external tables to reference HDFS data files in their original location. With this technique, you avoid
copying the files, and you can map more than one Impala table to the same set of data files. When you drop
the Impala table, the data files are left undisturbed. See External Tables on page 209 for details.
– Use the LOAD DATA statement to move HDFS files into the data directory for an Impala table from inside
Impala, without the need to specify the HDFS path of the destination directory. This technique works for both
internal and external tables. See LOAD DATA Statement on page 274 for details.
• Make sure that the HDFS trashcan is configured correctly. When you remove files from HDFS, the space might not
be reclaimed for use by other files until sometime later, when the trashcan is emptied. See DROP TABLE Statement
on page 258 and the FAQ entry Why is space not freed up when I issue DROP TABLE? on page 601 for details. See
User Account Requirements on page 24 for permissions needed for the HDFS trashcan to operate correctly.
• Drop all tables in a database before dropping the database itself. See DROP DATABASE Statement on page 252 for
details.
• Clean up temporary files after failed INSERT statements. If an INSERT statement encounters an error, and you
see a directory named .impala_insert_staging or _impala_insert_staging left behind in the data
directory for the table, it might contain temporary data files taking up space in HDFS. You might be able to salvage
these data files, for example if they are complete but could not be moved into place due to a permission error.
Or, you might delete those files through commands such as hadoop fs or hdfs dfs, to reclaim space before
re-trying the INSERT. Issue DESCRIBE FORMATTED table_name to see the HDFS path where you can check for
temporary files.
• By default, intermediate files used during large sort, join, aggregation, or analytic function operations are stored
in the directory /tmp/impala-scratch . These files are removed when the operation finishes. (Multiple concurrent
queries can perform operations that use the “spill to disk” technique, without any name conflicts for these
temporary files.) You can specify a different location by starting the impalad daemon with the
--scratch_dirs="path_to_directory" configuration option or the equivalent configuration option in the
Cloudera Manager user interface. You can specify a single directory, or a comma-separated list of directories. The
scratch directories must be on the local filesystem, not in HDFS. You might specify different directory paths for
different hosts, depending on the capacity and speed of the available storage devices. In CDH 5.5 / Impala 2.3 or
higher, Impala successfully starts (with a warning written to the log) if it cannot create or read and write files in
one of the scratch directories. If there is less than 1 GB free on the filesystem where that directory resides, Impala
still runs, but writes a warning message to its log. If Impala encounters an error reading or writing files in a scratch
directory during a query, Impala logs the error and the query fails.
• If you use the Amazon Simple Storage Service (S3) as a place to offload data to reduce the volume of local storage,
Impala 2.2.0 and higher can query the data directly from S3. See Using Impala to Query the Amazon S3 Filesystem
on page 566 for details.
Impala Security
Impala includes a fine-grained authorization framework for Hadoop, based on the Sentry open source project. Sentry
authorization was added in Impala 1.1.0. Together with the Kerberos authentication framework, Sentry takes Hadoop
security to a new level needed for the requirements of highly regulated industries such as healthcare, financial services,
and government. Impala also includes an auditing capability; Impala generates the audit data, the Cloudera Navigator
product consolidates the audit data from all nodes in the cluster, and Cloudera Manager lets you filter, visualize, and
produce reports. The auditing feature was added in Impala 1.1.1.
The Impala security features have several objectives. At the most basic level, security prevents accidents or mistakes
that could disrupt application processing, delete or corrupt data, or reveal data to unauthorized users. More advanced
security features and practices can harden the system against malicious users trying to gain unauthorized access or
perform other disallowed operations. The auditing feature provides a way to confirm that no unauthorized access
occurred, and detect whether any such attempts were made. This is a critical set of features for production deployments
in large organizations that handle important or sensitive data. It sets the stage for multi-tenancy, where multiple
applications run concurrently and are prevented from interfering with each other.
The material in this section presumes that you are already familiar with administering secure Linux systems. That is,
you should know the general security practices for Linux and Hadoop, and their associated commands and configuration
files. For example, you should know how to create Linux users and groups, manage Linux group membership, set Linux
and HDFS file permissions and ownership, and designate the default permissions and ownership for new files. You
should be familiar with the configuration of the nodes in your Hadoop cluster, and know how to apply configuration
changes or run a set of commands across all the nodes.
The security features are divided into these broad categories:
authorization
Which users are allowed to access which resources, and what operations are they allowed to perform? Impala relies
on the open source Sentry project for authorization. By default (when authorization is not enabled), Impala does
all read and write operations with the privileges of the impala user, which is suitable for a development/test
environment but not for a secure production environment. When authorization is enabled, Impala uses the OS user
ID of the user who runs impala-shell or other client program, and associates various privileges with each user.
See Enabling Sentry Authorization for Impala on page 95 for details about setting up and managing authorization.
authentication
How does Impala verify the identity of the user to confirm that they really are allowed to exercise the privileges
assigned to that user? Impala relies on the Kerberos subsystem for authentication. See Enabling Kerberos
Authentication for Impala on page 106 for details about setting up and managing authentication.
auditing
What operations were attempted, and did they succeed or not? This feature provides a way to look back and
diagnose whether attempts were made to perform unauthorized operations. You use this information to track
down suspicious activity, and to see where changes are needed in authorization policies. The audit data produced
by this feature is collected by the Cloudera Manager product and then presented in a user-friendly form by the
Cloudera Manager product. See Auditing Impala Operations on page 112 for details about setting up and managing
auditing.
• Restrict membership in the sudoers list (in the /etc/sudoers file). The users who can run the sudo command
can do many of the same things as the root user.
• Ensure the Hadoop ownership and permissions for Impala data files are restricted.
• Ensure the Hadoop ownership and permissions for Impala log files are restricted.
• Ensure that the Impala web UI (available by default on port 25000 on each Impala node) is password-protected.
See Impala Web User Interface for Debugging on page 580 for details.
• Create a policy file that specifies which Impala privileges are available to users in particular Hadoop groups (which
by default map to Linux OS groups). Create the associated Linux groups using the groupadd command if necessary.
• The Impala authorization feature makes use of the HDFS file ownership and permissions mechanism; for background
information, see the CDH HDFS Permissions Guide. Set up users and assign them to groups at the OS level,
corresponding to the different categories of users with different access levels for various databases, tables, and
HDFS locations (URIs). Create the associated Linux users using the useradd command if necessary, and add them
to the appropriate groups with the usermod command.
• Design your databases, tables, and views with database and table structure to allow policy rules to specify simple,
consistent rules. For example, if all tables related to an application are inside a single database, you can assign
privileges for that database and use the * wildcard for the table name. If you are creating views with different
privileges than the underlying base tables, you might put the views in a separate database so that you can use
the * wildcard for the database containing the base tables, while specifying the precise names of the individual
views. (For specifying table or database names, you either specify the exact name or * to mean all the databases
on a server, or all the tables and views in a database.)
• Enable authorization by running the impalad daemons with the -server_name and
-authorization_policy_file options on all nodes. (The authorization feature does not apply to the
statestored daemon, which has no access to schema objects or data files.)
• Set up authentication using Kerberos, to make sure users really are who they say they are.
host to the network and creates user IDs that match your privileged IDs, they will be blocked from connecting to
Impala at all from that host.
Note:
If the private key is encrypted using a passphrase, Impala will ask for that passphrase on startup, which
is not useful for a large cluster. In that case, remove the passphrase and make the .pem file readable
only by Impala and administrators.
When you turn on TLS/SSL for the Impala web UI, the associated URLs change from http:// prefixes
to https://. Adjust any bookmarks or application code that refers to those URLs.
Important:
• You can use either Cloudera Manager or the following command-line instructions to complete
this configuration.
• This information applies specifically to CDH 5.6.x. If you use an earlier version of CDH, see the
documentation for that version located at Cloudera Documentation.
Property Description
Enable TLS/SSL for Impala Encrypt communication between clients (like ODBC, JDBC, and the Impala shell)
Client Services and the Impala daemon using Transport Layer Security (TLS) (formerly known
as Secure Socket Layer (SSL)).
SSL/TLS Certificate for Clients Local path to the X509 certificate that identifies the Impala daemon to clients
during TLS/SSL connections. This file must be in PEM format.
SSL/TLS Private Key for Clients Local path to the private key that matches the certificate specified in the
Certificate for Clients. This file must be in PEM format.
SSL/TLS Private Key Password A shell command for Impala to run on startup to retrieve the password for a
for Clients password-protected private key file. The output of the command is truncated
to a maximum of 1024 bytes, and any trailing whitespace (such as spaces or
newline characters) is trimmed. If the command exits with an error, Impala does
not start. If the password is incorrect, clients cannot connect to the server
regardless of whether the public key is correct.
SSL/TLS CA Certificate Must be specified for TLS/SSL encryption to be enabled for communication
between internal Impala components.
SSL/TLS Certificate for Impala There are three of these configuration settings, one each for “Impala Daemon”,
component Webserver “Catalog Server”, and “Statestore”. Each of these Impala components has its
own internal web server that powers the associated web UI with diagnostic
information. The configuration setting represents the local path to the X509
certificate that identifies the web server to clients during TLS/SSL connections.
This file must be in PEM format.
For information on configuring TLS/SSL communication with the impala-shell interpreter, see Configuring TLS/SSL
Communication for the Impala Shell on page 95.
Warning: Prior to CDH 5.5.2 / Impala 2.3.2, you could enable Kerberos authentication between Impala
internal components, or SSL encryption between Impala internal components, but not both at the
same time. This restriction has now been lifted. See IMPALA-2598 to see the maintenance releases
for different levels of CDH where the fix has been published.
If either of these flags are set, both must be set. In that case, Impala starts listening for Beeswax and HiveServer2
requests on SSL-secured ports only. (The port numbers stay the same; see Ports Used by Impala on page 583 for details.)
Since Impala uses passphrase-less certificates in PEM format, you can reuse a host's existing Java keystore by converting
it to the PEM format. For instructions, see
http://www.cloudera.com/documentation/enterprise/latest/topics/cm_sg_openssl_jks.html.
a secure production environment. When authorization is enabled, Impala uses the OS user ID of the user who runs
impala-shell or other client program, and associates various privileges with each user.
Note: Sentry is typically used in conjunction with Kerberos authentication, which defines which hosts
are allowed to connect to each server. Using the combination of Sentry and Kerberos prevents malicious
users from being able to connect by creating a named account on an untrusted machine. See Enabling
Kerberos Authentication for Impala on page 106 for details about Kerberos authentication.
• Specifying the -authorization_policy_file option in addition to -server_name makes Impala read privilege
information from a policy file, rather than from the metastore database. The argument to the
-authorization_policy_file option specifies the HDFS path to the policy file that defines the privileges on
different schema objects.
For example, you might adapt your /etc/default/impala configuration to contain lines like the following. To use
the Sentry service rather than the policy file:
IMPALA_SERVER_ARGS=" \
-server_name=server1 \
...
IMPALA_SERVER_ARGS=" \
-authorization_policy_file=/user/hive/warehouse/auth-policy.ini \
-server_name=server1 \
...
The preceding examples set up a symbolic name of server1 to refer to the current instance of Impala. This symbolic
name is used in the following ways:
• In an environment managed by Cloudera Manager, the server name is specified through Impala (Service-Wide) >
Category > Advanced > Sentry Service and Hive > Service-Wide > Advanced > Sentry Service. The values must
be the same for both, so that Impala and Hive can share the privilege rules. Restart the Impala and Hive services
after setting or changing this value.
• In an environment not managed by Cloudera Manager, you specify this value for the sentry.hive.server
property in the sentry-site.xml configuration file for Hive, as well as in the -server_name option for impalad.
If the impalad daemon is not already running, start it as described in Starting Impala on page 42. If it is already
running, restart it with the command sudo /etc/init.d/impala-server restart. Run the appropriate
commands on all the nodes where impalad normally runs.
• If you use the mode of operation using the policy file, the rules in the [roles] section of the policy file refer to
this same server1 name. For example, the following rule sets up a role report_generator that lets users with
that role query any table in a database named reporting_db on a node where the impalad daemon was started
up with the -server_name=server1 option:
[roles]
report_generator = server=server1->db=reporting_db->table=*->action=SELECT
When impalad is started with one or both of the -server_name=server1 and -authorization_policy_file
options, Impala authorization is enabled. If Impala detects any errors or inconsistencies in the authorization settings
or the policy file, the daemon refuses to start.
Using Impala with the Sentry Service (CDH 5.1 or higher only)
When you use the Sentry service rather than the policy file, you set up privileges through GRANT and REVOKE statement
in either Impala or Hive, then both components use those same privileges automatically. (Impala added the GRANT
and REVOKE statements in Impala 2.0.0 / CDH 5.2.0.)
Hive already had GRANT and REVOKE statements prior to CDH 5.1, but those statements were not production-ready.
CDH 5.1 is the first release where those statements use the Sentry framework and are considered GA level. If you used
the Hive GRANT and REVOKE statements prior to CDH 5.1, you must set up these privileges with the CDH 5.1 versions
of GRANT and REVOKE to take advantage of Sentry authorization.
For information about using the updated Hive GRANT and REVOKE statements, see Sentry service topic in the CDH 5
Security Guide.
Note:
In CDH 5 and higher, Cloudera recommends managing privileges through SQL statements, as described
in Using Impala with the Sentry Service (CDH 5.1 or higher only) on page 97. If you are still using policy
files, plan to migrate to the new approach some time in the future.
The location of the policy file is listed in the auth-site.xml configuration file. To minimize overhead, the security
information from this file is cached by each impalad daemon and refreshed automatically, with a default interval of
5 minutes. After making a substantial change to security policies, restart all Impala daemons to pick up the changes
immediately.
server=server_name->db=database_name->table=table_name->action=SELECT
server=server_name->db=database_name->table=table_name->action=CREATE
server=server_name->db=database_name->table=table_name->action=ALL
For the server_name value, substitute the same symbolic name you specify with the impalad -server_name option.
You can use * wildcard characters at each level of the privilege specification to allow access to all such objects. For
example:
server=impala-host.example.com->db=default->table=t1->action=SELECT
server=impala-host.example.com->db=*->table=*->action=CREATE
server=impala-host.example.com->db=*->table=audit_log->action=SELECT
server=impala-host.example.com->db=default->table=t1->action=*
When authorization is enabled, Impala uses the policy file as a whitelist, representing every privilege available to any
user on any object. That is, only operations specified for the appropriate combination of object, role, group, and user
are allowed; all other operations are not allowed. If a group or role is defined multiple times in the policy file, the last
definition takes precedence.
To understand the notion of whitelisting, set up a minimal policy file that does not provide any privileges for any object.
When you connect to an Impala node where this policy file is in effect, you get no results for SHOW DATABASES, and
an error when you issue any SHOW TABLES, USE database_name, DESCRIBE table_name, SELECT, and or other
statements that expect to access databases or tables, even if the corresponding databases and tables exist.
The contents of the policy file are cached, to avoid a performance penalty for each query. The policy file is re-checked
by each impalad node every 5 minutes. When you make a non-time-sensitive change such as adding new privileges
or new users, you can let the change take effect automatically a few minutes later. If you remove or reduce privileges,
and want the change to take effect immediately, restart the impalad daemon on all nodes, again specifying the
-server_name and -authorization_policy_file options so that the rules from the updated policy file are
applied.
[groups]
supergroup = all_databases
[roles]
read_all_tables = server=server1->db=*->table=*->action=SELECT
all_tables = server=server1->db=*->table=*
all_databases = server=server1->db=*
one_database = server=server1->db=test_db
entire_server = server=server1
The following sample policy file shows some of the syntax that is appropriate as the policy file grows, such as the #
comment syntax, \ continuation syntax, and comma separation for roles assigned to groups or privileges assigned to
roles.
[groups]
cloudera = training_sysadmin, instructor
visitor = student
[roles]
training_sysadmin = server=server1->db=training, \
server=server1->db=instructor_private, \
server=server1->db=lesson_development
instructor = server=server1->db=training->table=*->action=*, \
server=server1->db=instructor_private->table=*->action=*, \
server=server1->db=lesson_development->table=lesson*
# This particular course is all about queries, so the students can SELECT but not INSERT
or CREATE/DROP.
student = server=server1->db=training->table=lesson_*->action=SELECT
[groups]
cloudera = external_table, staging_dir
[roles]
external_table_admin = server=server1->db=external_table
external_table = server=server1->db=external_table->table=sample->action=*
staging_dir =
server=server1->uri=hdfs://127.0.0.1:8020/user/cloudera/external_data->action=*
impala-shell session:
| sample |
+--------+
Returned 1 row(s) in 0.02s
[groups]
supergroup = training_sysadmin
cloudera = instructor
visitor = student
[roles]
training_sysadmin = server=server1->db=training
instructor = server=server1->db=training->table=*->action=*
student = server=server1->db=training->table=*->action=SELECT
[databases]
# Defines the location of the per-DB policy files for the 'customers' and 'sales'
databases.
customers = hdfs://ha-nn-uri/etc/access/customers.ini
sales = hdfs://ha-nn-uri/etc/access/sales.ini
To enable URIs in per-DB policy files, add the following string in the Cloudera Manager field Impala Service Environment
Advanced Configuration Snippet (Safety Valve):
JAVA_TOOL_OPTIONS="-Dsentry.allow.uri.db.policyfile=true"
Important: Enabling URIs in per-DB policy files introduces a security risk by allowing the owner of
the db-level policy file to grant himself/herself load privileges to anything the impala user has read
permissions for in HDFS (including data in other databases controlled by different db-level policy files).
Server
URI
Database
Table
The server name is specified by the -server_name option when impalad starts. Specify the same name for all
impalad nodes in the cluster.
URIs represent the HDFS paths you specify as part of statements such as CREATE EXTERNAL TABLE and LOAD DATA.
Typically, you specify what look like UNIX paths, but these locations can also be prefixed with hdfs:// to make clear
that they are really URIs. To set privileges for a URI, specify the name of a directory, and the privilege applies to all the
files in that directory and any directories underneath it.
In CDH 5.5 / Impala 2.3 and higher, you can specify privileges for individual columns, as described in
https://www.cloudera.com/documentation/enterprise/latest/topics/sg_hive_sql.html. Formerly, to specify read
privileges at this level, you created a view that queried specific columns and/or partitions from a base table, and gave
SELECT privilege on the view but not the underlying table.
URIs must start with either hdfs:// or file://. If a URI starts with anything else, it will cause an exception and the
policy file will be invalid. When defining URIs for HDFS, you must also specify the NameNode. For example:
data_read = server=server1->uri=file:///path/to/dir, \
server=server1->uri=hdfs://namenode:port/path/to/dir
Warning:
Because the NameNode host and port must be specified, Cloudera strongly recommends you use High
Availability (HA). This ensures that the URI will remain constant even if the NameNode changes.
data_read = server=server1->uri=file:///path/to/dir,\
server=server1->uri=hdfs://ha-nn-uri/path/to/dir
Privilege Object
INSERT DB, TABLE
SELECT DB, TABLE, COLUMN
ALL SERVER, TABLE, DB, URI
Note:
Although this document refers to the ALL privilege, currently if you use the policy file mode, you do
not use the actual keyword ALL in the policy file. When you code role entries in the policy file:
• To specify the ALL privilege for a server, use a role like server=server_name.
• To specify the ALL privilege for a database, use a role like
server=server_name->db=database_name.
• To specify the ALL privilege for a table, use a role like
server=server_name->db=database_name->table=table_name->action=*.
which indicate each evaluation Sentry makes. The FilePermission is from the policy file, while RequestPermission
is the privilege required for the query. A RequestPermission will iterate over all appropriate FilePermission
settings until a match is found. If no matching privilege is found, Sentry returns false indicating “Access Denied” .
Note: The Hive/Impala > Service-Wide > Policy File Based Sentry tab contains parameters only
relevant to configuring Sentry using policy files. In particular, make sure that Enable Sentry
Authorization using Policy Files parameter is unchecked when using the Sentry service. Cloudera
Manager throws a validation error if you attempt to configure the Sentry service and policy file at the
same time.
Impala Authentication
Authentication is the mechanism to ensure that only specified hosts and users can connect to Impala. It also verifies
that when clients connect to Impala, they are connected to a legitimate server. This feature prevents spoofing such as
impersonation (setting up a phony client system with the same account and group names as a legitimate user) and
man-in-the-middle attacks (intercepting application requests before they reach Impala and eavesdropping on sensitive
information in the requests or the results).
Impala supports authentication using either Kerberos or LDAP.
Note: Regardless of the authentication mechanism used, Impala always creates HDFS directories and
data files owned by the same user (typically impala). To implement user-level access to different
databases, tables, columns, partitions, and so on, use the Sentry authorization feature, as explained
in Enabling Sentry Authorization for Impala on page 95.
Once you are finished setting up authentication, move on to authorization, which involves specifying what databases,
tables, HDFS directories, and so on can be accessed by particular users when they connect through Impala. See Enabling
Sentry Authorization for Impala on page 95 for details.
In Impala 2.0 and later, user() returns the full Kerberos principal string, such as user@example.com, in a Kerberized
environment.
Note: Regardless of the authentication mechanism used, Impala always creates HDFS directories and
data files owned by the same user (typically impala). To implement user-level access to different
databases, tables, columns, partitions, and so on, use the Sentry authorization feature, as explained
in Enabling Sentry Authorization for Impala on page 95.
An alternative form of authentication you can use is LDAP, described in Enabling LDAP Authentication for Impala on
page 109.
Important:
• If you plan to use Impala in your cluster, you must configure your KDC to allow tickets to be
renewed, and you must configure krb5.conf to request renewable tickets. Typically, you can
do this by adding the max_renewable_life setting to your realm in kdc.conf, and by adding
the renew_lifetime parameter to the libdefaults section of krb5.conf.
For more information about renewable tickets, see the Kerberos documentation.
• The Impala Web UI does not support Kerberos authentication.
• You cannot use the Impala resource management feature on a cluster that has Kerberos
authentication enabled.
Start all impalad and statestored daemons with the --principal and --keytab-file flags set to the principal
and full path name of the keytab file containing the credentials for the principal.
Impala supports the Cloudera ODBC driver and the Kerberos interface provided. To use Kerberos through the ODBC
driver, the host type must be set depending on the level of the ODBD driver:
• SecImpala for the ODBC 1.0 driver.
• SecBeeswax for the ODBC 1.2 driver.
• Blank for the ODBC 2.0 driver or higher, when connecting to a secure cluster.
• HS2NoSasl for the ODBC 2.0 driver or higher, when connecting to a non-secure cluster.
To enable Kerberos in the Impala shell, start the impala-shell command using the -k flag.
To enable Impala to work with Kerberos security on your Hadoop cluster, make sure you perform the installation and
configuration steps in Authentication in the CDH 5 Security Guide.
$ kadmin
kadmin: addprinc -requires_preauth -randkey
impala/impala_host.example.com@TEST.EXAMPLE.COM
Note: The HTTP component of the service principal must be uppercase as shown in the preceding
example.
4. Use ktutil to read the contents of the two keytab files and then write those contents to a new file. For example:
$ ktutil
ktutil: rkt impala.keytab
ktutil: rkt http.keytab
ktutil: wkt impala-http.keytab
ktutil: quit
5. (Optional) Test that credentials in the merged keytab file are valid, and that the “renew until” date is in the future.
For example:
$ klist -e -k -t impala-http.keytab
6. Copy the impala-http.keytab file to the Impala configuration directory. Change the permissions to be only
read for the file owner and change the file owner to the impala user. By default, the Impala user and group are
both named impala. For example:
$ cp impala-http.keytab /etc/impala/conf
$ cd /etc/impala/conf
$ chmod 400 impala-http.keytab
$ chown impala:impala impala-http.keytab
7. Add Kerberos options to the Impala defaults file, /etc/default/impala. Add the options for both the impalad
and statestored daemons, using the IMPALA_SERVER_ARGS and IMPALA_STATE_STORE_ARGS variables. For
example, you might add:
-kerberos_reinit_interval=60
-principal=impala_1/impala_host.example.com@TEST.EXAMPLE.COM
-keytab_file=/var/run/cloudera-scm-agent/process/3212-impala-IMPALAD/impala.keytab
For more information on changing the Impala defaults specified in /etc/default/impala, see Modifying Impala
Startup Options.
Note: Restart impalad and statestored for these configuration changes to take effect.
Note: Regardless of the authentication mechanism used, Impala always creates HDFS directories and
data files owned by the same user (typically impala). To implement user-level access to different
databases, tables, columns, partitions, and so on, use the Sentry authorization feature, as explained
in Enabling Sentry Authorization for Impala on page 95.
An alternative form of authentication you can use is Kerberos, described in Enabling Kerberos Authentication for Impala
on page 106.
[impala]
auth_username=<LDAP username of Hue user to be authenticated>
auth_password=<LDAP password of Hue user to be authenticated>
These login details are only used by Impala to authenticate to LDAP. The Impala service trusts Hue to have already
validated the user being impersonated, rather than simply passing on the credentials.
Note: Make sure to use single quotes or escape characters to ensure that any * characters do not
undergo wildcard expansion when specified in command-line arguments.
See Modifying Impala Startup Options on page 43 for details about adding or changing impalad startup options. See
this Cloudera blog post for background information about the delegation capability in HiveServer2.
To set up authentication for the delegated users:
• On the server side, configure either user/password authentication through LDAP, or Kerberos authentication, for
all the delegated users. See Enabling LDAP Authentication for Impala on page 109 or Enabling Kerberos Authentication
for Impala on page 106 for details.
• On the client side, follow the instructions in the “Using User Name and Password” section in the ODBC driver
installation guide. Then search for “delegation” in that same installation guide to learn about the Delegation UID
field and DelegationUID configuration keyword to enable the delegation feature for ODBC-based BI tools.
Because any Impala host can process a query, enable auditing on all hosts where the impalad daemon runs. Each
host stores its own log files, in a directory in the local filesystem. The log data is periodically flushed to disk (through
an fsync() system call) to avoid loss of audit data in case of a crash.
The runtime overhead of auditing applies to whichever host serves as the coordinator for the query, that is, the host
you connect to when you issue the query. This might be the same host for all queries, or different applications or users
might connect to and issue queries through different hosts.
To avoid excessive I/O overhead on busy coordinator hosts, Impala syncs the audit log data (using the fsync() system
call) periodically rather than after every query. Currently, the fsync() calls are issued at a fixed interval, every 5
seconds.
By default, Impala avoids losing any audit log data in the case of an error during a logging operation (such as a disk full
error), by immediately shutting down impalad on the host where the auditing problem occurred. You can override
this setting by specifying the option -abort_on_failed_audit_event=false in the impalad startup options.
The audit log does not contain entries for queries that could not be parsed and analyzed. For example, a query that
fails due to a syntax error is not recorded in the audit log. The audit log also does not contain queries that fail due to
a reference to a table that does not exist, if you would be authorized to access the table if it did exist.
Certain statements in the impala-shell interpreter, such as CONNECT, SUMMARY, PROFILE, SET, and QUIT, do not
correspond to actual SQL queries, and these statements are not reflected in the audit log.
Column Lineage
Column lineage tracks information in fine detail, at the level of particular columns rather than entire tables.
For example, if you have a table with information derived from web logs, you might copy that data into other tables
as part of the ETL process. The ETL operations might involve transformations through expressions and function calls,
and rearranging the columns into more or fewer tables (normalizing or denormalizing the data). Then for reporting,
you might issue queries against multiple tables and views. In this example, column lineage helps you determine that
data that entered the system as RAW_LOGS.FIELD1 was then turned into WEBSITE_REPORTS.IP_ADDRESS through
an INSERT ... SELECT statement. Or, conversely, you could start with a reporting query against a view, and trace
the origin of the data in a field such as TOP_10_VISITORS.USER_ID back to the underlying table and even further
back to the point where the data was first loaded into Impala.
When you have tables where you need to track or control access to sensitive information at the column level, see
Enabling Sentry Authorization for Impala on page 95 for how to implement column-level security. You set up
authorization using the Sentry framework, create views that refer to specific sets of columns, and then assign
authorization privileges to those views rather than the underlying tables.
To enable or disable this feature on a system not managed by Cloudera Manager, set or remove the
-lineage_event_log_dir configuration option for the impalad daemon. For information about turning the lineage
feature on and off through Cloudera Manager, see
http://www.cloudera.com/documentation/enterprise/latest/topics/datamgmt_impala_lineage_log.html.
Impala supports data types with the same names and semantics as the equivalent Hive data types: string, TINYINT,
SMALLINT, INT, BIGINT, FLOAT, DOUBLE, BOOLEAN, STRING, TIMESTAMP.
For full details about Impala SQL syntax and semantics, see Impala SQL Statements on page 214.
Most HiveQL SELECT and INSERT statements run unmodified with Impala. For information about Hive syntax not
available in Impala, see SQL Differences Between Impala and Hive on page 461.
For a list of the built-in functions available in Impala queries, see Impala Built-In Functions on page 338.
Comments
Impala supports the familiar styles of SQL comments:
• All text from a -- sequence to the end of the line is considered a comment and ignored. This type of comment
can occur on a single line by itself, or after all or part of a statement.
• All text from a /* sequence to the next */ sequence is considered a comment and ignored. This type of comment
can stretch over multiple lines. This type of comment can occur on one or more lines by itself, in the middle of a
statement, or before or after a statement.
For example:
/*
This is a multi-line comment about a query.
*/
select ...;
Data Types
Impala supports a set of data types that you can use for table columns, expression values, and function arguments and
return values.
Note: Currently, Impala supports only scalar types, not composite or nested types. Accessing a table
containing any columns with unsupported types causes an error.
For the notation to write literals of each of these data types, see Literals on page 184.
See SQL Differences Between Impala and Hive on page 461 for differences between Impala and Hive data types.
Usage notes:
Because complex types are often used in combination, for example an ARRAY of STRUCT elements, if you are unfamiliar
with the Impala complex types, start with Complex Types (CDH 5.5 or higher only) on page 156 for background information
and usage examples.
The elements of the array have no names. You refer to the value of the array item using the ITEM pseudocolumn, or
its position in the array with the POS pseudocolumn. See ITEM and POS Pseudocolumns on page 170 for information
about these pseudocolumns.
Each row can have a different number of elements (including none) in the array for that row.
When an array contains items of scalar types, you can use aggregation functions on the array elements without using
join notation. For example, you can find the COUNT(), AVG(), SUM(), and so on of numeric array elements, or the
MAX() and MIN() of any scalar array elements by referring to table_name.array_column in the FROM clause of
the query. When you need to cross-reference values from the array with scalar values from the same row, such as by
including a GROUP BY clause to produce a separate aggregated result for each row, then the join clause is required.
A common usage pattern with complex types is to have an array as the top-level type for the column: an array of
structs, an array of maps, or an array of arrays. For example, you can model a denormalized table by creating a column
that is an ARRAY of STRUCT elements; each item in the array represents a row from a table that would normally be
used in a join query. This kind of data structure lets you essentially denormalize tables by associating multiple rows
from one table with the matching row in another table.
You typically do not create more than one top-level ARRAY column, because if there is some relationship between the
elements of multiple arrays, it is convenient to model the data as an array of another complex type element (either
STRUCT or MAP).
You can pass a qualified name to DESCRIBE to specify an ARRAY, STRUCT, or MAP column and visualize its structure
as if it were a table. An ARRAY is shown as a two-column table, with ITEM and POS columns. A STRUCT is shown as a
table with each field representing a column in the table. A MAP is shown as a two-column table, with KEY and VALUE
columns.
Added in: CDH 5.5.0 (Impala 2.3.0)
Restrictions:
• Columns with this data type can only be used in tables or partitions with the Parquet file format.
• Columns with this data type cannot be used as partition key columns in a partitioned table.
• The COMPUTE STATS statement does not produce any statistics for columns of this data type.
• The maximum length of the column definition for any complex type, including declarations for any nested types,
is 4000 characters.
• See Limitations and Restrictions for Complex Types on page 161 for a full list of limitations and associated guidelines
about complex type columns.
Examples:
Note: Many of the complex type examples refer to tables such as CUSTOMER and REGION adapted
from the tables used in the TPC-H benchmark. See Sample Schema and Data for Experimenting with
Impala Complex Types on page 178 for the table definitions.
The following example shows how to construct a table with various kinds of ARRAY columns, both at the top level and
nested within other complex types. Whenever the ARRAY consists of a scalar value, such as in the PETS column or the
CHILDREN field, you can see that future expansion is limited. For example, you could not easily evolve the schema to
record the kind of pet or the child's birthday alongside the name. Therefore, it is more common to use an ARRAY whose
elements are of STRUCT type, to associate multiple fields with each array element.
Note: Practice the CREATE TABLE and query notation for complex type columns using empty tables,
until you can visualize a complex data structure and construct corresponding SQL statements reliably.
The following example shows how to examine the structure of a table containing one or more ARRAY columns by using
the DESCRIBE statement. You can visualize each ARRAY as its own two-column table, with columns ITEM and POS.
DESCRIBE array_demo;
+--------------+---------------------------+
| name | type |
+--------------+---------------------------+
| id | bigint |
| name | string |
| pets | array<string> |
| marriages | array<struct< |
| | spouse:string, |
| | children:array<string> |
| | >> |
| places_lived | array<struct< |
| | place:string, |
| | start_year:int |
| | >> |
| ancestors | map<string,array<string>> |
+--------------+---------------------------+
DESCRIBE array_demo.pets;
+------+--------+
| name | type |
+------+--------+
| item | string |
| pos | bigint |
+------+--------+
DESCRIBE array_demo.marriages;
+------+--------------------------+
| name | type |
+------+--------------------------+
| item | struct< |
| | spouse:string, |
| | children:array<string> |
| | > |
| pos | bigint |
+------+--------------------------+
DESCRIBE array_demo.places_lived;
+------+------------------+
| name | type |
+------+------------------+
| item | struct< |
| | place:string, |
| | start_year:int |
| | > |
| pos | bigint |
+------+------------------+
DESCRIBE array_demo.ancestors;
+-------+---------------+
| name | type |
+-------+---------------+
| key | string |
| value | array<string> |
+-------+---------------+
The following example shows queries involving ARRAY columns containing elements of scalar or complex types. You
“unpack” each ARRAY column by referring to it in a join query, as if it were a separate table with ITEM and POS columns.
If the array element is a scalar type, you refer to its value using the ITEM pseudocolumn. If the array element is a
STRUCT, you refer to the STRUCT fields using dot notation and the field names. If the array element is another ARRAY
or a MAP, you use another level of join to unpack the nested collection elements.
-- Array of structs.
-- Now each array element has named fields, possibly of different types.
-- You can consider an ARRAY of STRUCT to represent a table inside another table.
SELECT id, name, places_lived.pos, places_lived.item.place, places_lived.item.start_year
FROM array_demo, array_demo.places_lived;
-- The .ITEM name is optional for array elements that are structs.
-- The following query is equivalent to the previous one, with .ITEM
-- removed from the column references.
SELECT id, name, places_lived.pos, places_lived.place, places_lived.start_year
FROM array_demo, array_demo.places_lived;
-- To filter specific items from the array, do comparisons against the .POS or .ITEM
-- pseudocolumns, or names of struct fields, in the WHERE clause.
SELECT id, name, pets.item FROM array_demo, array_demo.pets
WHERE pets.pos in (0, 1, 3);
Related information:
Complex Types (CDH 5.5 or higher only) on page 156, STRUCT Complex Type (CDH 5.5 or higher only) on page 142, MAP
Complex Type (CDH 5.5 or higher only) on page 136
column_name BIGINT
Usage notes:
BIGINT is a convenient type to use for column declarations because you can use any kind of integer values in INSERT
statements and they are promoted to BIGINT where necessary. However, BIGINT also requires the most bytes of
any integer type on disk and in memory, meaning your queries are not as efficient and scalable as possible if you
overuse this type. Therefore, prefer to use the smallest integer type with sufficient range to hold all input values, and
CAST() when necessary to the appropriate type.
For a convenient and automated way to check the bounds of the BIGINT type, call the functions MIN_BIGINT() and
MAX_BIGINT().
If an integer value is too large to be represented as a BIGINT, use a DECIMAL instead with sufficient digits of precision.
NULL considerations: Casting any non-numeric value to this type produces a NULL value.
Partitioning: Prefer to use this type for a partition key column. Impala can process the numeric type more efficiently
than a STRING representation of the value.
HBase considerations: This data type is fully compatible with HBase tables.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Internal details: Represented in memory as an 8-byte value.
Added in: Available in all versions of Impala.
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
Sqoop considerations:
If you use Sqoop to convert RDBMS data to Parquet, be careful with interpreting any resulting values from DATE,
DATETIME, or TIMESTAMP columns. The underlying values are represented as the Parquet INT64 type, which is
represented as BIGINT in the Impala table. The Parquet values represent the time in milliseconds, while Impala
interprets BIGINT as the time in seconds. Therefore, if you have a BIGINT column in a Parquet table that was imported
this way from Sqoop, divide the values by 1000 when interpreting as the TIMESTAMP type.
Related information:
Numeric Literals on page 184, TINYINT Data Type on page 153, SMALLINT Data Type on page 140, INT Data Type on page
135, BIGINT Data Type on page 120, DECIMAL Data Type (CDH 5.1 or higher only) on page 125, Impala Mathematical
Functions on page 339
column_name BOOLEAN
Range: TRUE or FALSE. Do not use quotation marks around the TRUE and FALSE literal values. You can write the literal
values in uppercase, lowercase, or mixed case. The values queried from a table are always returned in lowercase, true
or false.
Conversions: Impala does not automatically convert any other type to BOOLEAN. All conversions must use an explicit
call to the CAST() function.
You can use CAST() to convert any integer or floating-point type to BOOLEAN: a value of 0 represents false, and any
non-zero value is converted to true.
When you cast the opposite way, from BOOLEAN to a numeric type, the result becomes either 1 or 0:
You can cast DECIMAL values to BOOLEAN, with the same treatment of zero and non-zero values as the other numeric
types. You cannot cast a BOOLEAN to a DECIMAL.
You cannot cast a STRING value to BOOLEAN, although you can cast a BOOLEAN value to STRING, returning '1' for
true values and '0' for false values.
Although you can cast a TIMESTAMP to a BOOLEAN or a BOOLEAN to a TIMESTAMP, the results are unlikely to be useful.
Any non-zero TIMESTAMP (that is, any value other than 1970-01-01 00:00:00) becomes TRUE when converted to
BOOLEAN, while 1970-01-01 00:00:00 becomes FALSE. A value of FALSE becomes 1970-01-01 00:00:00 when
converted to BOOLEAN, and TRUE becomes one second past this epoch date, that is, 1970-01-01 00:00:01.
NULL considerations: An expression of this type produces a NULL value if any argument of the expression is NULL.
Partitioning:
Do not use a BOOLEAN column as a partition key. Although you can create such a table, subsequent operations produce
errors:
ERROR: AnalysisException: INSERT into table with BOOLEAN partition column (truth) is
not supported: partitioning.truth_table
Examples:
SELECT 1 < 2;
SELECT 2 = 5;
SELECT 100 < NULL, 100 > NULL;
CREATE TABLE assertions (claim STRING, really BOOLEAN);
INSERT INTO assertions VALUES
("1 is less than 2", 1 < 2),
("2 is the same as 5", 2 = 5),
("Grass is green", true),
("The moon is made of green cheese", false);
SELECT claim FROM assertions WHERE really = TRUE;
HBase considerations: This data type is fully compatible with HBase tables.
Parquet considerations: This type is fully compatible with Parquet tables.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
Related information: Boolean Literals on page 187, SQL Operators on page 188, Impala Conditional Functions on page
391
column_name CHAR(length)
• Any trailing spaces, whether implicitly or explicitly specified, are not written to the Parquet data files.
• Parquet data files might contain values that are longer than allowed by the CHAR(n) length limit. Impala ignores
any extra trailing characters when it processes those values during a query.
Text table considerations:
Text data files might contain values that are longer than allowed for a particular CHAR(n) column. Any extra trailing
characters are ignored when Impala processes those values during a query. Text data files can also contain values that
are shorter than the defined length limit, and Impala pads them with trailing spaces up to the specified length. Any
text data files produced by Impala INSERT statements do not include any trailing blanks for CHAR columns.
Avro considerations:
The Avro specification allows string values up to 2**64 bytes in length. Impala queries for Avro tables use 32-bit integers
to hold string lengths. In CDH 5.7 / Impala 2.5 and higher, Impala truncates CHAR and VARCHAR values in Avro tables
to (2**31)-1 bytes. If a query encounters a STRING value longer than (2**31)-1 bytes in an Avro table, the query fails.
In earlier releases, encountering such long values in an Avro table could cause a crash.
Compatibility:
This type is available using Impala 2.0 or higher under CDH 4, or with Impala on CDH 5.2 or higher. There are no
compatibility issues with other components when exchanging data files or running Impala on CDH 4.
Some other database systems make the length specification optional. For Impala, the length is required.
Internal details: Represented in memory as a byte array with the same size as the length specification. Values that are
shorter than the specified length are padded on the right with trailing spaces.
Added in: CDH 5.2.0 (Impala 2.0.0)
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
UDF considerations: This type cannot be used for the argument or return type of a user-defined function (UDF) or
user-defined aggregate function (UDA).
Examples:
These examples show how trailing spaces are not considered significant when comparing or processing CHAR values.
CAST() truncates any longer string to fit within the defined length. If a CHAR value is shorter than the specified length,
it is padded on the right with spaces until it matches the specified length. Therefore, LENGTH() represents the length
including any trailing spaces, and CONCAT() also treats the column value as if it has trailing spaces.
This example shows a case where data values are known to have a specific length, where CHAR is a logical data type
to use.
The following example shows how values written by Impala do not physically include the trailing spaces. It creates a
table using text format, with CHAR values much shorter than the declared length, and then prints the resulting data
file to show that the delimited values are not separated by spaces. The same behavior applies to binary-format Parquet
data files.
The following example further illustrates the treatment of spaces. It replaces the contents of the previous table with
some values including leading spaces, trailing spaces, or both. Any leading spaces are preserved within the data file,
but trailing spaces are discarded. Then when the values are retrieved by a query, the leading spaces are retrieved
verbatim while any necessary trailing spaces are supplied by Impala.
Restrictions:
Because the blank-padding behavior requires allocating the maximum length for each value in memory, for scalability
reasons avoid declaring CHAR columns that are much longer than typical values in that column.
All data in CHAR and VARCHAR columns must be in a character encoding that is compatible with UTF-8. If you have
binary data from another database system (that is, a BLOB type), use a STRING column to hold it.
When an expression compares a CHAR with a STRING or VARCHAR, the CHAR value is implicitly converted to STRING
first, with trailing spaces preserved.
This behavior differs from other popular database systems. To get the expected result of TRUE, cast the expressions
on both sides to CHAR values of the appropriate length:
column_name DECIMAL[(precision[,scale])]
• For addition and subtraction, the precision and scale are based on the maximum possible result, that is, if all the
digits of the input values were 9s and the absolute values were added together.
• For multiplication, the precision is the sum of the precisions of the input values. The scale is the sum of the scales
of the input values.
• For division, Impala sets the precision and scale to values large enough to represent the whole and fractional parts
of the result.
• For UNION, the scale is the larger of the scales of the input values, and the precision is increased if necessary to
accommodate any additional fractional digits. If the same input value has the largest precision and the largest
scale, the result value has the same precision and scale. If one value has a larger precision but smaller scale, the
scale of the result value is increased. For example, DECIMAL(20,2) UNION DECIMAL(8,6) produces a result
of type DECIMAL(24,6). The extra 4 fractional digits of scale (6-2) are accommodated by extending the precision
by the same amount (20+4).
• To doublecheck, you can always call the PRECISION() and SCALE() functions on the results of an arithmetic
expression to see the relevant values, or use a CREATE TABLE AS SELECT statement to define a column based
on the return type of the expression.
Compatibility:
• Using the DECIMAL type is only supported under CDH 5.1.0 and higher.
• Use the DECIMAL data type in Impala for applications where you used the NUMBER data type in Oracle. The Impala
DECIMAL type does not support the Oracle idioms of * for scale or negative values for precision.
To avoid potential conversion errors, you can use CAST() to convert DECIMAL values to FLOAT, TINYINT, SMALLINT,
INT, BIGINT, STRING, TIMESTAMP, or BOOLEAN. You can use exponential notation in DECIMAL literals or when casting
from STRING, for example 1.0e6 to represent one million.
If you cast a value with more fractional digits than the scale of the destination type, any extra fractional digits are
truncated (not rounded). Casting a value to a target type with not enough precision produces a result of NULL and
displays a runtime warning.
When you specify integer literals, for example in INSERT ... VALUES statements or arithmetic expressions, those
numbers are interpreted as the smallest applicable integer type. You must use CAST() calls for some combinations
of integer literals and DECIMAL precision. For example, INT has a maximum value that is 10 digits long, TINYINT has
a maximum value that is 3 digits long, and so on. If you specify a value such as 123456 to go into a DECIMAL column,
Impala checks if the column has enough precision to represent the largest value of that integer type, and raises an
error if not. Therefore, use an expression like CAST(123456 TO DECIMAL(9,0)) for DECIMAL columns with precision
9 or less, CAST(50 TO DECIMAL(2,0)) for DECIMAL columns with precision 2 or less, and so on. For DECIMAL
columns with precision 10 or greater, Impala automatically interprets the value as the correct DECIMAL type; however,
because DECIMAL(10) requires 8 bytes of storage while DECIMAL(9) requires only 4 bytes, only use precision of 10
or higher when actually needed.
Be aware that in memory and for binary file formats such as Parquet or Avro, DECIMAL(10) or higher consumes 8
bytes while DECIMAL(9) (the default for DECIMAL) or lower consumes 4 bytes. Therefore, to conserve space in large
tables, use the smallest-precision DECIMAL type that is appropriate and CAST() literal values where necessary, rather
than declaring DECIMAL columns with high precision for convenience.
To represent a very large or precise DECIMAL value as a literal, for example one that contains more digits than can be
represented by a BIGINT literal, use a quoted string or a floating-point value for the number, and CAST() to the
desired DECIMAL type:
insert into decimals_38_5 values (1), (2), (4), (8), (16), (1024), (32768), (65536),
(1000000),
(cast("999999999999999999999999999999" as decimal(38,5))),
(cast(999999999999999999999999999999. as decimal(38,5)));
• The result of the SUM() aggregate function on DECIMAL values is promoted to a precision of 38, with the same
precision as the underlying column. Thus, the result can represent the largest possible value at that particular
precision.
• STRING columns, literals, or expressions can be converted to DECIMAL as long as the overall number of digits and
digits to the right of the decimal point fit within the specified precision and scale for the declared DECIMAL type.
By default, a DECIMAL value with no specified scale or precision can hold a maximum of 9 digits of an integer
value. If there are more digits in the string value than are allowed by the DECIMAL scale and precision, the result
is NULL.
The following examples demonstrate how STRING values with integer and fractional parts are represented when
converted to DECIMAL. If the scale is 0, the number is treated as an integer value with a maximum of precision
digits. If the precision is greater than 0, the scale must be increased to account for the digits both to the left and
right of the decimal point. As the precision increases, output values are printed with additional trailing zeros after
the decimal point if needed. Any trailing zeros after the decimal point in the STRING value must fit within the
number of digits specified by the precision.
[localhost:21000] > select cast('100' as decimal); -- Small integer value fits within
9 digits of scale.
+-----------------------------+
| cast('100' as decimal(9,0)) |
+-----------------------------+
| 100 |
+-----------------------------+
[localhost:21000] > select cast('100' as decimal(3,0)); -- Small integer value fits
within 3 digits of scale.
+-----------------------------+
| cast('100' as decimal(3,0)) |
+-----------------------------+
| 100 |
+-----------------------------+
[localhost:21000] > select cast('100' as decimal(2,0)); -- 2 digits of scale is not
enough!
+-----------------------------+
| cast('100' as decimal(2,0)) |
+-----------------------------+
| NULL |
+-----------------------------+
[localhost:21000] > select cast('100' as decimal(3,1)); -- (3,1) = 2 digits left of the
decimal point, 1 to the right. Not enough.
+-----------------------------+
| cast('100' as decimal(3,1)) |
+-----------------------------+
| NULL |
+-----------------------------+
[localhost:21000] > select cast('100' as decimal(4,1)); -- 4 digits total, 1 to the
right of the decimal point.
+-----------------------------+
| cast('100' as decimal(4,1)) |
+-----------------------------+
| 100.0 |
+-----------------------------+
[localhost:21000] > select cast('98.6' as decimal(3,1)); -- (3,1) can hold a 3 digit
number with 1 fractional digit.
+------------------------------+
| cast('98.6' as decimal(3,1)) |
+------------------------------+
| 98.6 |
+------------------------------+
[localhost:21000] > select cast('98.6' as decimal(15,1)); -- Larger scale allows bigger
numbers but still only 1 fractional digit.
+-------------------------------+
| cast('98.6' as decimal(15,1)) |
+-------------------------------+
| 98.6 |
+-------------------------------+
[localhost:21000] > select cast('98.6' as decimal(15,5)); -- Larger precision allows
more fractional digits, outputs trailing zeros.
+-------------------------------+
| cast('98.6' as decimal(15,5)) |
+-------------------------------+
| 98.60000 |
+-------------------------------+
[localhost:21000] > select cast('98.60000' as decimal(15,1)); -- Trailing zeros in the
string must fit within 'scale' digits (1 in this case).
+-----------------------------------+
| cast('98.60000' as decimal(15,1)) |
+-----------------------------------+
| NULL |
+-----------------------------------+
• Most built-in arithmetic functions such as SIN() and COS() continue to accept only DOUBLE values because they
are so commonly used in scientific context for calculations of IEEE 754-compliant values. The built-in functions
that accept and return DECIMAL are:
– ABS()
– CEIL()
– COALESCE()
– FLOOR()
– FNV_HASH()
– GREATEST()
– IF()
– ISNULL()
– LEAST()
– NEGATIVE()
– NULLIF()
– POSITIVE()
– PRECISION()
– ROUND()
– SCALE()
– TRUNCATE()
– ZEROIFNULL()
+--------------------------+
[localhost:21000] > select cast(10 as decimal(1,1));
+--------------------------+
| cast(10 as decimal(1,1)) |
+--------------------------+
| 10.0 |
+--------------------------+
[localhost:21000] > select cast(100 as decimal(1,1));
+---------------------------+
| cast(100 as decimal(1,1)) |
+---------------------------+
| 100.0 |
+---------------------------+
[localhost:21000] > select cast(1000 as decimal(1,1));
+----------------------------+
| cast(1000 as decimal(1,1)) |
+----------------------------+
| 1000.0 |
+----------------------------+
• When a DECIMAL value is converted to any of the integer types, any fractional part is truncated (that is, rounded
towards zero):
• You cannot directly cast TIMESTAMP or BOOLEAN values to or from DECIMAL values. You can turn a DECIMAL
value into a time-related representation using a two-step process, by converting it to an integer value and then
using that result in a call to a date and time function such as from_unixtime().
• Because values in INSERT statements are checked rigorously for type compatibility, be prepared to use CAST()
function calls around literals, column references, or other expressions that you are inserting into a DECIMAL
column.
NULL considerations: Casting any non-numeric value to this type produces a NULL value.
DECIMAL differences from integer and floating-point types:
With the DECIMAL type, you are concerned with the number of overall digits of a number rather than powers of 2 (as
in TINYINT, SMALLINT, and so on). Therefore, the limits with integral values of DECIMAL types fall around 99, 999,
9999, and so on rather than 32767, 65535, 2 32 -1, and so on. For fractional values, you do not need to account for
imprecise representation of the fractional part according to the IEEE-954 standard (as in FLOAT and DOUBLE). Therefore,
when you insert a fractional value into a DECIMAL column, you can compare, sum, query, GROUP BY, and so on that
column and get back the original values rather than some “close but not identical” value.
FLOAT and DOUBLE can cause problems or unexpected behavior due to inability to precisely represent certain fractional
values, for example dollar and cents values for currency. You might find output values slightly different than you
inserted, equality tests that do not match precisely, or unexpected values for GROUP BY columns. DECIMAL can help
reduce unexpected behavior and rounding errors, at the expense of some performance overhead for assignments and
comparisons.
Literals and expressions:
• When you use an integer literal such as 1 or 999 in a SQL statement, depending on the context, Impala will treat
it as either the smallest appropriate DECIMAL type, or the smallest integer type (TINYINT, SMALLINT, INT, or
BIGINT). To minimize memory usage, Impala prefers to treat the literal as the smallest appropriate integer type.
• When you use a floating-point literal such as 1.1 or 999.44 in a SQL statement, depending on the context, Impala
will treat it as either the smallest appropriate DECIMAL type, or the smallest floating-point type (FLOAT or DOUBLE).
To avoid loss of accuracy, Impala prefers to treat the literal as a DECIMAL.
Storage considerations:
• Only the precision determines the storage size for DECIMAL values; the scale setting has no effect on the storage
size.
• Text, RCFile, and SequenceFile tables all use ASCII-based formats. In these text-based file formats, leading zeros
are not stored, but trailing zeros are stored. In these tables, each DECIMAL value takes up as many bytes as there
are digits in the value, plus an extra byte if the decimal point is present and an extra byte for negative values.
Once the values are loaded into memory, they are represented in 4, 8, or 16 bytes as described in the following
list items. The on-disk representation varies depending on the file format of the table.
• Parquet and Avro tables use binary formats, In these tables, Impala stores each value in as few bytes as possible
depending on the precision specified for the DECIMAL column.
– In memory, DECIMAL values with precision of 9 or less are stored in 4 bytes.
– In memory, DECIMAL values with precision of 10 through 18 are stored in 8 bytes.
– In memory, DECIMAL values with precision greater than 18 are stored in 16 bytes.
• Parquet and Avro tables use binary formats, In these tables, Impala stores each value in 4, 8, or 16 bytes depending
on the precision specified for the DECIMAL column.
UDF considerations: When writing a C++ UDF, use the DecimalVal data type defined in
/usr/include/impala_udf/udf.h.
Partitioning:
You can use a DECIMAL column as a partition key. Doing so provides a better match between the partition key values
and the HDFS directory names than using a DOUBLE or FLOAT partitioning column:
Schema evolution considerations:
• For text-based formats (text, RCFile, and SequenceFile tables), you can issue an ALTER TABLE ... REPLACE
COLUMNS statement to change the precision and scale of an existing DECIMAL column. As long as the values in
the column fit within the new precision and scale, they are returned correctly by a query. Any values that do not
fit within the new precision and scale are returned as NULL, and Impala reports the conversion error. Leading
zeros do not count against the precision value, but trailing zeros after the decimal point do.
• For binary formats (Parquet and Avro tables), although an ALTER TABLE ... REPLACE COLUMNS statement
that changes the precision or scale of a DECIMAL column succeeds, any subsequent attempt to query the changed
column results in a fatal error. (The other columns can still be queried successfully.) This is because the metadata
about the columns is stored in the data files themselves, and ALTER TABLE does not actually make any updates
to the data files. If the metadata in the data files disagrees with the metadata in the metastore database, Impala
cancels the query.
Examples:
Restrictions:
Currently, the COMPUTE STATS statement under CDH 4 does not store any statistics for DECIMAL columns. When
Impala runs under CDH 5, which has better support for DECIMAL in the metastore database, COMPUTE STATS does
collect statistics for DECIMAL columns and Impala uses the statistics to optimize query performance.
HBase considerations: This data type is fully compatible with HBase tables.
Parquet considerations: This type is fully compatible with Parquet tables.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
Related information:
Numeric Literals on page 184, TINYINT Data Type on page 153, SMALLINT Data Type on page 140, INT Data Type on page
135, BIGINT Data Type on page 120, DECIMAL Data Type (CDH 5.1 or higher only) on page 125, Impala Mathematical
Functions on page 339 (especially PRECISION() and SCALE())
column_name DOUBLE
Partitioning: Because fractional values of this type are not always represented precisely, when this type is used for a
partition key column, the underlying HDFS directories might not be named exactly as you expect. Prefer to partition
on a DECIMAL column instead.
HBase considerations: This data type is fully compatible with HBase tables.
Parquet considerations: This type is fully compatible with Parquet tables.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Internal details: Represented in memory as an 8-byte value.
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
Restrictions:
Due to the way arithmetic on FLOAT and DOUBLE columns uses high-performance hardware instructions, and distributed
queries can perform these operations in different order for each query, results can vary slightly for aggregate function
calls such as SUM() and AVG() for FLOAT and DOUBLE columns, particularly on large data sets where millions or billions
of values are summed or averaged. For perfect consistency and repeatability, use the DECIMAL data type for such
operations instead of FLOAT or DOUBLE.
The inability to exactly represent certain floating-point values means that DECIMAL is sometimes a better choice than
DOUBLE or FLOAT when precision is critical, particularly when transferring data from other database systems that use
different representations or file formats.
Related information:
Numeric Literals on page 184, Impala Mathematical Functions on page 339, FLOAT Data Type on page 134
column_name FLOAT
Partitioning: Because fractional values of this type are not always represented precisely, when this type is used for a
partition key column, the underlying HDFS directories might not be named exactly as you expect. Prefer to partition
on a DECIMAL column instead.
HBase considerations: This data type is fully compatible with HBase tables.
Parquet considerations: This type is fully compatible with Parquet tables.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Internal details: Represented in memory as a 4-byte value.
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
Restrictions:
Due to the way arithmetic on FLOAT and DOUBLE columns uses high-performance hardware instructions, and distributed
queries can perform these operations in different order for each query, results can vary slightly for aggregate function
calls such as SUM() and AVG() for FLOAT and DOUBLE columns, particularly on large data sets where millions or billions
of values are summed or averaged. For perfect consistency and repeatability, use the DECIMAL data type for such
operations instead of FLOAT or DOUBLE.
The inability to exactly represent certain floating-point values means that DECIMAL is sometimes a better choice than
DOUBLE or FLOAT when precision is critical, particularly when transferring data from other database systems that use
different representations or file formats.
Related information:
Numeric Literals on page 184, Impala Mathematical Functions on page 339, DOUBLE Data Type on page 133
column_name INT
Partitioning: Prefer to use this type for a partition key column. Impala can process the numeric type more efficiently
than a STRING representation of the value.
HBase considerations: This data type is fully compatible with HBase tables.
Parquet considerations:
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Internal details: Represented in memory as a 4-byte value.
Added in: Available in all versions of Impala.
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
Related information:
Numeric Literals on page 184, TINYINT Data Type on page 153, SMALLINT Data Type on page 140, INT Data Type on page
135, BIGINT Data Type on page 120, DECIMAL Data Type (CDH 5.1 or higher only) on page 125, Impala Mathematical
Functions on page 339
Usage notes:
Because complex types are often used in combination, for example an ARRAY of STRUCT elements, if you are unfamiliar
with the Impala complex types, start with Complex Types (CDH 5.5 or higher only) on page 156 for background information
and usage examples.
The MAP complex data type represents a set of key-value pairs. Each element of the map is indexed by a primitive type
such as BIGINT or STRING, letting you define sequences that are not continuous or categories with arbitrary names.
You might find it convenient for modelling data produced in other languages, such as a Python dictionary or Java
HashMap, where a single scalar value serves as the lookup key.
In a big data context, the keys in a map column might represent a numeric sequence of events during a manufacturing
process, or TIMESTAMP values corresponding to sensor observations. The map itself is inherently unordered, so you
choose whether to make the key values significant (such as a recorded TIMESTAMP) or synthetic (such as a random
global universal ID).
Note: Behind the scenes, the MAP type is implemented in a similar way as the ARRAY type. Impala
does not enforce any uniqueness constraint on the KEY values, and the KEY values are processed by
looping through the elements of the MAP rather than by a constant-time lookup. Therefore, this type
is primarily for ease of understanding when importing data and algorithms from non-SQL contexts,
rather than optimizing the performance of key lookups.
You can pass a qualified name to DESCRIBE to specify an ARRAY, STRUCT, or MAP column and visualize its structure
as if it were a table. An ARRAY is shown as a two-column table, with ITEM and POS columns. A STRUCT is shown as a
table with each field representing a column in the table. A MAP is shown as a two-column table, with KEY and VALUE
columns.
Added in: CDH 5.5.0 (Impala 2.3.0)
Restrictions:
• Columns with this data type can only be used in tables or partitions with the Parquet file format.
• Columns with this data type cannot be used as partition key columns in a partitioned table.
• The COMPUTE STATS statement does not produce any statistics for columns of this data type.
• The maximum length of the column definition for any complex type, including declarations for any nested types,
is 4000 characters.
• See Limitations and Restrictions for Complex Types on page 161 for a full list of limitations and associated guidelines
about complex type columns.
Examples:
Note: Many of the complex type examples refer to tables such as CUSTOMER and REGION adapted
from the tables used in the TPC-H benchmark. See Sample Schema and Data for Experimenting with
Impala Complex Types on page 178 for the table definitions.
The following example shows a table with various kinds of MAP columns, both at the top level and nested within other
complex types. Each row represents information about a specific country, with complex type fields of various levels
of nesting to represent different information associated with the country: factual measurements such as area and
population, notable people in different categories, geographic features such as cities, points of interest within each
city, and mountains with associated facts. Practice the CREATE TABLE and query notation for complex type columns
using empty tables, until you can visualize a complex data structure and construct corresponding SQL statements
reliably.
-- MAP that is an element within an ARRAY. The MAP is inside a STRUCT field to associate
-- the mountain name with all the facts about the mountain.
-- The "key" of the map (the first STRING field) represents the name of some fact whose
value
-- can be expressed as an integer, such as 'Height', 'Year First Climbed', and so on.
mountains ARRAY < STRUCT < name: STRING, facts: MAP <STRING, INT > > >
)
STORED AS PARQUET;
DESCRIBE map_demo;
+------------+------------------------------------------------+
| name | type |
+------------+------------------------------------------------+
| country_id | bigint |
| metrics | map<string,bigint> |
| notables | map<string,array<string>> |
| cities | array<struct< |
| | name:string, |
| | points_of_interest:map<string,array<string>> |
| | >> |
| mountains | array<struct< |
| | name:string, |
| | facts:map<string,int> |
| | >> |
+------------+------------------------------------------------+
DESCRIBE map_demo.metrics;
+-------+--------+
| name | type |
+-------+--------+
| key | string |
| value | bigint |
+-------+--------+
DESCRIBE map_demo.notables;
+-------+---------------+
| name | type |
+-------+---------------+
| key | string |
| value | array<string> |
+-------+---------------+
DESCRIBE map_demo.notables.value;
+------+--------+
| name | type |
+------+--------+
| item | string |
| pos | bigint |
+------+--------+
DESCRIBE map_demo.cities;
+------+------------------------------------------------+
| name | type |
+------+------------------------------------------------+
| item | struct< |
| | name:string, |
| | points_of_interest:map<string,array<string>> |
| | > |
| pos | bigint |
+------+------------------------------------------------+
DESCRIBE map_demo.cities.item.points_of_interest;
+-------+---------------+
| name | type |
+-------+---------------+
| key | string |
| value | array<string> |
+-------+---------------+
DESCRIBE map_demo.cities.item.points_of_interest.value;
+------+--------+
| name | type |
+------+--------+
| item | string |
| pos | bigint |
+------+--------+
DESCRIBE map_demo.mountains;
+------+-------------------------+
| name | type |
+------+-------------------------+
| item | struct< |
| | name:string, |
| | facts:map<string,int> |
| | > |
| pos | bigint |
+------+-------------------------+
DESCRIBE map_demo.mountains.item.facts;
+-------+--------+
| name | type |
+-------+--------+
| key | string |
| value | int |
+-------+--------+
The following example shows a table that uses a variety of data types for the MAP “key” field. Typically, you use BIGINT
or STRING to use numeric or character-based keys without worrying about exceeding any size or length constraints.
CREATE TABLE celebrities (name STRING, birth_year MAP < STRING, SMALLINT >) STORED AS
PARQUET;
-- A typical row might represent values with 2 different birth years, such as:
-- ("Joe Movie Star", { "real": 1972, "claimed": 1977 })
CREATE TABLE countries (name STRING, famous_leaders MAP < INT, STRING >) STORED AS
PARQUET;
-- A typical row might represent values with different leaders, with key values
corresponding to their numeric sequence, such as:
-- ("United States", { 1: "George Washington", 3: "Thomas Jefferson", 16: "Abraham
Lincoln" })
CREATE TABLE airlines (name STRING, special_meals MAP < STRING, MAP < STRING, STRING >
>) STORED AS PARQUET;
-- A typical row might represent values with multiple kinds of meals, each with several
components:
-- ("Elegant Airlines",
-- {
-- "vegetarian": { "breakfast": "pancakes", "snack": "cookies", "dinner": "rice
pilaf" },
-- "gluten free": { "breakfast": "oatmeal", "snack": "fruit", "dinner": "chicken"
}
-- } )
Related information:
Complex Types (CDH 5.5 or higher only) on page 156, ARRAY Complex Type (CDH 5.5 or higher only) on page 117, STRUCT
Complex Type (CDH 5.5 or higher only) on page 142
These examples show how you can use the type names REAL and DOUBLE interchangeably, and behind the scenes
Impala treats them always as DOUBLE.
column_name SMALLINT
Parquet considerations:
Physically, Parquet files represent TINYINT and SMALLINT values as 32-bit integers. Although Impala rejects attempts
to insert out-of-range values into such columns, if you create a new table with the CREATE TABLE ... LIKE PARQUET
syntax, any TINYINT or SMALLINT columns in the original table turn into INT columns in the new table.
Partitioning: Prefer to use this type for a partition key column. Impala can process the numeric type more efficiently
than a STRING representation of the value.
HBase considerations: This data type is fully compatible with HBase tables.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Internal details: Represented in memory as a 2-byte value.
column_name STRING
Length: Maximum of 32,767 bytes. Do not use any length constraint when declaring STRING columns, as you might
be familiar with from VARCHAR, CHAR, or similar column types from relational database systems. If you do need to
manipulate string values with precise or maximum lengths, in Impala 2.0 and higher you can declare columns as
VARCHAR(max_length) or CHAR(length), but for best performance use STRING where practical.
Character sets: For full support in all Impala subsystems, restrict string values to the ASCII character set. Although
some UTF-8 character data can be stored in Impala and retrieved through queries, UTF-8 strings containing non-ASCII
characters are not guaranteed to work properly in combination with many SQL aspects, including but not limited to:
• String manipulation functions.
• Comparison operators.
• The ORDER BY clause.
• Values in partition key columns.
For any national language aspects such as collation order or interpreting extended ASCII variants such as ISO-8859-1
or ISO-8859-2 encodings, Impala does not include such metadata with the table definition. If you need to sort, manipulate,
or display data depending on those national language characteristics of string data, use logic on the application side.
Conversions:
• Impala does not automatically convert STRING to any numeric type. Impala does automatically convert STRING
to TIMESTAMP if the value matches one of the accepted TIMESTAMP formats; see TIMESTAMP Data Type on page
148 for details.
• You can use CAST() to convert STRING values to TINYINT, SMALLINT, INT, BIGINT, FLOAT, DOUBLE, or
TIMESTAMP.
• You cannot directly cast a STRING value to BOOLEAN. You can use a CASE expression to evaluate string values
such as 'T', 'true', and so on and return Boolean true and false values as appropriate.
• You can cast a BOOLEAN value to STRING, returning '1' for true values and '0' for false values.
Partitioning:
Although it might be convenient to use STRING columns for partition keys, even when those columns contain numbers,
for performance and scalability it is much better to use numeric columns as partition keys whenever practical. Although
the underlying HDFS directory name might be the same in either case, the in-memory storage for the partition key
columns is more compact, and computations are faster, if partition key columns such as YEAR, MONTH, DAY and so on
are declared as INT, SMALLINT, and so on.
Zero-length strings: For purposes of clauses such as DISTINCT and GROUP BY, Impala considers zero-length strings
(""), NULL, and space to all be different values.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Avro considerations:
The Avro specification allows string values up to 2**64 bytes in length. Impala queries for Avro tables use 32-bit integers
to hold string lengths. In CDH 5.7 / Impala 2.5 and higher, Impala truncates CHAR and VARCHAR values in Avro tables
to (2**31)-1 bytes. If a query encounters a STRING value longer than (2**31)-1 bytes in an Avro table, the query fails.
In earlier releases, encountering such long values in an Avro table could cause a crash.
Column statistics considerations: Because the values of this type have variable size, none of the column statistics fields
are filled in until you run the COMPUTE STATS statement.
Examples:
The following examples demonstrate double-quoted and single-quoted string literals, and required escaping for
quotation marks within string literals:
The following examples demonstrate calls to string manipulation functions to concatenate strings, convert numbers
to strings, or pull out substrings:
SELECT CONCAT("Once upon a time, there were ", CAST(3 AS STRING), ' little pigs.');
SELECT SUBSTR("hello world",7,5);
The following examples show how to perform operations on STRING columns within a table:
Related information:
String Literals on page 185, CHAR Data Type (CDH 5.2 or higher only) on page 122, VARCHAR Data Type (CDH 5.2 or
higher only) on page 154, Impala String Functions on page 394, Impala Date and Time Functions on page 363
The names and number of fields within the STRUCT are fixed. Each field can be a different type. A field within a STRUCT
can also be another STRUCT, or an ARRAY or a MAP, allowing you to create nested data structures with a maximum
nesting depth of 100.
A STRUCT can be the top-level type for a column, or can itself be an item within an ARRAY or the value part of the
key-value pair in a MAP.
When a STRUCT is used as an ARRAY element or a MAP value, you use a join clause to bring the ARRAY or MAP elements
into the result set, and then refer to array_name.ITEM.field or map_name.VALUE.field. In the case of a STRUCT
directly inside an ARRAY or MAP, you can omit the .ITEM and .VALUE pseudocolumns and refer directly to
array_name.field or map_name.field.
Usage notes:
Because complex types are often used in combination, for example an ARRAY of STRUCT elements, if you are unfamiliar
with the Impala complex types, start with Complex Types (CDH 5.5 or higher only) on page 156 for background information
and usage examples.
A STRUCT is similar conceptually to a table row: it contains a fixed number of named fields, each with a predefined
type. To combine two related tables, while using complex types to minimize repetition, the typical way to represent
that data is as an ARRAY of STRUCT elements.
Because a STRUCT has a fixed number of named fields, it typically does not make sense to have a STRUCT as the type
of a table column. In such a case, you could just make each field of the STRUCT into a separate column of the table.
The STRUCT type is most useful as an item of an ARRAY or the value part of the key-value pair in a MAP. A nested type
column with a STRUCT at the lowest level lets you associate a variable number of row-like objects with each row of
the table.
The STRUCT type is straightforward to reference within a query. You do not need to include the STRUCT column in a
join clause or give it a table alias, as is required for the ARRAY and MAP types. You refer to the individual fields using
dot notation, such as struct_column_name.field_name, without any pseudocolumn such as ITEM or VALUE.
You can pass a qualified name to DESCRIBE to specify an ARRAY, STRUCT, or MAP column and visualize its structure
as if it were a table. An ARRAY is shown as a two-column table, with ITEM and POS columns. A STRUCT is shown as a
table with each field representing a column in the table. A MAP is shown as a two-column table, with KEY and VALUE
columns.
Internal details:
Within the Parquet data file, the values for each STRUCT field are stored adjacent to each other, so that they can be
encoded and compressed using all the Parquet techniques for storing sets of similar or repeated values. The adjacency
applies even when the STRUCT values are part of an ARRAY or MAP. During a query, Impala avoids unnecessary I/O by
reading only the portions of the Parquet data file containing the requested STRUCT fields.
Added in: CDH 5.5.0 (Impala 2.3.0)
Restrictions:
• Columns with this data type can only be used in tables or partitions with the Parquet file format.
• Columns with this data type cannot be used as partition key columns in a partitioned table.
• The COMPUTE STATS statement does not produce any statistics for columns of this data type.
• The maximum length of the column definition for any complex type, including declarations for any nested types,
is 4000 characters.
• See Limitations and Restrictions for Complex Types on page 161 for a full list of limitations and associated guidelines
about complex type columns.
Examples:
Note: Many of the complex type examples refer to tables such as CUSTOMER and REGION adapted
from the tables used in the TPC-H benchmark. See Sample Schema and Data for Experimenting with
Impala Complex Types on page 178 for the table definitions.
The following example shows a table with various kinds of STRUCT columns, both at the top level and nested within
other complex types. Practice the CREATE TABLE and query notation for complex type columns using empty tables,
until you can visualize a complex data structure and construct corresponding SQL statements reliably.
The following example shows how to examine the structure of a table containing one or more STRUCT columns by
using the DESCRIBE statement. You can visualize each STRUCT as its own table, with columns named the same as
each field of the STRUCT. If the STRUCT is nested inside another complex type, such as ARRAY, you can extend the
qualified name passed to DESCRIBE until the output shows just the STRUCT fields.
DESCRIBE struct_demo;
+-------------------+--------------------------+
| name | type |
+-------------------+--------------------------+
| id | bigint |
| name | string |
| employee_info | struct< |
| | employer:string, |
| | id:bigint, |
| | address:string |
| | > |
| places_lived | array<struct< |
| | street:string, |
| | city:string, |
| | country:string |
| | >> |
| memorable_moments | map<string,struct< |
| | year:int, |
| | place:string, |
| | details:string |
| | >> |
| current_address | struct< |
| | street_address:struct< |
| | street_number:int, |
| | street_name:string, |
| | street_type:string |
| | >, |
| | country:string, |
| | postal_code:string |
| | > |
+-------------------+--------------------------+
The top-level column EMPLOYEE_INFO is a STRUCT. Describing table_name.struct_name displays the fields of the
STRUCT as if they were columns of a table:
DESCRIBE struct_demo.employee_info;
+----------+--------+
| name | type |
+----------+--------+
| employer | string |
| id | bigint |
| address | string |
+----------+--------+
Because PLACES_LIVED is a STRUCT inside an ARRAY, the initial DESCRIBE shows the structure of the ARRAY:
DESCRIBE struct_demo.places_lived;
+------+------------------+
| name | type |
+------+------------------+
| item | struct< |
| | street:string, |
| | city:string, |
| | country:string |
| | > |
| pos | bigint |
+------+------------------+
Ask for the details of the ITEM field of the ARRAY to see just the layout of the STRUCT:
DESCRIBE struct_demo.places_lived.item;
+---------+--------+
| name | type |
+---------+--------+
| street | string |
| city | string |
| country | string |
+---------+--------+
Likewise, MEMORABLE_MOMENTS has a STRUCT inside a MAP, which requires an extra level of qualified name to see just
the STRUCT part:
DESCRIBE struct_demo.memorable_moments;
+-------+------------------+
| name | type |
+-------+------------------+
| key | string |
| value | struct< |
| | year:int, |
| | place:string, |
| | details:string |
| | > |
+-------+------------------+
For a MAP, ask to see the VALUE field to see the corresponding STRUCT fields in a table-like structure:
DESCRIBE struct_demo.memorable_moments.value;
+---------+--------+
| name | type |
+---------+--------+
| year | int |
| place | string |
| details | string |
+---------+--------+
For a STRUCT inside a STRUCT, we can see the fields of the outer STRUCT:
DESCRIBE struct_demo.current_address;
+----------------+-----------------------+
| name | type |
+----------------+-----------------------+
| street_address | struct< |
| | street_number:int, |
| | street_name:string, |
| | street_type:string |
| | > |
| country | string |
| postal_code | string |
+----------------+-----------------------+
Then we can use a further qualified name to see just the fields of the inner STRUCT:
DESCRIBE struct_demo.current_address.street_address;
+---------------+--------+
| name | type |
+---------------+--------+
| street_number | int |
| street_name | string |
| street_type | string |
+---------------+--------+
The following example shows how to examine the structure of a table containing one or more STRUCT columns by
using the DESCRIBE statement. You can visualize each STRUCT as its own table, with columns named the same as
each field of the STRUCT. If the STRUCT is nested inside another complex type, such as ARRAY, you can extend the
qualified name passed to DESCRIBE until the output shows just the STRUCT fields.
DESCRIBE struct_demo;
+-------------------+--------------------------+---------+
| name | type | comment |
+-------------------+--------------------------+---------+
| id | bigint | |
| name | string | |
| employee_info | struct< | |
| | employer:string, | |
| | id:bigint, | |
| | address:string | |
| | > | |
| places_lived | array<struct< | |
| | street:string, | |
| | city:string, | |
| | country:string | |
| | >> | |
| memorable_moments | map<string,struct< | |
| | year:int, | |
| | place:string, | |
| | details:string | |
| | >> | |
| current_address | struct< | |
| | street_address:struct< | |
| | street_number:int, | |
| | street_name:string, | |
| | street_type:string | |
| | >, | |
| | country:string, | |
| | postal_code:string | |
| | > | |
+-------------------+--------------------------+---------+
places_lived.country
FROM struct_demo, struct_demo.places_lived;
For example, this table uses a struct that encodes several data values for each phone number associated with a person.
Each person can have a variable-length array of associated phone numbers, and queries can refer to the category field
to locate specific home, work, mobile, and so on kinds of phone numbers.
Because structs are naturally suited to composite values where the fields have different data types, you might use
them to decompose things such as addresses:
In a big data context, splitting out data fields such as the number part of the address and the street name could let
you do analysis on each field independently. For example, which streets have the largest number range of addresses,
what are the statistical properties of the street names, which areas have a higher proportion of “Roads”, “Courts” or
“Boulevards”, and so on.
Related information:
Complex Types (CDH 5.5 or higher only) on page 156, ARRAY Complex Type (CDH 5.5 or higher only) on page 117, MAP
Complex Type (CDH 5.5 or higher only) on page 136
column_name TIMESTAMP
Range: Allowed date values range from 1400-01-01 to 9999-12-31; this range is different from the Hive TIMESTAMP
type. Internally, the resolution of the time portion of a TIMESTAMP value is in nanoseconds.
INTERVAL expressions:
You can perform date arithmetic by adding or subtracting a specified number of time units, using the INTERVAL
keyword and the + and - operators or date_add() and date_sub() functions. You can specify units as YEAR[S],
MONTH[S], WEEK[S], DAY[S], HOUR[S], MINUTE[S], SECOND[S], MILLISECOND[S], MICROSECOND[S], and
NANOSECOND[S]. You can only specify one time unit in each interval expression, for example INTERVAL 3 DAYS or
INTERVAL 25 HOURS, but you can produce any granularity by adding together successive INTERVAL values, such as
timestamp_value + INTERVAL 3 WEEKS - INTERVAL 1 DAY + INTERVAL 10 MICROSECONDS.
For example:
Time zones:
By default, Impala does not store timestamps using the local timezone, to avoid undesired results from unexpected
time zone issues. Timestamps are stored and interpreted relative to UTC, both when written to or read from data files,
or when converted to or from Unix time values through functions such as from_unixtime() or unix_timestamp().
To convert such a TIMESTAMP value to one that represents the date and time in a specific time zone, convert the
original value with the from_utc_timestamp() function.
Because Impala does not assume that TIMESTAMP values are in any particular time zone, you must be conscious of
the time zone aspects of data that you query, insert, or convert.
For consistency with Unix system calls, the TIMESTAMP returned by the now() function represents the local time in
the system time zone, rather than in UTC. To store values relative to the current time in a portable way, convert any
now() return values using the to_utc_timestamp() function first. For example, the following example shows that
the current time in California (where this Impala cluster is located) is shortly after 2 PM. If that value was written to a
data file, and shipped off to a distant server to be analyzed alongside other data from far-flung locations, the dates
and times would not match up precisely because of time zone differences. Therefore, the to_utc_timestamp()
function converts it using a common reference point, the UTC time zone (descended from the old Greenwich Mean
Time standard). The 'PDT' argument indicates that the original value is from the Pacific time zone with Daylight Saving
Time in effect. When servers in all geographic locations run the same transformation on any local date and time values
(with the appropriate time zone argument), the stored data uses a consistent representation. Impala queries can use
functions such as EXTRACT(), MIN(), AVG(), and so on to do time-series analysis on those timestamps.
The converse function, from_utc_timestamp(), lets you take stored TIMESTAMP data or calculated results and
convert back to local date and time for processing on the application side. The following example shows how you might
represent some future date (such as the ending date and time of an auction) in UTC, and then convert back to local
time when convenient for reporting or other processing. The final query in the example tests whether this arbitrary
UTC date and time has passed yet, by converting it back to the local time zone and comparing it against the current
date and time.
If you have data files written by Hive, those TIMESTAMP values represent the local timezone of the host where the
data was written, potentially leading to inconsistent results when processed by Impala. To avoid compatibility problems
or having to code workarounds, you can specify one or both of these impalad startup flags:
-use_local_tz_for_unix_timestamp_conversions=true
-convert_legacy_hive_parquet_utc_timestamps=true. Although
-convert_legacy_hive_parquet_utc_timestamps is turned off by default to avoid performance overhead,
Cloudera recommends turning it on when processing TIMESTAMP columns in Parquet files written by Hive, to avoid
unexpected behavior.
The -use_local_tz_for_unix_timestamp_conversions setting affects conversions from TIMESTAMP to BIGINT,
or from BIGINT to TIMESTAMP. By default, Impala treats all TIMESTAMP values as UTC, to simplify analysis of time-series
data from different geographic regions. When you enable the -use_local_tz_for_unix_timestamp_conversions
setting, these operations treat the input values as if they are in the local tie zone of the host doing the processing. See
Impala Date and Time Functions on page 363 for the list of functions affected by the
-use_local_tz_for_unix_timestamp_conversions setting.
The following sequence of examples shows how the interpretation of TIMESTAMP values in Parquet tables is affected
by the setting of the -convert_legacy_hive_parquet_utc_timestamps setting.
Regardless of the -convert_legacy_hive_parquet_utc_timestamps setting, TIMESTAMP columns in text tables
can be written and read interchangeably by Impala and Hive:
| 2015-04-08 15:43:02.892403000 |
+-------------------------------+
[localhost:21000] > select to_utc_timestamp(x, 'PDT') from t1;
+-------------------------------+
| to_utc_timestamp(x, 'pdt') |
+-------------------------------+
| 2015-04-07 22:43:02.892403000 |
| 2015-04-08 22:43:02.892403000 |
+-------------------------------+
When the table uses Parquet format, Impala expects any time zone adjustment to be applied prior to writing, while
TIMESTAMP values written by Hive are adjusted to be in the UTC time zone. When Hive queries Parquet data files that
it wrote, it adjusts the TIMESTAMP values back to the local time zone, while Impala does no conversion. Hive does no
time zone conversion when it queries Impala-written Parquet files.
The discrepancy arises when Impala queries the Hive-created Parquet table. The underlying values in the TIMESTAMP
column are different from the ones written by Impala, even though they were copied from one table to another by an
INSERT ... SELECT statement in Hive. Hive did an implicit conversion from the local time zone to UTC as it wrote
the values to Parquet.
Impala query for TIMESTAMP values from Impala-written and Hive-written data:
| 2015-04-07 15:43:02.892403000 |
| 2015-04-08 15:43:02.892403000 |
+-------------------------------+
Fetched 2 row(s) in 0.29s
[localhost:21000] > select * from h1;
+-------------------------------+
| x |
+-------------------------------+
| 2015-04-07 22:43:02.892403000 |
| 2015-04-08 22:43:02.892403000 |
+-------------------------------+
Fetched 2 row(s) in 0.41s
Conversions:
Impala automatically converts STRING literals of the correct format into TIMESTAMP values. Timestamp values are
accepted in the format "yyyy-MM-dd HH:mm:ss.SSSSSS", and can consist of just the date, or just the time, with
or without the fractional second portion. For example, you can specify TIMESTAMP values such as '1966-07-30',
'08:30:00', or '1985-09-25 17:45:30.005'. Casting an integer or floating-point value N to TIMESTAMP produces
a value that is N seconds past the start of the epoch date (January 1, 1970). By default, the result value represents a
date and time in the UTC time zone. If the setting --use_local_tz_for_unix_timestamp_conversions=true
is in effect, the resulting TIMESTAMP represents a date and time in the local time zone.
In Impala 1.3 and higher, the FROM_UNIXTIME() and UNIX_TIMESTAMP() functions allow a wider range of format
strings, with more flexibility in element order, repetition of letter placeholders, and separator characters. In CDH 5.5
/ Impala 2.3 and higher, the UNIX_TIMESTAMP() function also allows a numeric timezone offset to be specified as
part of the input string. See Impala Date and Time Functions on page 363 for details.
In Impala 2.2.0 and higher, built-in functions that accept or return integers representing TIMESTAMP values use the
BIGINT type for parameters and return values, rather than INT. This change lets the date and time functions avoid
an overflow error that would otherwise occur on January 19th, 2038 (known as the “Year 2038 problem” or “Y2K38
problem”). This change affects the from_unixtime() and unix_timestamp() functions. You might need to change
application code that interacts with these functions, change the types of columns that store the return values, or add
CAST() calls to SQL statements that call these functions.
Partitioning:
Although you cannot use a TIMESTAMP column as a partition key, you can extract the individual years, months, days,
hours, and so on and partition based on those columns. Because the partition key column values are represented in
HDFS directory names, rather than as fields in the data files themselves, you can also keep the original TIMESTAMP
values if desired, without duplicating data or wasting storage space. See Partition Key Columns on page 525 for more
details on partitioning with date and time values.
Examples:
NULL considerations: Casting any unrecognized STRING value to this type produces a NULL value.
Partitioning: Because this type potentially has so many distinct values, it is often not a sensible choice for a partition
key column. For example, events 1 millisecond apart would be stored in different partitions. Consider using the TRUNC()
function to condense the number of distinct values, and partition on a new column with the truncated values.
HBase considerations: This data type is fully compatible with HBase tables.
Parquet considerations: This type is fully compatible with Parquet tables.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Internal details: Represented in memory as a 16-byte value.
Added in: Available in all versions of Impala.
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
Sqoop considerations:
If you use Sqoop to convert RDBMS data to Parquet, be careful with interpreting any resulting values from DATE,
DATETIME, or TIMESTAMP columns. The underlying values are represented as the Parquet INT64 type, which is
represented as BIGINT in the Impala table. The Parquet values represent the time in milliseconds, while Impala
interprets BIGINT as the time in seconds. Therefore, if you have a BIGINT column in a Parquet table that was imported
this way from Sqoop, divide the values by 1000 when interpreting as the TIMESTAMP type.
Restrictions:
If you cast a STRING with an unrecognized format to a TIMESTAMP, the result is NULL rather than an error. Make sure
to test your data pipeline to be sure any textual date and time values are in a format that Impala TIMESTAMP can
recognize.
Currently, Avro tables cannot contain TIMESTAMP columns. If you need to store date and time values in Avro tables,
as a workaround you can use a STRING representation of the values, convert the values to BIGINT with the
UNIX_TIMESTAMP() function, or create separate numeric columns for individual date and time fields using the
EXTRACT() function.
Related information:
• Timestamp Literals on page 187.
• To convert to or from different date formats, or perform date arithmetic, use the date and time functions described
in Impala Date and Time Functions on page 363. In particular, the from_unixtime() function requires a
case-sensitive format string such as "yyyy-MM-dd HH:mm:ss.SSSS", matching one of the allowed variations
of a TIMESTAMP value (date plus time, only date, only time, optional fractional seconds).
• See SQL Differences Between Impala and Hive on page 461 for details about differences in TIMESTAMP handling
between Impala and Hive.
column_name TINYINT
For a convenient and automated way to check the bounds of the TINYINT type, call the functions MIN_TINYINT()
and MAX_TINYINT().
If an integer value is too large to be represented as a TINYINT, use a SMALLINT instead.
NULL considerations: Casting any non-numeric value to this type produces a NULL value.
Examples:
Parquet considerations:
Physically, Parquet files represent TINYINT and SMALLINT values as 32-bit integers. Although Impala rejects attempts
to insert out-of-range values into such columns, if you create a new table with the CREATE TABLE ... LIKE PARQUET
syntax, any TINYINT or SMALLINT columns in the original table turn into INT columns in the new table.
HBase considerations: This data type is fully compatible with HBase tables.
Text table considerations: Values of this type are potentially larger in text tables than in tables using Parquet or other
binary formats.
Internal details: Represented in memory as a 1-byte value.
Added in: Available in all versions of Impala.
Column statistics considerations: Because this type has a fixed size, the maximum and average size fields are always
filled in for column statistics, even before you run the COMPUTE STATS statement.
Related information:
Numeric Literals on page 184, TINYINT Data Type on page 153, SMALLINT Data Type on page 140, INT Data Type on page
135, BIGINT Data Type on page 120, DECIMAL Data Type (CDH 5.1 or higher only) on page 125, Impala Mathematical
Functions on page 339
column_name VARCHAR(max_length)
Text data files can contain values that are longer than allowed by the VARCHAR(n) length limit. Any extra trailing
characters are ignored when Impala processes those values during a query.
Avro considerations:
The Avro specification allows string values up to 2**64 bytes in length. Impala queries for Avro tables use 32-bit integers
to hold string lengths. In CDH 5.7 / Impala 2.5 and higher, Impala truncates CHAR and VARCHAR values in Avro tables
to (2**31)-1 bytes. If a query encounters a STRING value longer than (2**31)-1 bytes in an Avro table, the query fails.
In earlier releases, encountering such long values in an Avro table could cause a crash.
Schema evolution considerations:
You can use ALTER TABLE ... CHANGE to switch column data types to and from VARCHAR. You can convert from
STRING to VARCHAR(n), or from VARCHAR(n) to STRING, or from CHAR(n) to VARCHAR(n), or from VARCHAR(n)
to CHAR(n). When switching back and forth between VARCHAR and CHAR, you can also change the length value. This
schema evolution works the same for tables using any file format. If a table contains values longer than the maximum
length defined for a VARCHAR column, Impala does not return an error. Any extra trailing characters are ignored when
Impala processes those values during a query.
Compatibility:
This type is available on CDH 5.2 or higher.
Internal details: Represented in memory as a byte array with the minimum size needed to represent each value.
Added in: CDH 5.2.0 (Impala 2.0.0)
Column statistics considerations: Because the values of this type have variable size, none of the column statistics fields
are filled in until you run the COMPUTE STATS statement.
Restrictions:
All data in CHAR and VARCHAR columns must be in a character encoding that is compatible with UTF-8. If you have
binary data from another database system (that is, a BLOB type), use a STRING column to hold it.
Examples:
The following examples show how long and short VARCHAR values are treated. Values longer than the maximum
specified length are truncated by CAST(), or when queried from existing data files. Values shorter than the maximum
specified length are represented as the actual length of the value, with no extra padding as seen with CHAR values.
+------+
[localhost:21000] > select * from varchar_20;
+-------+
| s |
+-------+
| a |
| b |
| hello |
| world |
+-------+
select concat('[',s,']') as s from varchar_20;
+---------+
| s |
+---------+
| [a] |
| [b] |
| [hello] |
| [world] |
+---------+
The following example shows how identical VARCHAR values compare as equal, even if the columns are defined with
different maximum lengths. Both tables contain 'a' and 'b' values. The longer 'hello' and 'world' values from
the VARCHAR_20 table were truncated when inserted into the VARCHAR_1 table.
The following examples show how VARCHAR values are freely interchangeable with STRING values in contexts such
as comparison operators and built-in functions:
UDF considerations: This type cannot be used for the argument or return type of a user-defined function (UDF) or
user-defined aggregate function (UDA).
Related information:
STRING Data Type on page 141, CHAR Data Type (CDH 5.2 or higher only) on page 122, String Literals on page 185, Impala
String Functions on page 394
Once you understand the basics of complex types, refer to the individual type topics when you need to refresh your
memory about syntax and examples:
• ARRAY Complex Type (CDH 5.5 or higher only) on page 117
The Impala complex type support produces result sets with all scalar values, and the scalar components of complex
types can be used with all SQL clauses, such as GROUP BY, ORDER BY, all kinds of joins, subqueries, and inline views.
The ability to process complex type data entirely in SQL reduces the need to write application-specific code in Java or
other programming languages to deconstruct the underlying data structures.
Note:
Although Impala can query complex types that are present in Parquet files, Impala currently cannot
create new Parquet files containing complex types. Therefore, the discussion and examples presume
that you are working with existing Parquet data produced through Hive, Spark, or some other source.
See Constructing Parquet Files with Complex Columns Using Hive on page 179 for examples of
constructing Parquet data files with complex type columns.
For learning purposes, you can create empty tables with complex type columns and practice query
syntax, even if you do not have sample data with the required structure.
Complex types represent a middle ground that addresses these performance and volume concerns. By physically
locating related data within the same data files, complex types increase locality and reduce the expense of join
queries. By associating an arbitrary amount of data with a single row, complex types avoid the need to repeat
lengthy values such as strings. Because Impala knows which complex type values are associated with each row,
you can save storage by avoiding artificial foreign key values that are only used for joins. The flexibility of the
STRUCT, ARRAY, and MAP types lets you model familiar constructs such as fact and dimension tables from a data
warehouse, and wide tables representing sparse matrixes.
• For an ARRAY containing scalar values, all those values (represented by the ITEM pseudocolumn) are stored
adjacent to each other.
• For a MAP, the values of the KEY pseudocolumn are stored adjacent to each other. If the VALUE pseudocolumn is
a scalar type, its values are also stored adjacent to each other.
• If an ARRAY element, STRUCT field, or MAP VALUE part is another complex type, the column-oriented storage
applies to the next level down (or the next level after that, and so on for deeply nested types) where the final
elements, fields, or values are of scalar types.
The numbers represented by the POS pseudocolumn of an ARRAY are not physically stored in the data files. They are
synthesized at query time based on the order of the ARRAY elements associated with each row.
File Format Support for Impala Complex Types
Currently, Impala queries support complex type data only in the Parquet file format. See Using the Parquet File Format
with Impala Tables on page 535 for details about the performance benefits and physical layout of this file format.
Each table, or each partition within a table, can have a separate file format, and you can change file format at the table
or partition level through an ALTER TABLE statement. Because this flexibility makes it difficult to guarantee ahead of
time that all the data files for a table or partition are in a compatible format, Impala does not throw any errors when
you change the file format for a table or partition using ALTER TABLE. Any errors come at runtime when Impala
actually processes a table or partition that contains nested types and is not in one of the supported formats. If a query
on a partitioned table only processes some partitions, and all those partitions are in one of the supported formats, the
query succeeds.
Because Impala does not parse the data structures containing nested types for unsupported formats such as text, Avro,
SequenceFile, or RCFile, you cannot use data files in these formats with Impala, even if the query does not refer to the
nested type columns. Also, if a table using an unsupported format originally contained nested type columns, and then
those columns were dropped from the table using ALTER TABLE ... DROP COLUMN, any existing data files in the
table still contain the nested type data and Impala queries on that table will generate errors.
You can perform DDL operations (even CREATE TABLE) for tables involving complex types in file formats other than
Parquet. The DDL support lets you set up intermediate tables in your ETL pipeline, to be populated by Hive, before the
final stage where the data resides in a Parquet table and is queryable by Impala. Also, you can have a partitioned table
with complex type columns that uses a non-Parquet format, and use ALTER TABLE to change the file format to Parquet
for individual partitions. When you put Parquet data files into those partitions, Impala can execute queries against
that data as long as the query does not involve any of the non-Parquet partitions.
If you use the parquet-tools command to examine the structure of a Parquet data file that includes complex types,
you see that both ARRAY and MAP are represented as a Bag in Parquet terminology, with all fields marked Optional
because Impala allows any column to be nullable.
Impala supports either 2-level and 3-level encoding within each Parquet data file. When constructing Parquet data
files outside Impala, use either encoding style but do not mix 2-level and 3-level encoding within the same data file.
Choosing Between Complex Types and Normalized Tables
Choosing between multiple normalized fact and dimension tables, or a single table containing complex types, is an
important design decision.
• If you are coming from a traditional database or data warehousing background, you might be familiar with how
to split up data between tables. Your business intelligence tools might already be optimized for dealing with this
kind of multi-table scenario through join queries.
• If you are pulling data from Impala into an application written in a programming language that has data structures
analogous to the complex types, such as Python or Java, complex types in Impala could simplify data interchange
and improve understandability and reliability of your program logic.
• You might already be faced with existing infrastructure or receive high volumes of data that assume one layout
or the other. For example, complex types are popular with web-oriented applications, for example to keep
information about an online user all in one place for convenient lookup and analysis, or to deal with sparse or
constantly evolving data fields.
• If some parts of the data change over time while related data remains constant, using multiple normalized tables
lets you replace certain parts of the data without reloading the entire data set. Conversely, if you receive related
data all bundled together, such as in JSON files, using complex types can save the overhead of splitting the related
items across multiple tables.
• From a performance perspective:
– In Parquet tables, Impala can skip columns that are not referenced in a query, avoiding the I/O penalty of
reading the embedded data. When complex types are nested within a column, the data is physically divided
at a very granular level; for example, a query referring to data nested multiple levels deep in a complex type
column does not have to read all the data from that column, only the data for the relevant parts of the column
type hierarchy.
– Complex types avoid the possibility of expensive join queries when data from fact and dimension tables is
processed in parallel across multiple hosts. All the information for a row containing complex types is typically
to be in the same data block, and therefore does not need to be transmitted across the network when joining
fields that are all part of the same row.
– The tradeoff with complex types is that fewer rows fit in each data block. Whether it is better to have more
data blocks with fewer rows, or fewer data blocks with many rows, depends on the distribution of your data
and the characteristics of your query workload. If the complex columns are rarely referenced, using them
might lower efficiency. If you are seeing low parallelism due to a small volume of data (relatively few data
blocks) in each table partition, increasing the row size by including complex columns might produce more
data blocks and thus spread the work more evenly across the cluster. See Scalability Considerations for Impala
on page 516 for more on this advanced topic.
primitive_type
| array_type
| map_type
| struct_type
Note:
Currently, Impala queries allow complex types only in tables that use the Parquet format. If an Impala
query encounters complex types in a table or partition using another file format, the query returns a
runtime error.
The Impala DDL support for complex types works for all file formats, so that you can create tables
using text or other non-Parquet formats for Hive to use as staging tables in an ETL cycle that ends
with the data in a Parquet table. You can also use ALTER TABLE ... SET FILEFORMAT PARQUET
to change the file format of an existing table containing complex types to Parquet, after which Impala
can query it. Make sure to load Parquet files into the table after changing the file format, because the
ALTER TABLE ... SET FILEFORMAT statement does not convert existing data to the new file
format.
Partitioned tables can contain complex type columns. All the partition key columns must be scalar types.
Because use cases for Impala complex types require that you already have Parquet data files produced outside of
Impala, you can use the Impala CREATE TABLE LIKE PARQUET syntax to produce a table with columns that match
the structure of an existing Parquet file, including complex type columns for nested data structures. Remember to
include the STORED AS PARQUET clause in this case, because even with CREATE TABLE LIKE PARQUET, the default
file format of the resulting table is still text.
Because the complex columns are omitted from the result set of an Impala SELECT * or SELECT col_name query,
and because Impala currently does not support writing Parquet files with complex type columns, you cannot use the
CREATE TABLE AS SELECT syntax to create a table with nested type columns.
Note:
Once you have a table set up with complex type columns, use the DESCRIBE and SHOW CREATE
TABLE statements to see the correct notation with < and > delimiters and comma and colon separators
within the complex type definitions. If you do not have existing data with the same layout as the table,
you can query the empty table to practice with the notation for the SELECT statement. In the SELECT
list, you use dot notation and pseudocolumns such as ITEM, KEY, and VALUE for referring to items
within the complex type columns. In the FROM clause, you use join notation to construct table aliases
for any referenced ARRAY and MAP columns.
For example, when defining a table that holds contact information, you might represent phone numbers differently
depending on the expected layout and relationships of the data, and how well you can predict those properties in
advance.
Here are different ways that you might represent phone numbers in a traditional relational schema, with equivalent
representations using complex types.
The traditional, simplest way to represent phone numbers in a relational table is to store all contact info in a single
table, with all columns having scalar types, and each potential phone number represented as a separate column. In
this example, each person can only have these 3 types of phone numbers. If the person does not have a particular kind
of phone number, the corresponding column is NULL for that row.
, home_phone STRING
, work_phone STRING
, mobile_phone STRING
) STORED AS PARQUET;
Using a complex type column to represent the phone numbers adds some extra flexibility. Now there could be an
unlimited number of phone numbers. Because the array elements have an order but not symbolic names, you could
decide in advance that phone_number[0] is the home number, [1] is the work number, [2] is the mobile number, and
so on. (In subsequent examples, you will see how to create a more flexible naming scheme using other complex type
variations, such as a MAP or an ARRAY where each element is a STRUCT.)
Another way to represent an arbitrary set of phone numbers is with a MAP column. With a MAP, each element is
associated with a key value that you specify, which could be a numeric, string, or other scalar type. This example uses
a STRING key to give each phone number a name, such as 'home' or 'mobile'. A query could filter the data based
on the key values, or display the key values in reports.
If you are an experienced database designer, you already know how to work around the limitations of the single-table
schema from Figure 3: Traditional Relational Representation of Phone Numbers: Single Table on page 162. By normalizing
the schema, with the phone numbers in their own table, you can associate an arbitrary set of phone numbers with
each person, and associate additional details with each phone number, such as whether it is a home, work, or mobile
phone.
The flexibility of this approach comes with some drawbacks. Reconstructing all the data for a particular person requires
a join query, which might require performance tuning on Hadoop because the data from each table might be transmitted
from a different host. Data management tasks such as backups and refreshing the data require dealing with multiple
tables instead of a single table.
This example illustrates a traditional database schema to store contact info normalized across 2 tables. The fact table
establishes the identity and basic information about person. A dimension table stores information only about phone
numbers, using an ID value to associate each phone number with a person ID from the fact table. Each person can
have 0, 1, or many phones; the categories are not restricted to a few predefined ones; and the phone table can contain
as many columns as desired, to represent all sorts of details about each phone number.
CREATE TABLE fact_contacts (id BIGINT, name STRING, address STRING) STORED AS PARQUET;
CREATE TABLE dim_phones
(
contact_id BIGINT
, category STRING
, international_code STRING
, area_code STRING
, exchange STRING
, extension STRING
, mobile BOOLEAN
, carrier STRING
, current BOOLEAN
, service_start_date TIMESTAMP
, service_end_date TIMESTAMP
)
STORED AS PARQUET;
To represent a schema equivalent to the one from Figure 6: Traditional Relational Representation of Phone Numbers:
Normalized Tables on page 163 using complex types, this example uses an ARRAY where each array element is a STRUCT.
As with the earlier complex type examples, each person can have an arbitrary set of associated phone numbers. Making
each array element into a STRUCT lets us associate multiple data items with each phone number, and give a separate
name and type to each data item. The STRUCT fields of the ARRAY elements reproduce the columns of the dimension
table from the previous example.
You can do all the same kinds of queries with the complex type schema as with the normalized schema from the
previous example. The advantages of the complex type design are in the areas of convenience and performance. Now
your backup and ETL processes only deal with a single table. When a query uses a join to cross-reference the information
about a person with their associated phone numbers, all the relevant data for each row resides in the same HDFS data
block, meaning each row can be processed on a single host without requiring network transmission.
-- What goes inside the < > for an ARRAY is a single type, either a scalar or another
-- complex type (ARRAY, STRUCT, or MAP).
-- What goes inside the < > for a MAP is two comma-separated types specifying the types
of the key-value pair:
-- a scalar type representing the key, and a scalar or complex type representing the
value.
CREATE TABLE map_t
(
id BIGINT,
m1 MAP <STRING, STRING>,
m2 MAP <STRING, BIGINT>,
m3 MAP <BIGINT, STRING>,
m4 MAP <BIGINT, BIGINT>,
m5 MAP <STRING, ARRAY <STRING>>
)
STORED AS PARQUET;
-- What goes inside the < > for a STRUCT is a comma-separated list of fields, each field
defined as
-- name:type. The type can be a scalar or a complex type. The field names for each STRUCT
do not clash
-- with the names of table columns or fields in other STRUCTs. A STRUCT is most often
used inside
-- an ARRAY or a MAP rather than as a top-level column.
CREATE TABLE struct_t
(
id BIGINT,
s1 STRUCT <f1: STRING, f2: BIGINT>,
s2 ARRAY <STRUCT <f1: INT, f2: TIMESTAMP>>,
s3 MAP <BIGINT, STRUCT <name: STRING, birthday: TIMESTAMP>>
)
STORED AS PARQUET;
Note: Many of the complex type examples refer to tables such as CUSTOMER and REGION adapted
from the tables used in the TPC-H benchmark. See Sample Schema and Data for Experimenting with
Impala Complex Types on page 178 for the table definitions.
...
| c_orders | array<struct< |
| | o_orderkey:bigint, |
| | o_orderstatus:string, |
| | o_totalprice:decimal(12,2), |
...
| | >> |
+--------------+------------------------------------+
-- When we SELECT * from that table, only the scalar columns come back in the result
set.
CREATE TABLE select_star_customer STORED AS PARQUET AS SELECT * FROM customer;
+------------------------+
| summary |
+------------------------+
| Inserted 150000 row(s) |
+------------------------+
-- The c_orders column, being of complex type, was not included in the SELECT * result
set.
DESC select_star_customer;
+--------------+---------------+
| name | type |
+--------------+---------------+
| c_custkey | bigint |
| c_name | string |
| c_address | string |
| c_nationkey | smallint |
| c_phone | string |
| c_acctbal | decimal(12,2) |
| c_mktsegment | string |
| c_comment | string |
+--------------+---------------+
References to fields within STRUCT columns use dot notation. If the field name is unambiguous, you can omit qualifiers
such as table name, column name, or even the ITEM or VALUE pseudocolumn names for STRUCT elements inside an
ARRAY or a MAP.
References to fields within MAP columns use the KEY and VALUE pseudocolumns. In this example, once the query
establishes the alias MAP_FIELD for a MAP column with a STRING key and an INT value, the query can refer to
MAP_FIELD.KEY and MAP_FIELD.VALUE, which have zero, one, or many instances for each row from the containing
table.
DESCRIBE table_0;
+---------+-----------------------+
| name | type |
+---------+-----------------------+
| field_0 | string |
| field_1 | map<string,int> |
...
When complex types are nested inside each other, you use a combination of joins, pseudocolumn names, and dot
notation to refer to specific fields at the appropriate level. This is the most frequent form of query syntax for complex
columns, because the typical use case involves two levels of complex types, such as an ARRAY of STRUCT elements.
You can express relationships between ARRAY and MAP columns at different levels as joins. You include comparison
operators between fields at the top level and within the nested type columns so that Impala can do the appropriate
join operation.
Note: Many of the complex type examples refer to tables such as CUSTOMER and REGION adapted
from the tables used in the TPC-H benchmark. See Sample Schema and Data for Experimenting with
Impala Complex Types on page 178 for the table definitions.
For example, the following queries work equivalently. They each return customer and order data for customers that
have at least one order.
The following query using an outer join returns customers that have orders, plus customers with no orders (no entries
in the C_ORDERS array):
LIMIT 5;
+-----------+------------+
| c_custkey | o_orderkey |
+-----------+------------+
| 60210 | NULL |
| 147873 | NULL |
| 72578 | 558821 |
| 72578 | 2079810 |
| 72578 | 5768068 |
+-----------+------------+
The following query returns only customers that have no orders. (With LEFT ANTI JOIN or LEFT SEMI JOIN, the
query can only refer to columns from the left-hand table, because by definition there is no matching information in
the right-hand table.)
You can also perform correlated subqueries to examine the properties of complex type columns for each row in the
result set.
Count the number of orders per customer. Note the correlated reference to the table alias C. The COUNT(*) operation
applies to all the elements of the C_ORDERS array for the corresponding row, avoiding the need for a GROUP BY clause.
select c_name, howmany FROM customer c, (SELECT COUNT(*) howmany FROM c.c_orders) v
limit 5;
+--------------------+---------+
| c_name | howmany |
+--------------------+---------+
| Customer#000030065 | 15 |
| Customer#000065455 | 18 |
| Customer#000113644 | 21 |
| Customer#000111078 | 0 |
| Customer#000024621 | 0 |
+--------------------+---------+
Count the number of orders per customer, ignoring any customers that have not placed any orders:
Count the number of line items in each order. The reference to C.C_ORDERS in the FROM clause is needed because
the O_ORDERKEY field is a member of the elements in the C_ORDERS array. The subquery labelled SUBQ1 is correlated:
it is re-evaluated for the C_ORDERS.O_LINEITEMS array from each row of the CUSTOMERS table.
Get the number of orders, the average order price, and the maximum items in any order per customer. For this example,
the subqueries labelled SUBQ1 and SUBQ2 are correlated: they are re-evaluated for each row from the original CUSTOMER
table, and only apply to the complex columns associated with that row.
For example, these queries show how to access information about the ARRAY elements within the CUSTOMER table
from the “nested TPC-H” schema, starting with the initial ARRAY elements and progressing to examine the STRUCT
fields of the ARRAY, and then the elements nested within another ARRAY of STRUCT:
+-----------+------------+----------+
| 63367 | 4985959 | 7 |
| 53989 | 1972230 | 2 |
| 143513 | 5750498 | 5 |
| 17849 | 4857989 | 1 |
| 89881 | 1046437 | 1 |
+-----------+------------+----------+
Because an array is inherently an ordered data structure, Impala recognizes the special name array_column.POS to
represent the numeric position of each element within the array. The POS pseudocolumn lets you filter or reorder the
result set based on the sequence of array elements.
The following example uses a table from a flattened version of the TPC-H schema. The REGION table only has a few
rows, such as one row for Europe and one for Asia. The row for each region represents all the countries in that region
as an ARRAY of STRUCT elements:
To find the countries within a specific region, you use a join query. To find out the order of elements in the array, you
also refer to the POS pseudocolumn in the select list:
Once you know the positions of the elements, you can use that information in subsequent queries, for example to
change the ordering of results from the complex type column or to filter certain elements from the array:
The KEY must always be a scalar type, such as STRING, BIGINT, or TIMESTAMP. It can be NULL. Values of the KEY field
are not necessarily unique within the same MAP. You apply any required DISTINCT, GROUP BY, and other clauses in
the query, and loop through the result set to process all the values matching any specified keys.
The VALUE can be either a scalar type or another complex type. If the VALUE is a STRUCT, you can construct a qualified
name map_column.VALUE.struct_field to refer to the individual fields inside the value part. If the VALUE is an
ARRAY or another MAP, you must include another join condition that establishes a table alias for map_column.VALUE,
and then construct another qualified name using that alias, for example table_alias.ITEM or table_alias.KEY
and table_alias.VALUE
The following example shows different ways to access a MAP column using the KEY and VALUE pseudocolumns. The
DETAILS column has a STRING first part with short, standardized values such as 'Recurring', 'Lucid', or
'Anxiety'. This is the “key” that is used to look up particular kinds of elements from the MAP. The second part, also
a STRING, is a longer free-form explanation. Impala gives you the standard pseudocolumn names KEY and VALUE for
the two parts, and you apply your own conventions and interpretations to the underlying values.
Note: If you find that the single-item nature of the VALUE makes it difficult to model your data
accurately, the solution is typically to add some nesting to the complex type. For example, to have
several sets of key-value pairs, make the column an ARRAY whose elements are MAP. To make a set
of key-value pairs that holds more elaborate information, make a MAP column whose VALUE part
contains an ARRAY or a STRUCT.
-- Print a report of a subset of dreams, filtering based on both the lookup key
-- and the detailed value.
SELECT dream_id, details.KEY AS "Dream Type", details.VALUE AS "Dream Summary"
FROM dream_journal, dream_journal.details
WHERE
details.KEY IN ('Happy', 'Pleasant', 'Joyous')
AND details.VALUE LIKE '%childhood%';
The following example shows a more elaborate version of the previous table, where the VALUE part of the MAP entry
is a STRUCT rather than a scalar type. Now instead of referring to the VALUE pseudocolumn directly, you use dot
notation to refer to the STRUCT fields inside it.
)
STORED AS PARQUET;
-- Remember that if the ITEM or VALUE contains a STRUCT, you can reference
-- the STRUCT fields directly without the .ITEM or .VALUE qualifier.
SELECT dream_id, details.KEY AS "Dream Type", details.summary AS "Dream Summary",
details.duration AS "Duration"
FROM better_dream_journal, better_dream_journal.details
WHERE
details.KEY IN ('Anxiety', 'Nightmare')
AND details.duration > 60
AND details.woke_up = TRUE;
Note: Because Hive currently cannot construct individual rows using complex types through the
INSERT ... VALUES syntax, you prepare the data in flat form in a separate table, then copy it to
the table with complex columns using INSERT ... SELECT and the complex type constructors. See
Constructing Parquet Files with Complex Columns Using Hive on page 179 for examples.
For example, to model a relationship like a dimension table and a fact table, you typically use an ARRAY where each
array element is a STRUCT. The STRUCT fields represent what would traditionally be columns in a separate joined
table. It makes little sense to use a STRUCT as the top-level type for a column, because you could just make the fields
of the STRUCT into regular table columns.
Perhaps the only use case for a top-level STRUCT would be to to allow STRUCT fields with the same name as columns
to coexist in the same table. The following example shows how a table could have a column named ID, and two separate
STRUCT fields also named ID. Because the STRUCT fields are always referenced using qualified names, the identical
ID names do not cause a conflict.
It is common to make the value portion of each key-value pair in a MAP a STRUCT, ARRAY of STRUCT, or other complex
type variation. That way, each key in the MAP can be associated with a flexible and extensible data structure. The key
values are not predefined ahead of time (other than by specifying their data type). Therefore, the MAP can accommodate
a rapidly evolving schema, or sparse data structures where each row contains only a few data values drawn from a
large set of possible choices.
Although you can use an ARRAY of scalar values as the top-level column in a table, such a simple array is typically of
limited use for analytic queries. The only property of the array elements, aside from the element value, is the ordering
sequence available through the POS pseudocolumn. To record any additional item about each array element, such as
a TIMESTAMP or a symbolic name, you use an ARRAY of STRUCT rather than of scalar values.
If you are considering having multiple ARRAY or MAP columns, with related items under the same position in each
ARRAY or the same key in each MAP, prefer to use a STRUCT to group all the related items into a single ARRAY or MAP.
Doing so avoids the additional storage overhead and potential duplication of key values from having an extra complex
type column. Also, because each ARRAY or MAP that you reference in the query SELECT list requires an additional join
clause, minimizing the number of complex type columns also makes the query easier to read and maintain, relying
more on dot notation to refer to the relevant fields rather than a sequence of join clauses.
For example, here is a table with several complex type columns all at the top level and containing only scalar types.
To retrieve every data item for the row requires a separate join for each ARRAY or MAP column. The fields of the STRUCT
can be referenced using dot notation, but there is no real advantage to using the STRUCT at the top level rather than
just making separate columns FIELD1 and FIELD2.
describe complex_types_top_level;
+------+-----------------+
| name | type |
+------+-----------------+
| id | bigint |
| a1 | array<int> |
| a2 | array<string> |
| s | struct< |
| | field1:int, |
| | field2:string |
| | > |
| m1 | map<int,string> |
| m2 | map<string,int> |
+------+-----------------+
select
id,
a1.item,
a2.item,
s.field1,
s.field2,
m1.key,
m1.value,
m2.key,
m2.value
from
complex_types_top_level,
complex_types_top_level.a1,
complex_types_top_level.a2,
complex_types_top_level.m1,
complex_types_top_level.m2;
For example, here is a table with columns containing an ARRAY of STRUCT, a MAP where each key value is a STRUCT,
and a MAP where each key value is an ARRAY of STRUCT.
The DESCRIBE statement rearranges the < and > separators and the field names within each STRUCT for easy readability:
DESCRIBE nesting_demo;
+----------------+-----------------------------+
| name | type |
+----------------+-----------------------------+
| user_id | bigint |
| family_members | array<struct< |
| | name:string, |
| | email:string, |
| | date_joined:timestamp |
| | >> |
| foo | map<string,struct< |
| | f1:int, |
| | f2:int, |
| | f3:timestamp, |
| | f4:boolean |
| | >> |
| gameplay | map<string,array<struct< |
| | name:string, |
| | highest:bigint, |
| | lives_used:int, |
| | total_spent:decimal(16,2) |
| | >>> |
+----------------+-----------------------------+
To query the complex type columns, you use join notation to refer to the lowest-level scalar values. If the value is an
ARRAY element, the fully qualified name includes the ITEM pseudocolumn. If the value is inside a MAP, the fully qualified
name includes the KEY or VALUE pseudocolumn. Each reference to a different ARRAY or MAP (even if nested inside
another complex type) requires an additional join clause.
SELECT
-- The lone scalar field doesn't require any dot notation or join clauses.
user_id
-- Retrieve the fields of a STRUCT inside an ARRAY.
-- The FAMILY_MEMBERS name refers to the FAMILY_MEMBERS table alias defined later in
the FROM clause.
, family_members.item.name
, family_members.item.email
, family_members.item.date_joined
-- Retrieve the KEY and VALUE fields of a MAP, with the value being a STRUCT consisting
of more fields.
-- The FOO name refers to the FOO table alias defined later in the FROM clause.
, foo.key
, foo.value.f1
, foo.value.f2
, foo.value.f3
, foo.value.f4
-- Retrieve the KEY fields of a MAP, and expand the VALUE part into ARRAY items consisting
of STRUCT fields.
-- The GAMEPLAY name refers to the GAMEPLAY table alias defined later in the FROM clause
(referring to the MAP item).
-- The GAME_N name refers to the GAME_N table alias defined later in the FROM clause
(referring to the ARRAY
-- inside the MAP item's VALUE part.)
, gameplay.key
, game_n.name
, game_n.highest
, game_n.lives_used
, game_n.total_spent
FROM
nesting_demo
, nesting_demo.family_members AS family_members
, nesting_demo.foo AS foo
, nesting_demo.gameplay AS gameplay
, nesting_demo.gameplay.value AS game_n;
Once you understand the notation to refer to a particular data item in the SELECT list, you can use the same qualified
name to refer to that data item in other parts of the query, such as the WHERE clause, ORDER BY or GROUP BY clauses,
or calls to built-in functions. For example, you might frequently retrieve the VALUE part of each MAP item in the SELECT
list, while choosing the specific MAP items by running comparisons against the KEY part in the WHERE clause.
Accessing Complex Type Data in Flattened Form Using Views
The layout of complex and nested types is largely a physical consideration. The complex type columns reside in the
same data files rather than in separate normalized tables, for your convenience in managing related data sets and
performance in querying related data sets. You can use views to treat tables with complex types as if they were
flattened. By putting the join logic and references to the complex type columns in the view definition, you can query
the same tables using existing queries intended for tables containing only scalar columns. This technique also lets you
use tables with complex types with BI tools that are not aware of the data types and query notation for accessing
complex type columns.
For example, the variation of the TPC-H schema containing complex types has a table REGION. This table has 5 rows,
corresponding to 5 regions such as NORTH AMERICA and AFRICA. Each row has an ARRAY column, where each array
item is a STRUCT containing details about a country in that region.
DESCRIBE region;
+-------------+-------------------------+
| name | type |
+-------------+-------------------------+
| r_regionkey | smallint |
| r_name | string |
| r_comment | string |
| r_nations | array<struct< |
| | n_nationkey:smallint, |
| | n_name:string, |
| | n_comment:string |
| | >> |
+-------------+-------------------------+
The same data could be represented in traditional denormalized form, as a single table where the information about
each region is repeated over and over, alongside the information about each country. The nested complex types let
us avoid the repetition, while still keeping the data in a single table rather than normalizing across multiple tables.
To use this table with a JDBC or ODBC application that expected scalar columns, we could create a view that represented
the result set as a set of scalar columns (three columns from the original table, plus three more from the STRUCT fields
of the array elements). In the following examples, any column with an R_* prefix is taken unchanged from the original
table, while any column with an N_* prefix is extracted from the STRUCT inside the ARRAY.
Then we point the application queries at the view rather than the original table. From the perspective of the view,
there are 25 rows in the result set, one for each nation in each region, and queries can refer freely to fields related to
the region or the nation.
-- Retrieve info such as the nation name from the original R_NATIONS array elements.
select n_name from region_view where r_name in ('EUROPE', 'ASIA');
+----------------+
| n_name |
+----------------+
| UNITED KINGDOM |
| RUSSIA |
| ROMANIA |
| GERMANY |
| FRANCE |
| VIETNAM |
| CHINA |
| JAPAN |
| INDONESIA |
| INDIA |
+----------------+
-- For conciseness, we only list some view columns in the SELECT list.
-- SELECT * would bring back all the data, unlike SELECT *
-- queries on the original table with complex type columns.
SELECT r_regionkey, r_name, n_nationkey, n_name FROM region_view LIMIT 7;
+-------------+--------+-------------+----------------+
| r_regionkey | r_name | n_nationkey | n_name |
+-------------+--------+-------------+----------------+
| 3 | EUROPE | 23 | UNITED KINGDOM |
| 3 | EUROPE | 22 | RUSSIA |
| 3 | EUROPE | 19 | ROMANIA |
| 3 | EUROPE | 7 | GERMANY |
| 3 | EUROPE | 6 | FRANCE |
| 2 | ASIA | 21 | VIETNAM |
| 2 | ASIA | 18 | CHINA |
+-------------+--------+-------------+----------------+
SHOW TABLES;
+----------+
| name |
+----------+
| customer |
| part |
| region |
| supplier |
+----------+
DESCRIBE customer;
+--------------+------------------------------------+
| name | type |
+--------------+------------------------------------+
| c_custkey | bigint |
| c_name | string |
| c_address | string |
| c_nationkey | smallint |
| c_phone | string |
| c_acctbal | decimal(12,2) |
| c_mktsegment | string |
| c_comment | string |
| c_orders | array<struct< |
| | o_orderkey:bigint, |
| | o_orderstatus:string, |
| | o_totalprice:decimal(12,2), |
| | o_orderdate:string, |
| | o_orderpriority:string, |
| | o_clerk:string, |
| | o_shippriority:int, |
| | o_comment:string, |
| | o_lineitems:array<struct< |
| | l_partkey:bigint, |
| | l_suppkey:bigint, |
| | l_linenumber:int, |
| | l_quantity:decimal(12,2), |
| | l_extendedprice:decimal(12,2), |
| | l_discount:decimal(12,2), |
| | l_tax:decimal(12,2), |
| | l_returnflag:string, |
| | l_linestatus:string, |
| | l_shipdate:string, |
| | l_commitdate:string, |
| | l_receiptdate:string, |
| | l_shipinstruct:string, |
| | l_shipmode:string, |
| | l_comment:string |
| | >> |
| | >> |
+--------------+------------------------------------+
DESCRIBE part;
+---------------+---------------+
| name | type |
+---------------+---------------+
| p_partkey | bigint |
| p_name | string |
| p_mfgr | string |
| p_brand | string |
| p_type | string |
| p_size | int |
| p_container | string |
| p_retailprice | decimal(12,2) |
| p_comment | string |
+---------------+---------------+
DESCRIBE region;
+-------------+--------------------------------------------------------------------+
| name | type |
+-------------+--------------------------------------------------------------------+
| r_regionkey | smallint |
| r_name | string |
| r_comment | string |
| r_nations | array<struct<n_nationkey:smallint,n_name:string,n_comment:string>> |
+-------------+--------------------------------------------------------------------+
DESCRIBE supplier;
+-------------+----------------------------------------------+
| name | type |
+-------------+----------------------------------------------+
| s_suppkey | bigint |
| s_name | string |
| s_address | string |
| s_nationkey | smallint |
| s_phone | string |
| s_acctbal | decimal(12,2) |
| s_comment | string |
| s_partsupps | array<struct<ps_partkey:bigint, |
| | ps_availqty:int,ps_supplycost:decimal(12,2), |
| | ps_comment:string>> |
+-------------+----------------------------------------------+
Create table with ARRAY in Impala, load data in Hive, query in Impala:
This example shows the cycle of creating the tables and querying the complex data in Impala, and using Hive (either
the hive shell or beeline) for the data loading step. The data starts in flattened, denormalized form in a text table.
Hive writes the corresponding Parquet data, including an ARRAY column. Then Impala can run analytic queries on the
Parquet table, using join notation to unpack the ARRAY column.
/* Initial DDL and loading of flat, denormalized data happens in impala-shell */CREATE
TABLE flat_array (country STRING, city STRING);INSERT INTO flat_array VALUES
('Canada', 'Toronto') , ('Canada', 'Vancouver') , ('Canada', "St. John\'s")
, ('Canada', 'Saint John') , ('Canada', 'Montreal') , ('Canada', 'Halifax')
, ('Canada', 'Winnipeg') , ('Canada', 'Calgary') , ('Canada', 'Saskatoon')
, ('Canada', 'Ottawa') , ('Canada', 'Yellowknife') , ('France', 'Paris')
, ('France', 'Nice') , ('France', 'Marseilles') , ('France', 'Cannes')
, ('Greece', 'Athens') , ('Greece', 'Piraeus') , ('Greece', 'Hania')
, ('Greece', 'Heraklion') , ('Greece', 'Rethymnon') , ('Greece', 'Fira');
CREATE TABLE complex_array (country STRING, city ARRAY <STRING>) STORED AS PARQUET;
REFRESH complex_array;
SELECT country, city.item FROM complex_array, complex_array.city
+---------+-------------+
| country | item |
+---------+-------------+
| Canada | Toronto |
| Canada | Vancouver |
| Canada | St. John's |
| Canada | Saint John |
| Canada | Montreal |
| Canada | Halifax |
| Canada | Winnipeg |
| Canada | Calgary |
| Canada | Saskatoon |
| Canada | Ottawa |
| Canada | Yellowknife |
| France | Paris |
| France | Nice |
| France | Marseilles |
| France | Cannes |
| Greece | Athens |
| Greece | Piraeus |
| Greece | Hania |
| Greece | Heraklion |
| Greece | Rethymnon |
| Greece | Fira |
+---------+-------------+
Create table with STRUCT and ARRAY in Impala, load data in Hive, query in Impala:
This example shows the cycle of creating the tables and querying the complex data in Impala, and using Hive (either
the hive shell or beeline) for the data loading step. The data starts in flattened, denormalized form in a text table.
Hive writes the corresponding Parquet data, including a STRUCT column with an ARRAY field. Then Impala can run
analytic queries on the Parquet table, using join notation to unpack the ARRAY field from the STRUCT column.
CREATE TABLE complex_struct_array (continent STRING, country STRUCT <name: STRING, city:
ARRAY <STRING> >) STORED AS PARQUET;
REFRESH complex_struct_array;
SELECT t1.continent, t1.country.name, t2.item
FROM complex_struct_array t1, t1.country.city t2
+---------------+--------------+-------------+
| continent | country.name | item |
+---------------+--------------+-------------+
| Europe | France | Paris |
| Europe | France | Nice |
| Europe | France | Marseilles |
| Europe | France | Cannes |
| Europe | Greece | Athens |
| Europe | Greece | Piraeus |
| Europe | Greece | Hania |
| Europe | Greece | Heraklion |
| Europe | Greece | Rethymnon |
| Europe | Greece | Fira |
| North America | Canada | Toronto |
| North America | Canada | Vancouver |
| North America | Canada | St. John's |
| North America | Canada | Saint John |
| North America | Canada | Montreal |
| North America | Canada | Halifax |
| North America | Canada | Winnipeg |
| North America | Canada | Calgary |
| North America | Canada | Saskatoon |
| North America | Canada | Ottawa |
| North America | Canada | Yellowknife |
+---------------+--------------+-------------+
you can keep the same number of rows as in the original normalized table, and put all the associated data from the
other table in a single new column.
In this flattening scenario, you might frequently use a column that is an ARRAY consisting of STRUCT elements, where
each field within the STRUCT corresponds to a column name from the table that you are combining.
The following example shows a traditional normalized layout using two tables, and then an equivalent layout using
complex types in a single table.
-- This table just stores numbers, allowing us to look up details about the employee
-- and details about their vacation time using a three-table join query.
CREATE table employee_vacations
(
employee_id BIGINT,
vacation_id BIGINT
)
STORED AS PARQUET;
-- For analytic queries using complex types, we can bundle the dimension table
-- and multiple fact tables into a single table.
CREATE TABLE employee_vacations_nested_types
(
-- We might still use the employee_id for other join queries.
-- The table needs at least one scalar column to serve as an identifier
-- for the complex type columns.
employee_id BIGINT,
-- The ADDRESS_TYPE column, with a small number of predefined values that are distinct
-- for each employee, makes the EMPLOYEE_CONTACT table a good candidate to turn into a
MAP,
-- with each row represented as a STRUCT. The string value from ADDRESS_TYPE becomes
the
-- "key" (the anonymous first field) of the MAP.
contact MAP < STRING, STRUCT <
address: STRING,
phone: STRING,
email: STRING
>>
)
STORED AS PARQUET;
Interchanging Complex Type Tables and Data Files with Hive and Other Components
You can produce Parquet data files through several Hadoop components and APIs, as explained in
http://www.cloudera.com/documentation/enterprise/latest/topics/cdh_ig_parquet.html.
If you have a Hive-created Parquet table that includes ARRAY, STRUCT, or MAP columns, Impala can query that same
table in Impala 2.3 / CDH 5.5 and higher, subject to the usual restriction that all other columns are of data types
supported by Impala, and also that the file type of the table must be Parquet.
If you have a Parquet data file produced outside of Impala, Impala can automatically deduce the appropriate table
structure using the syntax CREATE TABLE ... LIKE PARQUET 'hdfs_path_of_parquet_file'. In Impala 2.3
/ CDH 5.5 and higher, this feature works for Parquet files that include ARRAY, STRUCT, or MAP types.
# In the Unix shell, find the path of any Parquet data file in that HDFS directory.
$ hdfs dfs -ls hdfs://localhost:20500/test-warehouse/tpch_nested_parquet.db/customer
Found 4 items
-rwxr-xr-x 3 dev supergroup 171298918 2015-09-22 23:30
hdfs://localhost:20500/blah/tpch_nested_parquet.db/customer/000000_0
...
/* Back in impala-shell, use the HDFS path in a CREATE TABLE LIKE PARQUET statement. */
CREATE TABLE customer_ctlp
LIKE PARQUET 'hdfs://localhost:20500/blah/tpch_nested_parquet.db/customer/000000_0'
STORED AS PARQUET;
/* Confirm that old and new tables have the same column layout, including complex types.
*/
DESCRIBE tpch_nested_parquet.customer
+--------------+------------------------------------+---------+
| name | type | comment |
+--------------+------------------------------------+---------+
| c_custkey | bigint | |
| c_name | string | |
| c_address | string | |
| c_nationkey | smallint | |
| c_phone | string | |
| c_acctbal | decimal(12,2) | |
| c_mktsegment | string | |
| c_comment | string | |
| c_orders | array<struct< | |
| | o_orderkey:bigint, | |
| | o_orderstatus:string, | |
| | o_totalprice:decimal(12,2), | |
| | o_orderdate:string, | |
| | o_orderpriority:string, | |
| | o_clerk:string, | |
| | o_shippriority:int, | |
| | o_comment:string, | |
| | o_lineitems:array<struct< | |
| | l_partkey:bigint, | |
| | l_suppkey:bigint, | |
| | l_linenumber:int, | |
| | l_quantity:decimal(12,2), | |
| | l_extendedprice:decimal(12,2), | |
| | l_discount:decimal(12,2), | |
| | l_tax:decimal(12,2), | |
| | l_returnflag:string, | |
| | l_linestatus:string, | |
| | l_shipdate:string, | |
| | l_commitdate:string, | |
| | l_receiptdate:string, | |
| | l_shipinstruct:string, | |
| | l_shipmode:string, | |
| | l_comment:string | |
| | >> | |
| | >> | |
+--------------+------------------------------------+---------+
describe customer_ctlp;
+--------------+------------------------------------+-----------------------------+
| name | type | comment |
+--------------+------------------------------------+-----------------------------+
| c_custkey | bigint | Inferred from Parquet file. |
| c_name | string | Inferred from Parquet file. |
| c_address | string | Inferred from Parquet file. |
| c_nationkey | int | Inferred from Parquet file. |
| c_phone | string | Inferred from Parquet file. |
| c_acctbal | decimal(12,2) | Inferred from Parquet file. |
| c_mktsegment | string | Inferred from Parquet file. |
| c_comment | string | Inferred from Parquet file. |
| c_orders | array<struct< | Inferred from Parquet file. |
| | o_orderkey:bigint, | |
| | o_orderstatus:string, | |
| | o_totalprice:decimal(12,2), | |
| | o_orderdate:string, | |
| | o_orderpriority:string, | |
| | o_clerk:string, | |
| | o_shippriority:int, | |
| | o_comment:string, | |
| | o_lineitems:array<struct< | |
| | l_partkey:bigint, | |
| | l_suppkey:bigint, | |
| | l_linenumber:int, | |
| | l_quantity:decimal(12,2), | |
| | l_extendedprice:decimal(12,2), | |
| | l_discount:decimal(12,2), | |
| | l_tax:decimal(12,2), | |
| | l_returnflag:string, | |
| | l_linestatus:string, | |
| | l_shipdate:string, | |
| | l_commitdate:string, | |
| | l_receiptdate:string, | |
| | l_shipinstruct:string, | |
| | l_shipmode:string, | |
| | l_comment:string | |
| | >> | |
| | >> | |
+--------------+------------------------------------+-----------------------------+
Literals
Each of the Impala data types has corresponding notation for literal values of that type. You specify literal values in
SQL statements, such as in the SELECT list or WHERE clause of a query, or as an argument to a function call. See Data
Types on page 116 for a complete list of types, ranges, and conversion rules.
Numeric Literals
To write literals for the integer types (TINYINT, SMALLINT, INT, and BIGINT), use a sequence of digits with optional
leading zeros.
To write literals for the floating-point types (DECIMAL, FLOAT, and DOUBLE), use a sequence of digits with an optional
decimal point (. character). To preserve accuracy during arithmetic expressions, Impala interprets floating-point literals
as the DECIMAL type with the smallest appropriate precision and scale, until required by the context to convert the
result to FLOAT or DOUBLE.
Integer values are promoted to floating-point when necessary, based on the context.
You can also use exponential notation by including an e character. For example, 1e6 is 1 times 10 to the power of 6
(1 million). A number in exponential notation is always interpreted as floating-point.
When Impala encounters a numeric literal, it considers the type to be the “smallest” that can accurately represent the
value. The type is promoted to larger or more accurate types if necessary, based on subsequent parts of an expression.
For example, you can see by the types Impala defines for the following table columns how it interprets the corresponding
numeric literals:
String Literals
String literals are quoted using either single or double quotation marks. You can use either kind of quotes for string
literals, even both kinds for different literals within the same statement.
Quoted literals are considered to be of type STRING. To use quoted literals in contexts requiring a CHAR or VARCHAR
value, CAST() the literal to a CHAR or VARCHAR of the appropriate length.
Escaping special characters:
To encode special characters within a string literal, precede them with the backslash (\) escape character:
• \t represents a tab.
• \n represents a newline or linefeed. This might cause extra line breaks in impala-shell output.
• \r represents a carriage return. This might cause unusual formatting (making it appear that some content is
overwritten) in impala-shell output.
• \b represents a backspace. This might cause unusual formatting (making it appear that some content is overwritten)
in impala-shell output.
• \0 represents an ASCII nul character (not the same as a SQL NULL). This might not be visible in impala-shell
output.
• \Z represents a DOS end-of-file character. This might not be visible in impala-shell output.
• \% and \_ can be used to escape wildcard characters within the string passed to the LIKE operator.
• \ followed by 3 octal digits represents the ASCII code of a single character; for example, \101 is ASCII 65, the
character A.
• Use two consecutive backslashes (\\) to prevent the backslash from being interpreted as an escape character.
• Use the backslash to escape single or double quotation mark characters within a string literal, if the literal is
enclosed by the same type of quotation mark.
• If the character following the \ does not represent the start of a recognized escape sequence, the character is
passed through unchanged.
Quotes within quotes:
To include a single quotation character within a string value, enclose the literal with either single or double quotation
marks, and optionally escape the single quote as a \' sequence. Earlier releases required escaping a single quote inside
double quotes. Continue using escape sequences in this case if you also need to run your SQL code on older versions
of Impala.
To include a double quotation character within a string value, enclose the literal with single quotation marks, no
escaping is necessary in this case. Or, enclose the literal with double quotation marks and escape the double quote as
a \" sequence.
Note: The CREATE TABLE clauses FIELDS TERMINATED BY, ESCAPED BY, and LINES TERMINATED
BY have special rules for the string literal used for their argument, because they all require a single
character. You can use a regular character surrounded by single or double quotation marks, an octal
sequence such as '\054' (representing a comma), or an integer in the range '-127'..'128' (with
quotation marks but no backslash), which is interpreted as a single-byte ASCII character. Negative
values are subtracted from 256; for example, FIELDS TERMINATED BY '-2' sets the field delimiter
to ASCII code 254, the “Icelandic Thorn” character used as a delimiter by some data formats.
impala-shell considerations:
When dealing with output that includes non-ASCII or non-printable characters such as linefeeds and backspaces, use
the impala-shell options to save to a file, turn off pretty printing, or both rather than relying on how the output
appears visually. See impala-shell Configuration Options on page 470 for a list of impala-shell options.
Boolean Literals
For BOOLEAN values, the literals are TRUE and FALSE, with no quotation marks and case-insensitive.
Examples:
select true;
select * from t1 where assertion = false;
select case bool_col when true then 'yes' when false 'no' else 'null' end from t1;
Timestamp Literals
Impala automatically converts STRING literals of the correct format into TIMESTAMP values. Timestamp values are
accepted in the format "yyyy-MM-dd HH:mm:ss.SSSSSS", and can consist of just the date, or just the time, with
or without the fractional second portion. For example, you can specify TIMESTAMP values such as '1966-07-30',
'08:30:00', or '1985-09-25 17:45:30.005'. Casting an integer or floating-point value N to TIMESTAMP produces
a value that is N seconds past the start of the epoch date (January 1, 1970). By default, the result value represents a
date and time in the UTC time zone. If the setting --use_local_tz_for_unix_timestamp_conversions=true
is in effect, the resulting TIMESTAMP represents a date and time in the local time zone.
You can also use INTERVAL expressions to add or subtract from timestamp literal values, such as CAST('1966-07-30'
AS TIMESTAMP) + INTERVAL 5 YEARS + INTERVAL 3 DAYS. See TIMESTAMP Data Type on page 148 for details.
Depending on your data pipeline, you might receive date and time data as text, in notation that does not exactly match
the format for Impala TIMESTAMP literals. See Impala Date and Time Functions on page 363 for functions that can
convert between a variety of string literals (including different field order, separators, and timezone notation) and
equivalent TIMESTAMP or numeric values.
NULL
The notion of NULL values is familiar from all kinds of database systems, but each SQL dialect can have its own behavior
and restrictions on NULL values. For Big Data processing, the precise semantics of NULL values are significant: any
misunderstanding could lead to inaccurate results or misformatted data, that could be time-consuming to correct for
large data sets.
• NULL is a different value than an empty string. The empty string is represented by a string literal with nothing
inside, "" or ''.
• In a delimited text file, the NULL value is represented by the special token \N.
• When Impala inserts data into a partitioned table, and the value of one of the partitioning columns is NULL or the
empty string, the data is placed in a special partition that holds only these two kinds of values. When these values
are returned in a query, the result is NULL whether the value was originally NULL or an empty string. This behavior
is compatible with the way Hive treats NULL values in partitioned tables. Hive does not allow empty strings as
partition keys, and it returns a string value such as __HIVE_DEFAULT_PARTITION__ instead of NULL when such
values are returned from a query. For example:
• There is no NOT NULL clause when defining a column to prevent NULL values in that column.
• There is no DEFAULT clause to specify a non-NULL default value.
• If an INSERT operation mentions some columns but not others, the unmentioned columns contain NULL for all
inserted rows.
• In Impala 1.2.1 and higher, all NULL values come at the end of the result set for ORDER BY ... ASC queries, and
at the beginning of the result set for ORDER BY ... DESC queries. In effect, NULL is considered greater than all
other values for sorting purposes. The original Impala behavior always put NULL values at the end, even for ORDER
BY ... DESC queries. The new behavior in Impala 1.2.1 makes Impala more compatible with other popular
database systems. In Impala 1.2.1 and higher, you can override or specify the sorting behavior for NULL by adding
the clause NULLS FIRST or NULLS LAST at the end of the ORDER BY clause.
Note: Because the NULLS FIRST and NULLS LAST keywords are not currently available in Hive
queries, any views you create using those keywords will not be available through Hive.
• In all other contexts besides sorting with ORDER BY, comparing a NULL to anything else returns NULL, making
the comparison meaningless. For example, 10 > NULL produces NULL, 10 < NULL also produces NULL, 5
BETWEEN 1 AND NULL produces NULL, and so on.
Several built-in functions serve as shorthand for evaluating expressions and returning NULL, 0, or some other substitution
value depending on the expression result: ifnull(), isnull(), nvl(), nullif(), nullifzero(), and
zeroifnull(). See Impala Conditional Functions on page 391 for details.
SQL Operators
SQL operators are a class of comparison functions that are widely used within the WHERE clauses of SELECT statements.
Arithmetic Operators
The arithmetic operators use expressions with a left-hand argument, the operator, and then (in most cases) a right-hand
argument.
Syntax:
• * and /: Multiplication and division respectively. Both arguments must be of numeric types.
When multiplying, the shorter argument is promoted if necessary (such as SMALLINT to INT or BIGINT, or FLOAT
to DOUBLE), and then the result is promoted again to the next larger type. Thus, multiplying a TINYINT and an
INT produces a BIGINT result. Multiplying a FLOAT and a FLOAT produces a DOUBLE result. Multiplying a FLOAT
and a DOUBLE or a DOUBLE and a DOUBLE produces a DECIMAL(38,17), because DECIMAL values can represent
much larger and more precise values than DOUBLE.
When dividing, Impala always treats the arguments and result as DOUBLE values to avoid losing precision. If you
need to insert the results of a division operation into a FLOAT column, use the CAST() function to convert the
result to the correct type.
• DIV: Integer division. Arguments are not promoted to a floating-point type, and any fractional result is discarded.
For example, 13 DIV 7 returns 1, 14 DIV 7 returns 2, and 15 DIV 7 returns 2. This operator is the same as
the QUOTIENT() function.
• %: Modulo operator. Returns the remainder of the left-hand argument divided by the right-hand argument. Both
arguments must be of one of the integer types.
• &, |, ~, and ^: Bitwise operators that return the logical AND, logical OR, NOT, or logical XOR (exclusive OR) of their
argument values. Both arguments must be of one of the integer types. If the arguments are of different type, the
argument with the smaller type is implicitly extended to match the argument with the longer type.
You can chain a sequence of arithmetic expressions, optionally grouping them with parentheses.
The arithmetic operators generally do not have equivalent calling conventions using functional notation. For example,
prior to Impala 2.2.0 / CDH 5.4.0, there is no MOD() function equivalent to the % modulo operator. Conversely, there
are some arithmetic functions that do not have a corresponding operator. For example, for exponentiation you use
the POW() function, but there is no ** exponentiation operator. See Impala Mathematical Functions on page 339 for
the arithmetic functions you can use.
Complex type considerations:
To access a column with a complex type (ARRAY, STRUCT, or MAP) in an aggregation function, you unpack the individual
elements using join notation in the query, and then apply the function to the final scalar item, field, key, or value at
the bottom of any nested type hierarchy in the column. See Complex Types (CDH 5.5 or higher only) on page 156 for
details about using complex types in Impala.
The following example demonstrates calls to several aggregation functions using values from a column containing
nested complex types (an ARRAY of STRUCT items). The array is unpacked inside the query using join notation. The
array elements are referenced using the ITEM pseudocolumn, and the structure fields inside the array elements are
referenced using dot notation. Numeric values such as SUM() and AVG() are computed using the numeric R_NATIONKEY
field, and the general-purpose MAX() and MIN() values are computed from the string N_NAME field.
describe region;
+-------------+-------------------------+---------+
| name | type | comment |
+-------------+-------------------------+---------+
| r_regionkey | smallint | |
| r_name | string | |
| r_comment | string | |
| r_nations | array<struct< | |
| | n_nationkey:smallint, | |
| | n_name:string, | |
| | n_comment:string | |
| | >> | |
+-------------+-------------------------+---------+
select r_name, r_nations.item.n_nationkey
from region, region.r_nations as r_nations
order by r_name, r_nations.item.n_nationkey;
+-------------+------------------+
| r_name | item.n_nationkey |
+-------------+------------------+
| AFRICA | 0 |
| AFRICA | 5 |
| AFRICA | 14 |
| AFRICA | 15 |
| AFRICA | 16 |
| AMERICA | 1 |
| AMERICA | 2 |
| AMERICA | 3 |
| AMERICA | 17 |
| AMERICA | 24 |
| ASIA | 8 |
| ASIA | 9 |
| ASIA | 12 |
| ASIA | 18 |
| ASIA | 21 |
| EUROPE | 6 |
| EUROPE | 7 |
| EUROPE | 19 |
| EUROPE | 22 |
| EUROPE | 23 |
| MIDDLE EAST | 4 |
| MIDDLE EAST | 10 |
| MIDDLE EAST | 11 |
| MIDDLE EAST | 13 |
| MIDDLE EAST | 20 |
+-------------+------------------+
select
r_name,
count(r_nations.item.n_nationkey) as count,
sum(r_nations.item.n_nationkey) as sum,
avg(r_nations.item.n_nationkey) as avg,
min(r_nations.item.n_name) as minimum,
max(r_nations.item.n_name) as maximum,
ndv(r_nations.item.n_nationkey) as distinct_vals
from
region, region.r_nations as r_nations
group by r_name
order by r_name;
+-------------+-------+-----+------+-----------+----------------+---------------+
| r_name | count | sum | avg | minimum | maximum | distinct_vals |
+-------------+-------+-----+------+-----------+----------------+---------------+
| AFRICA | 5 | 50 | 10 | ALGERIA | MOZAMBIQUE | 5 |
| AMERICA | 5 | 47 | 9.4 | ARGENTINA | UNITED STATES | 5 |
| ASIA | 5 | 68 | 13.6 | CHINA | VIETNAM | 5 |
| EUROPE | 5 | 77 | 15.4 | FRANCE | UNITED KINGDOM | 5 |
| MIDDLE EAST | 5 | 58 | 11.6 | EGYPT | SAUDI ARABIA | 5 |
+-------------+-------+-----+------+-----------+----------------+---------------+
You cannot refer to a column with a complex data type (ARRAY, STRUCT, or MAP) directly in an operator. You can apply
operators only to scalar values that make up a complex type (the fields of a STRUCT, the items of an ARRAY, or the key
or value portion of a MAP) as part of a join query that refers to the scalar value using the appropriate dot notation or
ITEM, KEY, or VALUE pseudocolumn names.
The following example shows how to do an arithmetic operation using a numeric field of a STRUCT type that is an item
within an ARRAY column. Once the scalar numeric value R_NATIONKEY is extracted, it can be used in an arithmetic
expression, such as multiplying by 10:
BETWEEN Operator
In a WHERE clause, compares an expression to both a lower and upper bound. The comparison is successful is the
expression is greater than or equal to the lower bound, and less than or equal to the upper bound. If the bound values
are switched, so the lower bound is greater than the upper bound, does not match any values.
Syntax:
Data types: Typically used with numeric data types. Works with any data type, although not very practical for BOOLEAN
values. (BETWEEN false AND true will match all BOOLEAN values.) Use CAST() if necessary to ensure the lower
and upper bound values are compatible types. Call string or date/time functions if necessary to extract or transform
the relevant portion to compare, especially if the value can be transformed into a number.
Usage notes:
Be careful when using short string operands. A longer string that starts with the upper bound value will not be included,
because it is considered greater than the upper bound. For example, BETWEEN 'A' and 'M' would not match the
string value 'Midway'. Use functions such as upper(), lower(), substr(), trim(), and so on if necessary to ensure
the comparison works as expected.
Complex type considerations:
You cannot refer to a column with a complex data type (ARRAY, STRUCT, or MAP) directly in an operator. You can apply
operators only to scalar values that make up a complex type (the fields of a STRUCT, the items of an ARRAY, or the key
or value portion of a MAP) as part of a join query that refers to the scalar value using the appropriate dot notation or
ITEM, KEY, or VALUE pseudocolumn names.
Examples:
-- Retrieve data for names beginning with 'A' through 'M' inclusive.
-- Only test the first letter to ensure all the values starting with 'M' are matched.
-- Do a case-insensitive comparison to match names with various capitalization
conventions.
select last_name from customers where upper(substr(last_name,1,1)) between 'A' and 'M';
The following example shows how to do a BETWEEN comparison using a numeric field of a STRUCT type that is an item
within an ARRAY column. Once the scalar numeric value R_NATIONKEY is extracted, it can be used in a comparison
operator:
| | >> | |
+-------------+-------------------------+---------+
Comparison Operators
Impala supports the familiar comparison operators for checking equality and sort order for the column data types:
Syntax:
The following example shows how to do an arithmetic operation using a numeric field of a STRUCT type that is an item
within an ARRAY column. Once the scalar numeric value R_NATIONKEY is extracted, it can be used with a comparison
operator such as <:
EXISTS Operator
The EXISTS operator tests whether a subquery returns any results. You typically use it to find values from one table
that have corresponding values in another table.
The converse, NOT EXISTS, helps to find all the values from one table that do not have any corresponding values in
another table.
Syntax:
EXISTS (subquery)
NOT EXISTS (subquery)
Usage notes:
The subquery can refer to a different table than the outer query block, or the same table. For example, you might use
EXISTS or NOT EXISTS to check the existence of parent/child relationships between two columns of the same table.
You can also use operators and function calls within the subquery to test for other kinds of relationships other than
strict equality. For example, you might use a call to COUNT() in the subquery to check whether the number of matching
values is higher or lower than some limit. You might call a UDF in the subquery to check whether values in one table
matches a hashed representation of those same values in a different table.
NULL considerations:
If the subquery returns any value at all (even NULL), EXISTS returns TRUE and NOT EXISTS returns false.
The following example shows how even when the subquery returns only NULL values, EXISTS still returns TRUE and
thus matches all the rows from the table in the outer query block.
However, if the table in the subquery is empty and so the subquery returns an empty result set, EXISTS returns FALSE:
You cannot refer to a column with a complex data type (ARRAY, STRUCT, or MAP) directly in an operator. You can apply
operators only to scalar values that make up a complex type (the fields of a STRUCT, the items of an ARRAY, or the key
or value portion of a MAP) as part of a join query that refers to the scalar value using the appropriate dot notation or
ITEM, KEY, or VALUE pseudocolumn names.
Examples:
The following examples refer to these simple tables containing small sets of integers or strings:
The following example shows a correlated subquery that finds all the values in one table that exist in another table.
For each value X from T1, the query checks if the Y column of T2 contains an identical value, and the EXISTS operator
returns TRUE or FALSE as appropriate in each case.
localhost:21000] > select x from t1 where exists (select y from t2 where t1.x = y);
+---+
| x |
+---+
| 2 |
| 4 |
| 6 |
+---+
An uncorrelated query is less interesting in this case. Because the subquery always returns TRUE, all rows from T1 are
returned. If the table contents where changed so that the subquery did not match any rows, none of the rows from
T1 would be returned.
[localhost:21000] > select x from t1 where exists (select y from t2 where y > 5);
+---+
| x |
+---+
| 1 |
| 2 |
| 3 |
| 4 |
| 5 |
| 6 |
+---+
The following example shows how an uncorrelated subquery can test for the existence of some condition within a
table. By using LIMIT 1 or an aggregate function, the query returns a single result or no result based on whether the
subquery matches any rows. Here, we know that T1 and T2 contain some even numbers, but T3 does not.
[localhost:21000] > select "contains an even number" from t1 where exists (select x from
t1 where x % 2 = 0) limit 1;
+---------------------------+
| 'contains an even number' |
+---------------------------+
| contains an even number |
+---------------------------+
[localhost:21000] > select "contains an even number" as assertion from t1 where exists
(select x from t1 where x % 2 = 0) limit 1;
+-------------------------+
| assertion |
+-------------------------+
| contains an even number |
+-------------------------+
[localhost:21000] > select "contains an even number" as assertion from t2 where exists
(select x from t2 where y % 2 = 0) limit 1;
ERROR: AnalysisException: couldn't resolve column reference: 'x'
[localhost:21000] > select "contains an even number" as assertion from t2 where exists
(select y from t2 where y % 2 = 0) limit 1;
+-------------------------+
| assertion |
+-------------------------+
| contains an even number |
+-------------------------+
[localhost:21000] > select "contains an even number" as assertion from t3 where exists
(select z from t3 where z % 2 = 0) limit 1;
[localhost:21000] >
The following example finds numbers in one table that are 1 greater than numbers from another table. The EXISTS
notation is simpler than an equivalent CROSS JOIN between the tables. (The example then also illustrates how the
same test could be performed using an IN operator.)
[localhost:21000] > select x from t1 where exists (select y from t2 where x = y + 1);
+---+
| x |
+---+
| 3 |
| 5 |
+---+
[localhost:21000] > select x from t1 where x in (select y + 1 from t2);
+---+
| x |
+---+
| 3 |
| 5 |
+---+
The following example finds values from one table that do not exist in another table.
[localhost:21000] > select x from t1 where not exists (select y from t2 where x = y);
+---+
| x |
+---+
| 1 |
| 3 |
| 5 |
+---+
The following example uses the NOT EXISTS operator to find all the leaf nodes in tree-structured data. This simplified
“tree of life” has multiple levels (class, order, family, and so on), with each item pointing upward through a PARENT
pointer. The example runs an outer query and a subquery on the same table, returning only those items whose ID
value is not referenced by the PARENT of any other item.
[localhost:21000] > create table tree (id int, parent int, name string);
[localhost:21000] > insert overwrite tree values
> (0, null, "animals"),
> (1, 0, "placentals"),
> (2, 0, "marsupials"),
> (3, 1, "bats"),
> (4, 1, "cats"),
> (5, 2, "kangaroos"),
> (6, 4, "lions"),
> (7, 4, "tigers"),
> (8, 5, "red kangaroo"),
> (9, 2, "wallabies");
[localhost:21000] > select name as "leaf node" from tree one
> where not exists (select parent from tree two where one.id =
two.parent);
+--------------+
| leaf node |
+--------------+
| bats |
| lions |
| tigers |
| red kangaroo |
| wallabies |
+--------------+
Related information:
Subqueries in Impala SELECT Statements on page 297
IN Operator
The IN operator compares an argument value to a set of values, and returns TRUE if the argument matches any value
in the set. The NOT IN operator reverses the comparison, and checks if the argument value is not part of a set of
values.
Syntax:
The left-hand expression and the set of comparison values must be of compatible types.
The left-hand expression must consist only of a single value, not a tuple. Although the left-hand expression is typically
a column name, it could also be some other value. For example, the WHERE clauses WHERE id IN (5) and WHERE
5 IN (id) produce the same results.
The set of values to check against can be specified as constants, function calls, column names, or other expressions in
the query text. The maximum number of expressions in the IN list is 9999. (The maximum number of elements of a
single expression is 10,000 items, and the IN operator itself counts as one.)
In Impala 2.0 and higher, the set of values can also be generated by a subquery. IN can evaluate an unlimited number
of results using a subquery.
Usage notes:
Any expression using the IN operator could be rewritten as a series of equality tests connected with OR, but the IN
syntax is often clearer, more concise, and easier for Impala to optimize. For example, with partitioned tables, queries
frequently use IN clauses to filter data by comparing the partition key columns to specific values.
NULL considerations:
If there really is a matching non-null value, IN returns TRUE:
If the searched value is not found in the comparison values, and the comparison values include NULL, the result is
NULL:
If the left-hand argument is NULL, IN always returns NULL. This rule applies even if the comparison values include
NULL.
Added in: Available in earlier Impala releases, but new capabilities were added in CDH 5.2.0 / Impala 2.0.0
Complex type considerations:
You cannot refer to a column with a complex data type (ARRAY, STRUCT, or MAP) directly in an operator. You can apply
operators only to scalar values that make up a complex type (the fields of a STRUCT, the items of an ARRAY, or the key
or value portion of a MAP) as part of a join query that refers to the scalar value using the appropriate dot notation or
ITEM, KEY, or VALUE pseudocolumn names.
The following example shows how to do an arithmetic operation using a numeric field of a STRUCT type that is an item
within an ARRAY column. Once the scalar numeric value R_NATIONKEY is extracted, it can be used in an arithmetic
expression, such as multiplying by 10:
Restrictions:
Correlated subqueries used in EXISTS and IN operators cannot include a LIMIT clause.
Examples:
Related information:
Subqueries in Impala SELECT Statements on page 297
IS NULL Operator
The IS NULL operator, and its converse the IS NOT NULL operator, test whether a specified value is NULL. Because
using NULL with any of the other comparison operators such as = or != also returns NULL rather than TRUE or FALSE,
you use a special-purpose comparison operator to check for this special condition.
Syntax:
expression IS NULL
expression IS NOT NULL
Usage notes:
In many cases, NULL values indicate some incorrect or incomplete processing during data ingestion or conversion. You
might check whether any values in a column are NULL, and if so take some followup action to fill them in.
With sparse data, often represented in “wide” tables, it is common for most values to be NULL with only an occasional
non-NULL value. In those cases, you can use the IS NOT NULL operator to identify the rows containing any data at
all for a particular column, regardless of the actual value.
With a well-designed database schema, effective use of NULL values and IS NULL and IS NOT NULL operators can
save having to design custom logic around special values such as 0, -1, 'N/A', empty string, and so on. NULL lets you
distinguish between a value that is known to be 0, false, or empty, and a truly unknown value.
Complex type considerations:
This operator is not applicable to complex type columns (STRUCT, ARRAY, or MAP). Using a complex type column with
IS NULL or IS NOT NULL causes a query error.
Examples:
LIKE Operator
A comparison operator for STRING data, with basic wildcard capability using _ to match a single character and % to
match multiple characters. The argument expression must match the entire string value. Typically, it is more efficient
to put any % wildcard match at the end of the string.
Syntax:
select distinct c_last_name from customer where c_last_name like 'Mc%' or c_last_name
like 'Mac%';
select count(c_last_name) from customer where c_last_name like 'M%';
select c_email_address from customer where c_email_address like '%.edu';
For a more general kind of search operator using regular expressions, see REGEXP Operator on page 202.
Logical Operators
Logical operators return a BOOLEAN value, based on a binary or unary logical operation between arguments that are
also Booleans. Typically, the argument expressions use comparison operators.
Syntax:
The following example shows how to do an arithmetic operation using a numeric field of a STRUCT type that is an item
within an ARRAY column. Once the scalar numeric value R_NATIONKEY is extracted, it can be used in an arithmetic
expression, such as multiplying by 10:
Examples:
+--------------------------------+
| (1 = 1) or ('hello' = 'world') |
+--------------------------------+
| true |
+--------------------------------+
[localhost:21000] > select (2 + 2 != 4) or (-1 > 0);
+--------------------------+
| (2 + 2 != 4) or (-1 > 0) |
+--------------------------+
| false |
+--------------------------+
REGEXP Operator
Tests whether a value matches a regular expression. Uses the POSIX regular expression syntax where ^ and $ match
the beginning and end of the string, . represents any single character, * represents a sequence of zero or more items,
+ represents a sequence of one or more items, ? produces a non-greedy match, and so on.
Syntax:
Usage notes:
The RLIKE operator is a synonym for REGEXP.
The | symbol is the alternation operator, typically used within () to match different sequences. The () groups do not
allow backreferences. To retrieve the part of a value matched within a () section, use the regexp_extract() built-in
function.
In Impala 1.3.1 and higher, the REGEXP and RLIKE operators now match a regular expression string that occurs
anywhere inside the target string, the same as if the regular expression was enclosed on each side by .*. See REGEXP
Operator on page 202 for examples. Previously, these operators only succeeded when the regular expression matched
the entire target string. This change improves compatibility with the regular expression support for popular database
systems. There is no change to the behavior of the regexp_extract() and regexp_replace() built-in functions.
In Impala 2.0 and later, the Impala regular expression syntax conforms to the POSIX Extended Regular Expression syntax
used by the Google RE2 library. For details, see the RE2 documentation. It has most idioms familiar from regular
expressions in Perl, Python, and so on, including .*? for non-greedy matches.
In Impala 2.0 and later, a change in the underlying regular expression library could cause changes in the way regular
expressions are interpreted by this function. Test any queries that use regular expressions and adjust the expression
patterns if necessary. See Incompatible Changes Introduced in Impala 2.0.0 / CDH 5.2.0 on page 628 for details.
Complex type considerations:
You cannot refer to a column with a complex data type (ARRAY, STRUCT, or MAP) directly in an operator. You can apply
operators only to scalar values that make up a complex type (the fields of a STRUCT, the items of an ARRAY, or the key
or value portion of a MAP) as part of a join query that refers to the scalar value using the appropriate dot notation or
ITEM, KEY, or VALUE pseudocolumn names.
Examples:
The following examples demonstrate the identical syntax for the REGEXP and RLIKE operators.
-- Find all customers whose first name starts with 'J', followed by 0 or more of any
character.
select c_first_name, c_last_name from customer where c_first_name regexp '^J.*';
select c_first_name, c_last_name from customer where c_first_name rlike '^J.*';
-- Find 'Macdonald', where the first 'a' is optional and the 'D' can be upper- or
lowercase.
-- The ^...$ are required, to match the start and end of the value.
select c_first_name, c_last_name from customer where c_last_name regexp '^Ma?c[Dd]onald$';
select c_first_name, c_last_name from customer where c_last_name rlike '^Ma?c[Dd]onald$';
-- Find names starting with 'S', then one or more vowels, then 'r', then any other
characters.
-- Matches 'Searcy', 'Sorenson', 'Sauer'.
select c_first_name, c_last_name from customer where c_last_name regexp '^S[aeiou]+r.*$';
select c_first_name, c_last_name from customer where c_last_name rlike '^S[aeiou]+r.*$';
-- Find names that end with 2 or more vowels: letters from the set a,e,i,o,u.
select c_first_name, c_last_name from customer where c_last_name regexp '.*[aeiou]{2,}$';
select c_first_name, c_last_name from customer where c_last_name rlike '.*[aeiou]{2,}$';
-- You can use letter ranges in the [] blocks, for example to find names starting with
A, B, or C.
select c_first_name, c_last_name from customer where c_last_name regexp '^[A-C].*';
select c_first_name, c_last_name from customer where c_last_name rlike '^[A-C].*';
-- If you are not sure about case, leading/trailing spaces, and so on, you can process
the
-- column using string functions first.
select c_first_name, c_last_name from customer where lower(trim(c_last_name)) regexp
'^de.*';
select c_first_name, c_last_name from customer where lower(trim(c_last_name)) rlike
'^de.*';
RLIKE Operator
Synonym for the REGEXP operator. See REGEXP Operator on page 202 for details.
Examples:
The following examples demonstrate the identical syntax for the REGEXP and RLIKE operators.
-- Find all customers whose first name starts with 'J', followed by 0 or more of any
character.
select c_first_name, c_last_name from customer where c_first_name regexp '^J.*';
select c_first_name, c_last_name from customer where c_first_name rlike '^J.*';
-- Find 'Macdonald', where the first 'a' is optional and the 'D' can be upper- or
lowercase.
-- The ^...$ are required, to match the start and end of the value.
-- Find names starting with 'S', then one or more vowels, then 'r', then any other
characters.
-- Matches 'Searcy', 'Sorenson', 'Sauer'.
select c_first_name, c_last_name from customer where c_last_name regexp '^S[aeiou]+r.*$';
select c_first_name, c_last_name from customer where c_last_name rlike '^S[aeiou]+r.*$';
-- Find names that end with 2 or more vowels: letters from the set a,e,i,o,u.
select c_first_name, c_last_name from customer where c_last_name regexp '.*[aeiou]{2,}$';
select c_first_name, c_last_name from customer where c_last_name rlike '.*[aeiou]{2,}$';
-- You can use letter ranges in the [] blocks, for example to find names starting with
A, B, or C.
select c_first_name, c_last_name from customer where c_last_name regexp '^[A-C].*';
select c_first_name, c_last_name from customer where c_last_name rlike '^[A-C].*';
-- If you are not sure about case, leading/trailing spaces, and so on, you can process
the
-- column using string functions first.
select c_first_name, c_last_name from customer where lower(trim(c_last_name)) regexp
'^de.*';
select c_first_name, c_last_name from customer where lower(trim(c_last_name)) rlike
'^de.*';
To set up an alias, add the AS alias clause immediately after any table, column, or expression name in the SELECT
list or FROM list of a query. The AS keyword is optional; you can also specify the alias immediately after the original
name.
-- The table aliases assigned in the FROM clause can be used both earlier
-- in the query (the SELECT list) and later (the WHERE clause).
SELECT one.name, two.address, three.phone
FROM census one, building_directory two, phonebook three
WHERE one.id = two.id and two.id = three.id;
-- The aliases c1 and c2 let the query handle columns with the same names from 2 joined
tables.
-- The aliases t1 and t2 let the query abbreviate references to long or cryptically
named tables.
SELECT t1.column_n AS c1, t2.column_n AS c2 FROM long_name_table AS t1,
very_long_name_table2 AS t2
WHERE c1 = c2;
SELECT t1.column_n c1, t2.column_n c2 FROM table1 t1, table2 t2
WHERE c1 = c2;
To use an alias name that matches one of the Impala reserved keywords (listed in Impala Reserved Words on page
585), surround the identifier with either single or double quotation marks, or `` characters (backticks).
Aliases follow the same rules as identifiers when it comes to case insensitivity. Aliases can be longer than identifiers
(up to the maximum length of a Java string) and can include additional characters such as spaces and dashes when
they are quoted using backtick characters.
Complex type considerations:
Queries involving the complex types (ARRAY, STRUCT, and MAP), typically make extensive use of table aliases. These
queries involve join clauses where the complex type column is treated as a joined table. To construct two-part or
three-part qualified names for the complex column elements in the FROM list, sometimes it is syntactically required to
construct a table alias for the complex column where it is referenced in the join clause. See Complex Types (CDH 5.5
or higher only) on page 156 for details and examples.
Alternatives:
Another way to define different names for the same tables or columns is to create views. See Overview of Impala Views
on page 210 for details.
Typically, you create a separate database for each project or application, to avoid naming conflicts between tables and
to make clear which tables are related to each other. The USE statement lets you switch between databases. Unqualified
references to tables, views, and functions refer to objects within the current database. You can also refer to objects
in other databases by using qualified names of the form dbname.object_name.
Each database is physically represented by a directory in HDFS. When you do not specify a LOCATION attribute, the
directory is located in the Impala data directory with the associated tables managed by Impala. When you do specify
a LOCATION attribute, any read and write operations for tables in that database are relative to the specified HDFS
directory.
There is a special database, named default, where you begin when you connect to Impala. Tables created in default
are physically located one level higher in HDFS than all the user-created databases.
Impala includes another predefined database, _impala_builtins, that serves as the location for the built-in functions.
To see the built-in functions, use a statement like the following:
Related statements:
CREATE DATABASE Statement on page 231, DROP DATABASE Statement on page 252, USE Statement on page 337, SHOW
DATABASES on page 326
[localhost:21000] > select name, population from country where continent = 'North America'
order by population desc limit 4;
[localhost:21000] > select upper(name), population from country where continent = 'North
America' order by population desc limit 4;
+-------------+------------+
| upper(name) | population |
+-------------+------------+
| USA | 320000000 |
| MEXICO | 122000000 |
| CANADA | 25000000 |
| GUATEMALA | 16000000 |
+-------------+------------+
Aggregate functions combine the results from multiple rows: either a single result for the entire table, or a separate
result for each group of rows. Aggregate functions are frequently used in combination with GROUP BY and HAVING
clauses in the SELECT statement. See Impala Aggregate Functions on page 403.
User-defined functions (UDFs) let you code your own logic. They can be either scalar or aggregate functions. UDFs let
you implement important business or scientific logic using high-performance code for Impala to automatically parallelize.
You can also use UDFs to implement convenience functions to simplify reporting or porting SQL from other database
systems. See Impala User-Defined Functions (UDFs) on page 447.
Each function is associated with a specific database. For example, if you issue a USE somedb statement followed by
CREATE FUNCTION somefunc, the new function is created in the somedb database, and you could refer to it through
the fully qualified name somedb.somefunc. You could then issue another USE statement and create a function with
the same name in a different database.
Impala built-in functions are associated with a special database named _impala_builtins, which lets you refer to
them from any database without qualifying the name.
Related statements: CREATE FUNCTION Statement on page 232, DROP FUNCTION Statement on page 254
Expected: IDENTIFIER
CAUSED BY: Exception: Syntax error
[localhost:21000] >
Important: Because the list of reserved words grows over time as new SQL syntax is added,
consider adopting coding conventions (especially for any automated scripts or in packaged
applications) to always quote all identifiers with backticks. Quoting all identifiers protects your
SQL from compatibility issues if new reserved words are added in later releases.
• Impala identifiers are always case-insensitive. That is, tables named t1 and T1 always refer to the same table,
regardless of quote characters. Internally, Impala always folds all specified table and column names to lowercase.
This is why the column headers in query output are always displayed in lowercase.
See Overview of Impala Aliases on page 204 for how to define shorter or easier-to-remember aliases if the original
names are long or cryptic identifiers. Aliases follow the same rules as identifiers when it comes to case insensitivity.
Aliases can be longer than identifiers (up to the maximum length of a Java string) and can include additional characters
such as spaces and dashes when they are quoted using backtick characters.
Another way to define different names for the same tables or columns is to create views. See Overview of Impala Views
on page 210 for details.
Internal Tables
The default kind of table produced by the CREATE TABLE statement is known as an internal table. (Its counterpart is
the external table, produced by the CREATE EXTERNAL TABLE syntax.)
• Impala creates a directory in HDFS to hold the data files.
• You can create data in internal tables by issuing INSERT or LOAD DATA statements.
• If you add or replace data using HDFS operations, issue the REFRESH command in impala-shell so that Impala
recognizes the changes in data files, block locations, and so on.
• When you issue a DROP TABLE statement, Impala physically removes all the data files from the directory.
• To see whether a table is internal or external, and its associated HDFS location, issue the statement DESCRIBE
FORMATTED table_name. The Table Type field displays MANAGED_TABLE for internal tables and
EXTERNAL_TABLE for external tables. The Location field displays the path of the table directory as an HDFS
URI.
• When you issue an ALTER TABLE statement to rename an internal table, all data files are moved into the new
HDFS directory for the table. The files are moved even if they were formerly in a directory outside the Impala data
directory, for example in an internal table with a LOCATION attribute pointing to an outside HDFS directory.
Examples:
You can switch a table from internal to external, or from external to internal, by using the ALTER TABLE statement:
Related information:
External Tables on page 209, CREATE TABLE Statement on page 235, DROP TABLE Statement on page 258, ALTER TABLE
Statement on page 215, DESCRIBE Statement on page 245
External Tables
The syntax CREATE EXTERNAL TABLE sets up an Impala table that points at existing data files, potentially in HDFS
locations outside the normal Impala data directories.. This operation saves the expense of importing the data into a
new table when you already have the data files in a known location in HDFS, in the desired file format.
• You can use Impala to query the data in this table.
• You can create data in external tables by issuing INSERT or LOAD DATA statements.
• If you add or replace data using HDFS operations, issue the REFRESH command in impala-shell so that Impala
recognizes the changes in data files, block locations, and so on.
• When you issue a DROP TABLE statement in Impala, that removes the connection that Impala has with the
associated data files, but does not physically remove the underlying data. You can continue to use the data files
with other Hadoop components and HDFS operations.
• To see whether a table is internal or external, and its associated HDFS location, issue the statement DESCRIBE
FORMATTED table_name. The Table Type field displays MANAGED_TABLE for internal tables and
EXTERNAL_TABLE for external tables. The Location field displays the path of the table directory as an HDFS
URI.
• When you issue an ALTER TABLE statement to rename an external table, all data files are left in their original
locations.
• You can point multiple external tables at the same HDFS directory by using the same LOCATION attribute for each
one. The tables could have different column definitions, as long as the number and types of columns are compatible
with the schema evolution considerations for the underlying file type. For example, for text data files, one table
might define a certain column as a STRING while another defines the same column as a BIGINT.
Examples:
You can switch a table from internal to external, or from external to internal, by using the ALTER TABLE statement:
Related information:
Internal Tables on page 208, CREATE TABLE Statement on page 235, DROP TABLE Statement on page 258, ALTER TABLE
Statement on page 215, DESCRIBE Statement on page 245
File Formats
Each table has an associated file format, which determines how Impala interprets the associated data files. See How
Impala Works with Hadoop File Formats on page 527 for details.
You set the file format during the CREATE TABLE statement, or change it later using the ALTER TABLE statement.
Partitioned tables can have a different file format for individual partitions, allowing you to change the file format used
in your ETL process for new data without going back and reconverting all the existing data in the same table.
Any INSERT statements produce new data files with the current file format of the table. For existing data files, changing
the file format of the table does not automatically do any data conversion. You must use TRUNCATE TABLE or INSERT
OVERWRITE to remove any previous data files that use the old file format. Then you use the LOAD DATA statement,
INSERT ... SELECT, or other mechanism to put data files of the correct format into the table.
The default file format, text, is the most flexible and easy to produce when you are just getting started with Impala.
The Parquet file format offers the highest query performance and uses compression to reduce storage requirements;
therefore, Cloudera recommends using Parquet for Impala tables with substantial amounts of data. Also, the complex
types (ARRAY, STRUCT, and MAP) available in CDH 5.5 / Impala 2.3 and higher are currently only supported with the
Parquet file type. Based on your existing ETL workflow, you might use other file formats such as Avro, possibly doing
a final conversion step to Parquet to take advantage of its performance for analytic queries.
• Reduce maintenance, by avoiding the duplication of complicated queries across multiple applications in multiple
languages:
create view v2 as select t1.c1, t1.c2, t2.c3 from t1 join t2 on (t1.id = t2.id);
-- This simple query is safer to embed in reporting applications than the longer query
above.
-- The view definition can remain stable even if the structure of the underlying tables
changes.
select c1, c2, c3 from v2;
• Build a new, more refined query on top of the original query by adding new clauses, select-list expressions, function
calls, and so on:
This technique lets you build up several more or less granular variations of the same query, and switch between
them when appropriate.
• Set up aliases with intuitive names for tables, columns, result sets from joins, and so on:
-- The original tables might have cryptic names inherited from a legacy system.
create view action_items as select rrptsk as assignee, treq as due_date, dmisc as notes
from vxy_t1_br;
-- You can leave original names for compatibility, build new applications using more
intuitive ones.
select assignee, due_date, notes from action_items;
• Swap tables with others that use different file formats, partitioning schemes, and so on without any downtime
for data copying or conversion:
-- After changing the view definition, queries will be faster due to partitioning,
-- binary format, and compression in the new table.
alter view report as select s from fast where x between 20 and 30;
select * from report;
• Avoid coding lengthy subqueries and repeating the same subquery text in many other queries.
• Set up fine-grained security where a user can query some columns from a table but not other columns. Because
CDH 5.5 / Impala 2.3 and higher support column-level authorization, this technique is no longer required. If you
formerly implemented column-level security through views, see
https://www.cloudera.com/documentation/enterprise/latest/topics/sg_hive_sql.html for details about the
column-level authorization feature.
The SQL statements that configure views are CREATE VIEW Statement on page 243, ALTER VIEW Statement on page
224, and DROP VIEW Statement on page 259. You can specify view names when querying data (SELECT Statement on
page 280) and copying data from one table to another (INSERT Statement on page 263). The WITH clause creates an
inline view, that only exists for the duration of a single query.
| Thomas | Zuniga | |
| Sarah | Zuniga | |
| Norma | Zuniga | |
| Lloyd | Zuniga | |
| Lisa | Zuniga | |
+--------------+-------------+---------+
Returned 5 row(s) in 0.48s
The previous example uses descending order for ORDERED_RESULTS because in the sample TPCD-H data, there are
some rows with empty strings for both C_FIRST_NAME and C_LAST_NAME, making the lowest-ordered names unuseful
in a sample query.
Usage notes:
To see the definition of a view, issue a DESCRIBE FORMATTED statement, which shows the query from the original
CREATE VIEW statement:
Prior to Impala 1.4.0, it was not possible to use the CREATE TABLE LIKE view_name syntax. In Impala 1.4.0 and
higher, you can create a table with the same column definitions as a view using the CREATE TABLE LIKE technique.
Although CREATE TABLE LIKE normally inherits the file format of the original table, a view has no underlying file
format, so CREATE TABLE LIKE view_name produces a text table by default. To specify a different file format,
include a STORED AS file_format clause at the end of the CREATE TABLE LIKE statement.
Complex type considerations:
For tables containing complex type columns (ARRAY, STRUCT, or MAP), you typically use join queries to refer to the
complex values. You can use views to hide the join notation, making such tables seem like traditional denormalized
tables, and making those tables queryable by business intelligence tools that do not have built-in support for those
complex types. See Accessing Complex Type Data in Flattened Form Using Views on page 176 for details.
The STRAIGHT_JOIN hint affects the join order of table references in the query block containing the hint. It does not
affect the join order of nested queries, such as views, inline views, or WHERE-clause subqueries. To use this hint for
performance tuning of complex queries, apply the hint to all query blocks that need a fixed join order.
Restrictions:
• You cannot insert into an Impala view. (In some database systems, this operation is allowed and inserts rows into
the base table.) You can use a view name on the right-hand side of an INSERT statement, in the SELECT part.
• If a view applies to a partitioned table, any partition pruning considers the clauses on both the original query and
any additional WHERE predicates in the query that refers to the view. Prior to Impala 1.4, only the WHERE clauses
on the original query from the CREATE VIEW statement were used for partition pruning.
• An ORDER BY clause without an additional LIMIT clause is ignored in any view definition. If you need to sort the
entire result set from a view, use an ORDER BY clause in the SELECT statement that queries the view. You can
still make a simple “top 10” report by combining the ORDER BY and LIMIT clauses in the same view definition:
Related statements: CREATE VIEW Statement on page 243, ALTER VIEW Statement on page 224, DROP VIEW Statement
on page 259
Note:
In the impala-shell interpreter, a semicolon at the end of each statement is required. Since the
semicolon is not actually part of the SQL syntax, we do not include it in the syntax definition of each
statement, but we do show it in examples intended to be run in impala-shell.
DDL Statements
DDL refers to “Data Definition Language”, a subset of SQL statements that change the structure of the database schema
in some way, typically by creating, deleting, or modifying schema objects such as databases, tables, and views. Most
Impala DDL statements start with the keywords CREATE, DROP, or ALTER.
The Impala DDL statements are:
• ALTER TABLE Statement on page 215
• ALTER VIEW Statement on page 224
• COMPUTE STATS Statement on page 226
• CREATE DATABASE Statement on page 231
• CREATE FUNCTION Statement on page 232
• CREATE ROLE Statement (CDH 5.2 or higher only) on page 235
• CREATE TABLE Statement on page 235
• CREATE VIEW Statement on page 243
• DROP DATABASE Statement on page 252
• DROP FUNCTION Statement on page 254
• DROP ROLE Statement (CDH 5.2 or higher only) on page 254
• DROP TABLE Statement on page 258
• DROP VIEW Statement on page 259
• GRANT Statement (CDH 5.2 or higher only) on page 263
• REVOKE Statement (CDH 5.2 or higher only) on page 279
After Impala executes a DDL command, information about available tables, columns, views, partitions, and so on is
automatically synchronized between all the Impala nodes in a cluster. (Prior to Impala 1.2, you had to issue a REFRESH
or INVALIDATE METADATA statement manually on the other nodes to make them aware of the changes.)
If the timing of metadata updates is significant, for example if you use round-robin scheduling where each query could
be issued through a different Impala node, you can enable the SYNC_DDL query option to make the DDL statement
wait until all nodes have been notified about the metadata changes.
See Using Impala to Query the Amazon S3 Filesystem on page 566 for details about how Impala DDL statements interact
with tables and partitions stored in the Amazon S3 filesystem.
Although the INSERT statement is officially classified as a DML (data manipulation language) statement, it also involves
metadata changes that must be broadcast to all Impala nodes, and so is also affected by the SYNC_DDL query option.
Because the SYNC_DDL query option makes each DDL operation take longer than normal, you might only enable it
before the last DDL operation in a sequence. For example, if you are running a script that issues multiple of DDL
operations to set up an entire new schema, add several new partitions, and so on, you might minimize the performance
overhead by enabling the query option only before the last CREATE, DROP, ALTER, or INSERT statement. The script
only finishes when all the relevant metadata changes are recognized by all the Impala nodes, so you could connect to
any node and issue queries through it.
The classification of DDL, DML, and other statements is not necessarily the same between Impala and Hive. Impala
organizes these statements in a way intended to be familiar to people familiar with relational databases or data
warehouse products. Statements that modify the metastore database, such as COMPUTE STATS, are classified as DDL.
Statements that only query the metastore database, such as SHOW or DESCRIBE, are put into a separate category of
utility statements.
Note: The query types shown in the Impala debug web user interface might not match exactly the
categories listed here. For example, currently the USE statement is shown as DDL in the debug web
UI. The query types shown in the debug web UI are subject to change, for improved consistency.
Related information:
The other major classifications of SQL statements are data manipulation language (see DML Statements on page 215)
and queries (see SELECT Statement on page 280).
DML Statements
DML refers to “Data Manipulation Language”, a subset of SQL statements that modify the data stored in tables. Because
Impala focuses on query performance and leverages the append-only nature of HDFS storage, currently Impala only
supports a small set of DML statements:
• INSERT Statement on page 263
• LOAD DATA Statement on page 274
INSERT in Impala is primarily optimized for inserting large volumes of data in a single statement, to make effective
use of the multi-megabyte HDFS blocks. This is the way in Impala to create new data files. If you intend to insert one
or a few rows at a time, such as using the INSERT ... VALUES syntax, that technique is much more efficient for
Impala tables stored in HBase. See Using Impala to Query HBase Tables on page 557 for details.
LOAD DATA moves existing data files into the directory for an Impala table, making them immediately available for
Impala queries. This is one way in Impala to work with data files produced by other Hadoop components. (CREATE
EXTERNAL TABLE is the other alternative; with external tables, you can query existing data files, while the files remain
in their original location.)
To simulate the effects of an UPDATE or DELETE statement in other database systems, typically you use INSERT or
CREATE TABLE AS SELECT to copy data from one table to another, filtering out or changing the appropriate rows
during the copy operation.
Although Impala currently does not have an UPDATE statement, you can achieve a similar result by using Impala tables
stored in HBase. When you insert a row into an HBase table, and the table already contains a row with the same value
for the key column, the older row is hidden, effectively the same as a single-row UPDATE.
Currently, Impala cannot perform DML operations for tables or partitions stored in the Amazon S3 filesystem. See
Using Impala to Query the Amazon S3 Filesystem on page 566 for details.
Related information:
The other major classifications of SQL statements are data definition language (see DDL Statements on page 214) and
queries (see SELECT Statement on page 280).
Syntax:
ALTER TABLE name { ADD [IF NOT EXISTS] | DROP [IF EXISTS] } PARTITION (partition_spec)
[PURGE]
ALTER TABLE name RECOVER PARTITIONS
In Impala 2.2 / CDH 5.4 and higher, the optional WITH REPLICATION clause for CREATE TABLE and ALTER TABLE
lets you specify a replication factor, the number of hosts on which to cache the same data blocks. When Impala
processes a cached data block, where the cache replication factor is greater than 1, Impala randomly selects a host
that has a cached copy of that data block. This optimization avoids excessive CPU usage on a single host when the
same cached data block is processed multiple times. Cloudera recommends specifying a value greater than or equal
to the HDFS block replication factor.
If you connect to different Impala nodes within an impala-shell session for load-balancing purposes, you can enable
the SYNC_DDL query option to make each DDL statement wait before returning, until the new or changed metadata
has been received by all the Impala nodes. See SYNC_DDL Query Option on page 321 for details.
The following sections show examples of the use cases for various ALTER TABLE clauses.
To rename a table (RENAME TO clause):
The RENAME TO clause lets you change the name of an existing table, and optionally which database it is located in.
For internal tables, this operation physically renames the directory within HDFS that contains the data files; the original
directory name no longer exists. By qualifying the table names with database names, you can use this technique to
move an internal table (and its associated data directory) from one database to another. For example:
The path you specify is the full HDFS path where the data files reside, or will be created. Impala does not create any
additional subdirectory named after the table. Impala does not move any data files to this new location or change any
data files that might already exist in that directory.
To set the location for a single partition, include the PARTITION clause. Specify all the same partitioning columns for
the table, with a constant value for each, to precisely identify the single partition affected by the statement:
Note: If you are creating a partition for the first time and specifying its location, for maximum efficiency,
use a single ALTER TABLE statement including both the ADD PARTITION and LOCATION clauses,
rather than separate statements with ADD PARTITION and SET LOCATION clauses.
To automatically detect new partition directories added through Hive or HDFS operations:
In CDH 5.5 / Impala 2.3 and higher, the RECOVER PARTITIONS clause scans a partitioned table to detect if any new
partition directories were added outside of Impala, such as by Hive ALTER TABLE statements or by hdfs dfs or
hadoop fs commands. The RECOVER PARTITIONS clause automatically recognizes any data files present in these
new directories, the same as the REFRESH statement does.
For example, here is a sequence of examples showing how you might create a partitioned table in Impala, create new
partitions through Hive, copy data files into the new partitions with the hdfs command, and have Impala recognize
the new partitions and new data:
In Impala, create the table, and a single partition for demonstration purposes:
In Hive, create some new partitions. In a real use case, you might create the partitions and populate them with data
as the final stages of an ETL pipeline.
For demonstration purposes, manually copy data (a single row) into these new partitions, using manual HDFS operations:
hive> quit;
In Impala, initially the partitions and data are not visible. Running ALTER TABLE with the RECOVER PARTITIONS
clause scans the table data directory to find any new partition directories, and the data files inside them:
The TBLPROPERTIES clause is primarily a way to associate arbitrary user-specified data items with a particular table.
The SERDEPROPERTIES clause sets up metadata defining how tables are read or written, needed in some cases by
Hive but not used extensively by Impala. You would use this clause primarily to change the delimiter in an existing text
table or partition, by setting the 'serialization.format' and 'field.delim' property values to the new delimiter
character:
Use the DESCRIBE FORMATTED statement to see the current values of these properties for an existing table. See
CREATE TABLE Statement on page 235 for more details about these clauses. See Setting the NUMROWS Value Manually
through ALTER TABLE on page 496 for an example of using table properties to fine-tune the performance-related table
statistics.
To reorganize columns for a table:
The column_spec is the same as in the CREATE TABLE statement: the column name, then its data type, then an optional
comment. You can add multiple columns at a time. The parentheses are required whether you add a single column or
multiple columns. When you replace columns, all the original column definitions are discarded. You might use this
technique if you receive a new set of data files with different data types or columns in a different order. (The data files
are retained, so if the new columns are incompatible with the old ones, use INSERT OVERWRITE or LOAD DATA
OVERWRITE to replace all the data before issuing any further queries.)
For example, here is how you might add columns to an existing table. The first ALTER TABLE adds two new columns,
and the second ALTER TABLE adds one new column. A single Impala query reads both the old and new data files,
containing different numbers of columns. For any columns not present in a particular data file, all the column values
are considered to be NULL.
You might use the CHANGE clause to rename a single column, or to treat an existing column as a different type than
before, such as to switch between treating a column as STRING and TIMESTAMP, or between INT and BIGINT. You
can only drop a single column at a time; to drop multiple columns, issue multiple ALTER TABLE statements, or define
the new set of columns with a single ALTER TABLE ... REPLACE COLUMNS statement.
The following examples show some safe operations to drop or change columns. Dropping the final column in a table
lets Impala ignore the data causing any disruption to existing data files. Changing the type of a column works if existing
data values can be safely converted to the new type. The type conversion rules depend on the file format of the
underlying table. For example, in a text table, the same value can be interpreted as a STRING or a numeric value, while
in a binary format such as Parquet, the rules are stricter and type conversions only work between certain sizes of
integers.
-- When the last column in the table is dropped, Impala ignores the
-- values that are no longer needed. (Dropping A1 but leaving A2
-- would cause problems, as we will see in a subsequent example.)
alter table optional_columns drop column a2;
alter table optional_columns drop column a1;
-- Once the type is changed, we can insert non-integer values into the X column
-- and treat that column as a string, for example by uppercasing or concatenating.
insert into int_to_string values ('three', 'trois');
select s, upper(x) from int_to_string;
+-------+----------+
| s | upper(x) |
+-------+----------+
| one | 1 |
| two | 2 |
| three | TROIS |
+-------+----------+
Remember that Impala does not actually do any conversion for the underlying data files as a result of ALTER TABLE
statements. If you use ALTER TABLE to create a table layout that does not agree with the contents of the underlying
files, you must replace the files yourself, such as using LOAD DATA to load a new set of data files, or INSERT OVERWRITE
to copy from another table and replace the original data.
The following example shows what happens if you delete the middle column from a Parquet table containing three
columns. The underlying data files still contain three columns of data. Because the columns are interpreted based on
their positions in the data file instead of the specific column names, a SELECT * query now reads the first and second
columns from the data file, potentially leading to unexpected results or conversion errors. For this reason, if you expect
to someday drop a column, declare it as the last column in the table, where its data can be ignored by queries after
the column is dropped. Or, re-run your ETL process and create new data files if you drop or change the type of a column
in a way that causes problems with existing data files.
-- Parquet table showing how dropping a column can produce unexpected results.
create table p1 (s1 string, s2 string, s3 string) stored as parquet;
-- Parquet table showing how dropping a column can produce conversion errors.
create table p2 (s1 string, x int, s3 string) stored as parquet;
To change the file format that Impala expects data to be in, for a table or partition:
Use an ALTER TABLE ... SET FILEFORMAT clause. You can include an optional PARTITION (col1=val1,
col2=val2, ... clause so that the file format is changed for a specific partition rather than the entire table.
Because this operation only changes the table metadata, you must do any conversion of existing data using regular
Hadoop techniques outside of Impala. Any new data created by the Impala INSERT statement will be in the new
format. You cannot specify the delimiter for Text files; the data files must be comma-delimited.
To set the file format for a single partition, include the PARTITION clause. Specify all the same partitioning columns
for the table, with a constant value for each, to precisely identify the single partition affected by the statement:
To add or drop partitions for a table, the table must already be partitioned (that is, created with a PARTITIONED BY
clause). The partition is a physical directory in HDFS, with a name that encodes a particular column value (the partition
key). The Impala INSERT statement already creates the partition if necessary, so the ALTER TABLE ... ADD
PARTITION is primarily useful for importing data by moving or copying existing data files into the HDFS directory
corresponding to a partition. (You can use the LOAD DATA statement to move files into the partition directory, or
ALTER TABLE ... PARTITION (...) SET LOCATION to point a partition at a directory that already contains data
files.
The DROP PARTITION clause is used to remove the HDFS directory and associated data files for a particular set of
partition key values; for example, if you always analyze the last 3 months worth of data, at the beginning of each month
you might drop the oldest partition that is no longer needed. Removing partitions reduces the amount of metadata
associated with the table and the complexity of calculating the optimal query plan, which can simplify and speed up
queries on partitioned tables, particularly join queries. Here is an example showing the ADD PARTITION and DROP
PARTITION clauses.
To avoid errors while adding or dropping partitions whose existence is not certain, add the optional IF [NOT] EXISTS
clause between the ADD or DROP keyword and the PARTITION keyword. That is, the entire clause becomes ADD IF
NOT EXISTS PARTITION or DROP IF EXISTS PARTITION. The following example shows how partitions can be
created automatically through INSERT statements, or manually through ALTER TABLE statements. The IF [NOT]
EXISTS clauses let the ALTER TABLE statements succeed even if a new requested partition already exists, or a partition
to be dropped does not exist.
Inserting 2 year values creates 2 partitions:
Without the IF NOT EXISTS clause, an attempt to add a new partition might fail:
The IF NOT EXISTS clause makes the statement succeed whether or not there was already a partition with the
specified key value:
Likewise, the IF EXISTS clause lets DROP PARTITION succeed whether or not the partition is already in the table:
The optional PURGE keyword, available in CDH 5.5 / Impala 2.3 and higher, is used with the DROP PARTITION clause
to remove associated HDFS data files immediately rather than going through the HDFS trashcan mechanism. Use this
keyword when dropping a partition if it is crucial to remove the data as quickly as possible to free up space, or if there
is a problem with the trashcan, such as the trashcan not being configured or being in a different HDFS encryption zone
than the data files.
-- If the data files for subsequent months were in a different file format,
-- you could set a different file format for the new partition as you create it.
alter table part_t add partition (month=3) set fileformat=parquet;
The value specified for a partition key can be an arbitrary constant expression, without any references to columns. For
example:
Note:
An alternative way to reorganize a table and its associated data files is to use CREATE TABLE to create
a variation of the original table, then use INSERT to copy the transformed or reordered data to the
new table. The advantage of ALTER TABLE is that it avoids making a duplicate copy of the data files,
allowing you to reorganize huge volumes of data in a space-efficient way using familiar Hadoop
techniques.
Syntax:
To see the definition of a view, issue a DESCRIBE FORMATTED statement, which shows the query from the original
CREATE VIEW statement:
Related information:
Overview of Impala Views on page 210, CREATE VIEW Statement on page 243, DROP VIEW Statement on page 259
The PARTITION clause is only allowed in combination with the INCREMENTAL clause. It is optional for COMPUTE
INCREMENTAL STATS, and required for DROP INCREMENTAL STATS. Whenever you specify partitions through the
PARTITION (partition_spec) clause in a COMPUTE INCREMENTAL STATS or DROP INCREMENTAL STATS
statement, you must include all the partitioning columns in the specification, and specify constant values for all the
partition key columns.
Usage notes:
Originally, Impala relied on users to run the Hive ANALYZE TABLE statement, but that method of gathering statistics
proved unreliable and difficult to use. The Impala COMPUTE STATS statement is built from the ground up to improve
the reliability and user-friendliness of this operation. COMPUTE STATS does not require any setup steps or special
configuration. You only run a single Impala COMPUTE STATS statement to gather both table and column statistics,
rather than separate Hive ANALYZE TABLE statements for each kind of statistics.
The COMPUTE INCREMENTAL STATS variation is a shortcut for partitioned tables that works on a subset of partitions
rather than the entire table. The incremental nature makes it suitable for large tables with many partitions, where a
full COMPUTE STATS operation takes too long to be practical each time a partition is added or dropped. See Overview
of Incremental Statistics on page 493 for full usage details.
COMPUTE INCREMENTAL STATS only applies to partitioned tables. If you use the INCREMENTAL clause for an
unpartitioned table, Impala automatically uses the original COMPUTE STATS statement. Such tables display false
under the Incremental stats column of the SHOW TABLE STATS output.
Note: Because many of the most performance-critical and resource-intensive operations rely on table
and column statistics to construct accurate and efficient plans, COMPUTE STATS is an important step
at the end of your ETL process. Run COMPUTE STATS on all tables as your first step during performance
tuning for slow queries, or troubleshooting for out-of-memory conditions:
• Accurate statistics help Impala construct an efficient query plan for join queries, improving
performance and reducing memory usage.
• Accurate statistics help Impala distribute the work effectively for insert operations into Parquet
tables, improving performance and reducing memory usage.
• Accurate statistics help Impala estimate the memory required for each query, which is important
when you use resource management features, such as admission control and the YARN resource
management framework. The statistics help Impala to achieve high concurrency, full utilization
of available memory, and avoid contention with workloads from other Hadoop components.
Currently, the statistics created by the COMPUTE STATS statement do not include information about complex type
columns. The column stats metrics for complex columns are always shown as -1. For queries involving complex type
columns, Impala uses heuristics to estimate the data distribution within such columns.
HBase considerations:
COMPUTE STATS works for HBase tables also. The statistics gathered for HBase tables are somewhat different than
for HDFS-backed tables, but that metadata is still used for optimization when HBase tables are involved in join queries.
Amazon S3 considerations:
COMPUTE STATS also works for tables where data resides in the Amazon Simple Storage Service (S3). See Using Impala
to Query the Amazon S3 Filesystem on page 566 for details.
Performance considerations:
The statistics collected by COMPUTE STATS are used to optimize join queries INSERT operations into Parquet tables,
and other resource-intensive kinds of SQL statements. See Table and Column Statistics on page 489 for details.
For large tables, the COMPUTE STATS statement itself might take a long time and you might need to tune its
performance. The COMPUTE STATS statement does not work with the EXPLAIN statement, or the SUMMARY command
in impala-shell. You can use the PROFILE statement in impala-shell to examine timing information for the
statement as a whole. If a basic COMPUTE STATS statement takes a long time for a partitioned table, consider switching
to the COMPUTE INCREMENTAL STATS syntax so that only newly added partitions are analyzed each time.
Examples:
This example shows two tables, T1 and T2, with a small number distinct values linked by a parent-child relationship
between T1.ID and T2.PARENT. T1 is tiny, while T2 has approximately 100K rows. Initially, the statistics includes
physical measurements such as the number of files, the total size, and size measurements for fixed-length columns
such as with the INT type. Unknown values are represented by -1. After running COMPUTE STATS for each table, much
more information is available through the SHOW STATS statements. If you were running a join query involving both
of these tables, you would need statistics for both tables to get the most effective optimization for the query.
The following example shows how to use the INCREMENTAL clause, available in Impala 2.1.0 and higher. The COMPUTE
INCREMENTAL STATS syntax lets you collect statistics for newly added or changed partitions, without rescanning the
entire table.
Note: Prior to Impala 1.4.0, COMPUTE STATS counted the number of NULL values in each column
and recorded that figure in the metastore database. Because Impala does not currently make use of
the NULL count during query planning, Impala 1.4.0 and higher speeds up the COMPUTE STATS
statement by skipping this NULL counting.
Internal details:
Behind the scenes, the COMPUTE STATS statement executes two statements: one to count the rows of each partition
in the table (or the entire table if unpartitioned) through the COUNT(*) function, and another to count the approximate
number of distinct values in each column through the NDV() function. You might see these queries in your monitoring
and diagnostic displays. The same factors that affect the performance, scalability, and execution of other queries (such
as parallel execution, memory usage, admission control, and timeouts) also apply to the queries run by the COMPUTE
STATS statement.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read permission for all affected
files in the source directory: all files in the case of an unpartitioned table or a partitioned table in the case of COMPUTE
STATS; or all the files in partitions without incremental stats in the case of COMPUTE INCREMENTAL STATS. It must
also have read and execute permissions for all relevant directories holding the data files. (Essentially, COMPUTE STATS
requires the same permissions as the underlying SELECT queries it runs against the table.)
Related information:
DROP STATS Statement on page 255, SHOW TABLE STATS Statement on page 330, SHOW COLUMN STATS Statement
on page 332, Table and Column Statistics on page 489
Impala includes another predefined database, _impala_builtins, that serves as the location for the built-in functions.
To see the built-in functions, use a statement like the following:
After creating a database, your impala-shell session or another impala-shell connected to the same node can
immediately access that database. To access the database through the Impala daemon on a different node, issue the
INVALIDATE METADATA statement first while connected to that other node.
Setting the LOCATION attribute for a new database is a way to work with sets of files in an HDFS directory structure
outside the default Impala data directory, as opposed to setting the LOCATION attribute for each individual table.
If you connect to different Impala nodes within an impala-shell session for load-balancing purposes, you can enable
the SYNC_DDL query option to make each DDL statement wait before returning, until the new or changed metadata
has been received by all the Impala nodes. See SYNC_DDL Query Option on page 321 for details.
Hive considerations:
When you create a database in Impala, the database can also be used by Hive. When you create a database in Hive,
issue an INVALIDATE METADATA statement in Impala to make Impala permanently aware of the new database.
The SHOW DATABASES statement lists all databases, or the databases whose name matches a wildcard pattern.
Cancellation: Cannot be cancelled.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have write permission for the
parent HDFS directory under which the database is located.
Examples:
-- Before dropping a database, first drop all the tables inside it,
-- or in CDH 5.5 and higher use the CASCADE clause.
drop database temp;
ERROR: ImpalaRuntimeException: Error making 'dropDatabase' RPC to Hive Metastore:
CAUSED BY: InvalidOperationException: Database temp is not empty
show tables in temp;
+------+
| name |
+------+
| t3 |
+------+
Related information:
Overview of Impala Databases on page 205, DROP DATABASE Statement on page 252, USE Statement on page 337, SHOW
DATABASES on page 326, Overview of Impala Tables on page 208
Note:
Variable-length argument lists are supported for C++ UDFs, but currently not for Java UDFs.
See Variable-Length Argument Lists on page 452 for how to code a C++ UDF to accept variable-length argument lists.
Scalar and aggregate functions:
The simplest kind of user-defined function returns a single scalar value each time it is called, typically once for each
row in the result set. This general kind of function is what is usually meant by UDF. User-defined aggregate functions
(UDAs) are a specialized kind of UDF that produce a single value based on the contents of multiple rows. You usually
use UDAs in combination with a GROUP BY clause to condense a large result set into a smaller one, or even a single
row summarizing column values across an entire table.
You create UDAs by using the CREATE AGGREGATE FUNCTION syntax. The clauses INIT_FN, UPDATE_FN, MERGE_FN,
SERIALIZE_FN, FINALIZE_FN, and INTERMEDIATE only apply when you create a UDA rather than a scalar UDF.
The *_FN clauses specify functions to call at different phases of function processing.
• Initialize: The function you specify with the INIT_FN clause does any initial setup, such as initializing member
variables in internal data structures. This function is often a stub for simple UDAs. You can omit this clause and a
default (no-op) function will be used.
• Update: The function you specify with the UPDATE_FN clause is called once for each row in the original result set,
that is, before any GROUP BY clause is applied. A separate instance of the function is called for each different
value returned by the GROUP BY clause. The final argument passed to this function is a pointer, to which you
write an updated value based on its original value and the value of the first argument.
• Merge: The function you specify with the MERGE_FN clause is called an arbitrary number of times, to combine
intermediate values produced by different nodes or different threads as Impala reads and processes data files in
parallel. The final argument passed to this function is a pointer, to which you write an updated value based on its
original value and the value of the first argument.
• Serialize: The function you specify with the SERIALIZE_FN clause frees memory allocated to intermediate results.
It is required if any memory was allocated by the Allocate function in the Init, Update, or Merge functions, or if
the intermediate type contains any pointers. See the UDA code samples for details.
• Finalize: The function you specify with the FINALIZE_FN clause does any required teardown for resources acquired
by your UDF, such as freeing memory, closing file handles if you explicitly opened any files, and so on. This function
is often a stub for simple UDAs. You can omit this clause and a default (no-op) function will be used. It is required
in UDAs where the final return type is different than the intermediate type. or if any memory was allocated by
the Allocate function in the Init, Update, or Merge functions. See the UDA code samples for details.
If you use a consistent naming convention for each of the underlying functions, Impala can automatically determine
the names based on the first such clause, so the others are optional.
For end-to-end examples of UDAs, see Impala User-Defined Functions (UDFs) on page 447.
Complex type considerations:
Currently, Impala UDFs cannot accept arguments or return values of the Impala complex types (STRUCT, ARRAY, or
MAP).
Usage notes:
• You can write Impala UDFs in either C++ or Java. C++ UDFs are new to Impala, and are the recommended format
for high performance utilizing native code. Java-based UDFs are compatible between Impala and Hive, and are
most suited to reusing existing Hive UDFs. (Impala can run Java-based Hive UDFs but not Hive UDAs.)
• The body of the UDF is represented by a .so or .jar file, which you store in HDFS and the CREATE FUNCTION
statement distributes to each Impala node.
• Impala calls the underlying code during SQL statement evaluation, as many times as needed to process all the
rows from the result set. All UDFs are assumed to be deterministic, that is, to always return the same result when
passed the same argument values. Impala might or might not skip some invocations of a UDF if the result value
is already known from a previous call. Therefore, do not rely on the UDF being called a specific number of times,
and do not return different result values based on some external factor such as the current time, a random number
function, or an external data source that could be updated while an Impala query is in progress.
• The names of the function arguments in the UDF are not significant, only their number, positions, and data types.
• You can overload the same function name by creating multiple versions of the function, each with a different
argument signature. For security reasons, you cannot make a UDF with the same name as any built-in function.
• In the UDF code, you represent the function return result as a struct. This struct contains 2 fields. The first
field is a boolean representing whether the value is NULL or not. (When this field is true, the return value is
interpreted as NULL.) The second field is the same type as the specified function return type, and holds the return
value when the function returns something other than NULL.
• In the UDF code, you represent the function arguments as an initial pointer to a UDF context structure, followed
by references to zero or more structs, corresponding to each of the arguments. Each struct has the same 2
fields as with the return value, a boolean field representing whether the argument is NULL, and a field of the
appropriate type holding any non-NULL argument value.
• For sample code and build instructions for UDFs, see the sample UDFs in the Impala github repo.
• Because the file representing the body of the UDF is stored in HDFS, it is automatically available to all the Impala
nodes. You do not need to manually copy any UDF-related files between servers.
• Because Impala currently does not have any ALTER FUNCTION statement, if you need to rename a function,
move it to a different database, or change its signature or other properties, issue a DROP FUNCTION statement
for the original function followed by a CREATE FUNCTION with the desired properties.
• Because each UDF is associated with a particular database, either issue a USE statement before doing any CREATE
FUNCTION statements, or specify the name of the function as db_name.function_name.
If you connect to different Impala nodes within an impala-shell session for load-balancing purposes, you can enable
the SYNC_DDL query option to make each DDL statement wait before returning, until the new or changed metadata
has been received by all the Impala nodes. See SYNC_DDL Query Option on page 321 for details.
Compatibility:
Impala can run UDFs that were created through Hive, as long as they refer to Impala-compatible data types (not
composite or nested column types). Hive can run Java-based UDFs that were created through Impala, but not Impala
UDFs written in C++.
Restrictions:
The Hive current_user() function cannot be called from a Java UDF through Impala.
Currently, Impala UDFs and UDAs are not persisted in the metastore database. Information about these functions is
held in the memory of the catalogd daemon. You must reload them by running the CREATE FUNCTION statements
again each time you restart the catalogd daemon.
Cancellation: Cannot be cancelled.
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Related information:
Impala User-Defined Functions (UDFs) on page 447 for more background information, usage instructions, and examples
for Impala UDFs; DROP FUNCTION Statement on page 254
Syntax:
Required privileges:
Only administrative users (those with ALL privileges on the server, defined in the Sentry policy file) can use this
statement.
Compatibility:
Impala makes use of any roles and privileges specified by the GRANT and REVOKE statements in Hive, and Hive makes
use of any roles and privileges specified by the GRANT and REVOKE statements in Impala. The Impala GRANT and REVOKE
statements for privileges do not require the ROLE keyword to be repeated before each role name, unlike the equivalent
Hive statements.
Cancellation: Cannot be cancelled.
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Related information:
Enabling Sentry Authorization for Impala on page 95, GRANT Statement (CDH 5.2 or higher only) on page 263, REVOKE
Statement (CDH 5.2 or higher only) on page 279, DROP ROLE Statement (CDH 5.2 or higher only) on page 254, SHOW
Statement on page 322
primitive_type:
TINYINT
| SMALLINT
| INT
| BIGINT
| BOOLEAN
| FLOAT
| DOUBLE
| DECIMAL
| STRING
| CHAR
| VARCHAR
| TIMESTAMP
complex_type:
struct_type
| array_type
| map_type
row_format:
DELIMITED [FIELDS TERMINATED BY 'char' [ESCAPED BY 'char']]
[LINES TERMINATED BY 'char']
file_format:
PARQUET
| TEXTFILE
| AVRO
| SEQUENCEFILE
| RCFILE
ctas_file_format:
PARQUET
| TEXTFILE
Typically, for an external table you include a LOCATION clause to specify the path to the HDFS directory where Impala
reads and writes files for the table. For example, if your data pipeline produces Parquet files in the HDFS directory
/user/etl/destination, you might create an external table as follows:
Although the EXTERNAL and LOCATION clauses are often specified together, LOCATION is optional for external tables,
and you can also specify LOCATION for internal tables. The difference is all about whether Impala “takes control” of
the underlying data files and moves them when you rename the table, or deletes them when you drop the table. For
more about internal and external tables and how they interact with the LOCATION attribute, see Overview of Impala
Tables on page 208.
Partitioned tables (PARTITIONED BY clause):
The PARTITIONED BY clause divides the data files based on the values from one or more specified columns. Impala
queries can use the partition metadata to minimize the amount of data that is read from disk or transmitted across
the network, particularly during join queries. For details about partitioning, see Partitioning for Impala Tables on page
522.
Specifying file format (STORED AS and ROW FORMAT clauses):
The STORED AS clause identifies the format of the underlying data files. Currently, Impala can query more types of
file formats than it can create or insert into. Use Hive to perform any create or data load operations that are not
currently available in Impala. For example, Impala can create an Avro, SequenceFile, or RCFile table but cannot insert
data into it. There are also Impala-specific procedures for using compression with each kind of file format. For details
about working with data files of various formats, see How Impala Works with Hadoop File Formats on page 527.
Note: In Impala 1.4.0 and higher, Impala can create Avro tables, which formerly required doing the
CREATE TABLE statement in Hive. See Using the Avro File Format with Impala Tables on page 547 for
details and examples.
By default (when no STORED AS clause is specified), data files in Impala tables are created as text files with Ctrl-A (hex
01) characters as the delimiter. Specify the ROW FORMAT DELIMITED clause to produce or ingest data files that use
a different delimiter character such as tab or |, or a different line end character such as carriage return or newline.
When specifying delimiter and line end characters with the FIELDS TERMINATED BY and LINES TERMINATED BY
clauses, use '\t' for tab, '\n' for newline or linefeed, '\r' for carriage return, and \0 for ASCII nul (hex 00). For
more examples of text tables, see Using Text Data Files with Impala Tables on page 528.
The ESCAPED BY clause applies both to text files that you create through an INSERT statement to an Impala TEXTFILE
table, and to existing data files that you put into an Impala table directory. (You can ingest existing data files either by
creating the table with CREATE EXTERNAL TABLE ... LOCATION, the LOAD DATA statement, or through an HDFS
operation such as hdfs dfs -put file hdfs_path.) Choose an escape character that is not used anywhere else
in the file, and put it in front of each instance of the delimiter character that occurs within a field value. Surrounding
field values with quotation marks does not help Impala to parse fields with embedded delimiter characters; the
quotation marks are considered to be part of the column value. If you want to use \ as the escape character, specify
the clause in impala-shell as ESCAPED BY '\\'.
Note: The CREATE TABLE clauses FIELDS TERMINATED BY, ESCAPED BY, and LINES TERMINATED
BY have special rules for the string literal used for their argument, because they all require a single
character. You can use a regular character surrounded by single or double quotation marks, an octal
sequence such as '\054' (representing a comma), or an integer in the range '-127'..'128' (with
quotation marks but no backslash), which is interpreted as a single-byte ASCII character. Negative
values are subtracted from 256; for example, FIELDS TERMINATED BY '-2' sets the field delimiter
to ASCII code 254, the “Icelandic Thorn” character used as a delimiter by some data formats.
To create an empty table with the same columns, comments, and other attributes as another table, use the following
variation. The CREATE TABLE ... LIKE form allows a restricted set of clauses, currently only the LOCATION,
COMMENT, and STORED AS clauses.
Note:
To clone the structure of a table and transfer data into it in a single operation, use the CREATE TABLE
AS SELECT syntax described in the next subsection.
When you clone the structure of an existing table using the CREATE TABLE ... LIKE syntax, the new table keeps
the same file format as the original one, so you only need to specify the STORED AS clause if you want to use a different
file format, or when specifying a view as the original table. (Creating a table “like” a view produces a text table by
default.)
Although normally Impala cannot create an HBase table directly, Impala can clone the structure of an existing HBase
table with the CREATE TABLE ... LIKE syntax, preserving the file format and metadata from the original table.
There are some exceptions to the ability to use CREATE TABLE ... LIKE with an Avro table. For example, you
cannot use this technique for an Avro table that is specified with an Avro schema but no columns. When in doubt,
check if a CREATE TABLE ... LIKE operation works in Hive; if not, it typically will not work in Impala either.
If the original table is partitioned, the new table inherits the same partition key columns. Because the new table is
initially empty, it does not inherit the actual partitions that exist in the original one. To create partitions in the new
table, insert data or issue ALTER TABLE ... ADD PARTITION statements.
Prior to Impala 1.4.0, it was not possible to use the CREATE TABLE LIKE view_name syntax. In Impala 1.4.0 and
higher, you can create a table with the same column definitions as a view using the CREATE TABLE LIKE technique.
Although CREATE TABLE LIKE normally inherits the file format of the original table, a view has no underlying file
format, so CREATE TABLE LIKE view_name produces a text table by default. To specify a different file format,
include a STORED AS file_format clause at the end of the CREATE TABLE LIKE statement.
Because CREATE TABLE ... LIKE only manipulates table metadata, not the physical data of the table, issue INSERT
INTO TABLE statements afterward to copy any data from the original table into the new one, optionally converting
the data to a new file format. (For some file formats, Impala can do a CREATE TABLE ... LIKE to create the table,
but Impala cannot insert data in that file format; in these cases, you must load the data in Hive. See How Impala Works
with Hadoop File Formats on page 527 for details.)
CREATE TABLE AS SELECT:
The CREATE TABLE AS SELECT syntax is a shorthand notation to create a table based on column definitions from
another table, and copy data from the source table to the destination table without issuing any separate INSERT
statement. This idiom is so popular that it has its own acronym, “CTAS”.
The following examples show how to copy data from a source table T1 to a variety of destinations tables, applying
various transformations to the table properties, table layout, or the data itself as part of the operation:
-- Clone all the columns and data from one table to another.
CREATE TABLE clone_of_t1 AS SELECT * FROM t1;
+-------------------+
| summary |
+-------------------+
| Inserted 3 row(s) |
+-------------------+
-- Clone the columns and data, and convert the data to a different file format.
CREATE TABLE parquet_version_of_t1 STORED AS PARQUET AS SELECT * FROM t1;
+-------------------+
| summary |
+-------------------+
| Inserted 3 row(s) |
+-------------------+
-- Same idea as CREATE TABLE LIKE: clone table layout but do not copy any data.
CREATE TABLE empty_clone_of_t1 AS SELECT * FROM t1 WHERE 1=0;
+-------------------+
| summary |
+-------------------+
| Inserted 0 row(s) |
+-------------------+
See SELECT Statement on page 280 for details about query syntax for the SELECT portion of a CREATE TABLE AS
SELECT statement.
The newly created table inherits the column names that you select from the original table, which you can override by
specifying column aliases in the query. Any column or table comments from the original table are not carried over to
the new table.
Sorting considerations: Although you can specify an ORDER BY clause in an INSERT ... SELECT statement, any
ORDER BY clause is ignored and the results are not necessarily sorted. An INSERT ... SELECT operation potentially
creates many different data files, prepared on different data nodes, and therefore the notion of the data being stored
in sorted order is impractical.
Note: When using the STORED AS clause with a CREATE TABLE AS SELECT statement, the
destination table must be a file format that Impala can write to: currently, text or Parquet. You cannot
specify an Avro, SequenceFile, or RCFile table as the destination table for a CTAS operation.
For example, the following statements show how you can clone all the data in a table, or a subset of the columns
and/or rows, or reorder columns, rename them, or construct them out of expressions:
As part of a CTAS operation, you can convert the data to any file format that Impala can write (currently, TEXTFILE
and PARQUET). You cannot specify the lower-level properties of a text table, such as the delimiter. Although you can
use a partitioned table as the source and copy data from it, you cannot specify any partitioning clauses for the new
table.
Sorting considerations: Although you can specify an ORDER BY clause in an INSERT ... SELECT statement, any
ORDER BY clause is ignored and the results are not necessarily sorted. An INSERT ... SELECT operation potentially
creates many different data files, prepared on different data nodes, and therefore the notion of the data being stored
in sorted order is impractical.
CREATE TABLE LIKE PARQUET:
The variation CREATE TABLE ... LIKE PARQUET 'hdfs_path_of_parquet_file' lets you skip the column
definitions of the CREATE TABLE statement. The column names and data types are automatically configured based
on the organization of the specified Parquet data file, which must already reside in HDFS. You can use a data file located
outside the Impala database directories, or a file from an existing Impala Parquet table; either way, Impala only uses
the column definitions from the file and does not use the HDFS location for the LOCATION attribute of the new table.
(Although you can also specify the enclosing directory with the LOCATION attribute, to both use the same schema as
the data file and point the Impala table at the associated directory for querying.)
The following considerations apply when you use the CREATE TABLE LIKE PARQUET technique:
• Any column comments from the original table are not preserved in the new table. Each column in the new table
has a comment stating the low-level Parquet field type used to deduce the appropriate SQL column type.
• If you use a data file from a partitioned Impala table, any partition key columns from the original table are left
out of the new table, because they are represented in HDFS directory names rather than stored in the data file.
To preserve the partition information, repeat the same PARTITION clause as in the original CREATE TABLE
statement.
• The file format of the new table defaults to text, as with other kinds of CREATE TABLE statements. To make the
new table also use Parquet format, include the clause STORED AS PARQUET in the CREATE TABLE LIKE
PARQUET statement.
• If the Parquet data file comes from an existing Impala table, currently, any TINYINT or SMALLINT columns are
turned into INT columns in the new table. Internally, Parquet stores such values as 32-bit integers.
• When the destination table uses the Parquet file format, the CREATE TABLE AS SELECT and INSERT ...
SELECT statements always create at least one data file, even if the SELECT part of the statement does not match
any rows. You can use such an empty Parquet data file as a template for subsequent CREATE TABLE LIKE
PARQUET statements.
For more details about creating Parquet tables, and examples of the CREATE TABLE LIKE PARQUET syntax, see
Using the Parquet File Format with Impala Tables on page 535.
Visibility and Metadata (TBLPROPERTIES and WITH SERDEPROPERTIES clauses):
You can associate arbitrary items of metadata with a table by specifying the TBLPROPERTIES clause. This clause takes
a comma-separated list of key-value pairs and stores those items in the metastore database. You can also change the
table properties later with an ALTER TABLE statement. You can observe the table properties for different delimiter
and escape characters using the DESCRIBE FORMATTED command, and change those settings for an existing table
with ALTER TABLE ... SET TBLPROPERTIES.
You can also associate SerDes properties with the table by specifying key-value pairs through the WITH
SERDEPROPERTIES clause. This metadata is not used by Impala, which has its own built-in serializer and deserializer
for the file formats it supports. Particular property values might be needed for Hive compatibility with certain variations
of file formats, particularly Avro.
Some DDL operations that interact with other Hadoop components require specifying particular values in the
SERDEPROPERTIES or TBLPROPERTIES fields, such as creating an Avro table or an HBase table. (You typically create
HBase tables in Hive, because they require additional clauses not currently available in Impala.)
To see the column definitions and column comments for an existing table, for example before issuing a CREATE TABLE
... LIKE or a CREATE TABLE ... AS SELECT statement, issue the statement DESCRIBE table_name. To see
even more detail, such as the location of data files and the values for clauses such as ROW FORMAT and STORED AS,
issue the statement DESCRIBE FORMATTED table_name. DESCRIBE FORMATTED is also needed to see any overall
table comment (as opposed to individual column comments).
After creating a table, your impala-shell session or another impala-shell connected to the same node can
immediately query that table. There might be a brief interval (one statestore heartbeat) before the table can be queried
through a different Impala node. To make the CREATE TABLE statement return only when the table is recognized by
all Impala nodes in the cluster, enable the SYNC_DDL query option.
HDFS caching (CACHED IN clause):
If you specify the CACHED IN clause, any existing or future data files in the table directory or the partition subdirectories
are designated to be loaded into memory with the HDFS caching mechanism. See Using HDFS Caching with Impala
(CDH 5.1 or higher only) on page 501 for details about using the HDFS caching feature.
In Impala 2.2 / CDH 5.4 and higher, the optional WITH REPLICATION clause for CREATE TABLE and ALTER TABLE
lets you specify a replication factor, the number of hosts on which to cache the same data blocks. When Impala
processes a cached data block, where the cache replication factor is greater than 1, Impala randomly selects a host
that has a cached copy of that data block. This optimization avoids excessive CPU usage on a single host when the
same cached data block is processed multiple times. Cloudera recommends specifying a value greater than or equal
to the HDFS block replication factor.
Column order:
If you intend to use the table to hold data files produced by some external source, specify the columns in the same
order as they appear in the data files.
If you intend to insert or copy data into the table through Impala, or if you have control over the way externally produced
data files are arranged, use your judgment to specify columns in the most convenient order:
• If certain columns are often NULL, specify those columns last. You might produce data files that omit these trailing
columns entirely. Impala automatically fills in the NULL values if so.
• If an unpartitioned table will be used as the source for an INSERT ... SELECT operation into a partitioned
table, specify last in the unpartitioned table any columns that correspond to partition key columns in the partitioned
table, and in the same order as the partition key columns are declared in the partitioned table. This technique
lets you use INSERT ... SELECT * when copying data to the partitioned table, rather than specifying each
column name individually.
• If you specify columns in an order that you later discover is suboptimal, you can sometimes work around the
problem without recreating the table. You can create a view that selects columns from the original table in a
permuted order, then do a SELECT * from the view. When inserting data into a table, you can specify a permuted
order for the inserted columns to match the order in the destination table.
Hive considerations:
Impala queries can make use of metadata about the table and columns, such as the number of rows in a table or the
number of different values in a column. Prior to Impala 1.2.2, to create this metadata, you issued the ANALYZE TABLE
statement in Hive to gather this information, after creating the table and loading representative data into it. In Impala
1.2.2 and higher, the COMPUTE STATS statement produces these statistics within Impala, without needing to use Hive
at all.
HBase considerations:
Note:
The Impala CREATE TABLE statement cannot create an HBase table, because it currently does not
support the STORED BY clause needed for HBase tables. Create such tables in Hive, then query them
through Impala. For information on using Impala with HBase tables, see Using Impala to Query HBase
Tables on page 557.
Amazon S3 considerations:
To create a table where the data resides in the Amazon Simple Storage Service (S3), specify a s3a:// prefix LOCATION
attribute pointing to the data files in S3. You can use this special LOCATION syntax when creating an empty table, but
not as part of a CREATE TABLE AS SELECT statement. See Using Impala to Query the Amazon S3 Filesystem on page
566 for details.
Sorting considerations: Although you can specify an ORDER BY clause in an INSERT ... SELECT statement, any
ORDER BY clause is ignored and the results are not necessarily sorted. An INSERT ... SELECT operation potentially
creates many different data files, prepared on different data nodes, and therefore the notion of the data being stored
in sorted order is impractical.
HDFS considerations:
The CREATE TABLE statement for an internal table creates a directory in HDFS. The CREATE EXTERNAL TABLE
statement associates the table with an existing HDFS directory, and does not create any new directory in HDFS. To
locate the HDFS data directory for a table, issue a DESCRIBE FORMATTED table statement. To examine the contents
of that HDFS directory, use an OS command such as hdfs dfs -ls hdfs://path, either from the OS command
line or through the shell or ! commands in impala-shell.
The CREATE TABLE AS SELECT syntax creates data files under the table data directory to hold any data copied by
the INSERT portion of the statement. (Even if no data is copied, Impala might create one or more empty data files.)
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have both execute and write
permission for the database directory where the table is being created.
Security considerations:
If these statements in your environment contain sensitive literal values such as credit card numbers or tax identifiers,
Impala can redact this sensitive information when displaying the statements in log files and other administrative
contexts. See http://www.cloudera.com/content/cloudera/en/documentation/core/latest/topics/sg_redaction.html
for details.
Cancellation: Certain multi-stage statements (CREATE TABLE AS SELECT and COMPUTE STATS) can be cancelled
during some stages, when running INSERT or SELECT operations internally. To cancel this statement, use Ctrl-C from
the impala-shell interpreter, the Cancel button from the Watch page in Hue, Actions > Cancel from the Queries
list in Cloudera Manager, or Cancel from the list of in-flight queries (for a particular node) on the Queries tab in the
Impala web UI (port 25000).
Related information:
Overview of Impala Tables on page 208, ALTER TABLE Statement on page 215, DROP TABLE Statement on page 258,
Partitioning for Impala Tables on page 522, Internal Tables on page 208, External Tables on page 209, COMPUTE STATS
Statement on page 226, SYNC_DDL Query Option on page 321, SHOW TABLES Statement on page 327, SHOW CREATE
TABLE Statement on page 329, DESCRIBE Statement on page 245
• To turn even the most lengthy and complicated SQL query into a one-liner. You can issue simple queries against
the view from applications, scripts, or interactive queries in impala-shell. For example:
The more complicated and hard-to-read the original query, the more benefit there is to simplifying the query
using a view.
• To hide the underlying table and column names, to minimize maintenance problems if those names change. In
that case, you re-create the view using the new names, and all queries that use the view rather than the underlying
tables keep running with no changes.
• To experiment with optimization techniques and make the optimized queries available to all applications. For
example, if you find a combination of WHERE conditions, join order, join hints, and so on that works the best for
a class of queries, you can establish a view that incorporates the best-performing techniques. Applications can
then make relatively simple queries against the view, without repeating the complicated and optimized logic over
and over. If you later find a better way to optimize the original query, when you re-create the view, all the
applications immediately take advantage of the optimized base query.
• To simplify a whole class of related queries, especially complicated queries involving joins between multiple tables,
complicated expressions in the column list, and other SQL syntax that makes the query difficult to understand
and debug. For example, you might create a view that joins several tables, filters using several WHERE conditions,
and selects several columns from the result set. Applications might issue queries against this view that only vary
in their LIMIT, ORDER BY, and similar simple clauses.
For queries that require repeating complicated clauses over and over again, for example in the select list, ORDER BY,
and GROUP BY clauses, you can use the WITH clause as an alternative to creating a view.
You can optionally specify the table-level and the column-level comments as in the CREATE TABLE statement.
Complex type considerations:
For tables containing complex type columns (ARRAY, STRUCT, or MAP), you typically use join queries to refer to the
complex values. You can use views to hide the join notation, making such tables seem like traditional denormalized
tables, and making those tables queryable by business intelligence tools that do not have built-in support for those
complex types. See Accessing Complex Type Data in Flattened Form Using Views on page 176 for details.
Because you cannot directly issue SELECT col_name against a column of complex type, you cannot use a view or a
WITH clause to “rename” a column by selecting it with a column alias.
If you connect to different Impala nodes within an impala-shell session for load-balancing purposes, you can enable
the SYNC_DDL query option to make each DDL statement wait before returning, until the new or changed metadata
has been received by all the Impala nodes. See SYNC_DDL Query Option on page 321 for details.
Security considerations:
If these statements in your environment contain sensitive literal values such as credit card numbers or tax identifiers,
Impala can redact this sensitive information when displaying the statements in log files and other administrative
contexts. See http://www.cloudera.com/content/cloudera/en/documentation/core/latest/topics/sg_redaction.html
for details.
Cancellation: Cannot be cancelled.
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Examples:
-- Create a view that includes only certain columns from the underlying table.
CREATE VIEW v2 AS SELECT c1, c3, c7 FROM t1;
-- Create a view that filters the values from the underlying table.
CREATE VIEW v3 AS SELECT DISTINCT c1, c3, c7 FROM t1 WHERE c1 IS NOT NULL AND c5 > 0;
-- Create a view that that reorders and renames columns from the underlying table.
CREATE VIEW v4 AS SELECT c4 AS last_name, c6 AS address, c2 AS birth_date FROM t1;
Related information:
Overview of Impala Views on page 210, ALTER VIEW Statement on page 224, DROP VIEW Statement on page 259
DESCRIBE Statement
The DESCRIBE statement displays metadata about a table, such as the column names and their data types. Its syntax
is:
You can use the abbreviation DESC for the DESCRIBE statement.
The DESCRIBE FORMATTED variation displays additional information, in a format familiar to users of Apache Hive.
The extra information includes low-level details such as whether the table is internal or external, when it was created,
the file format, the location of the data in HDFS, whether the object is a table or a view, and (for views) the text of the
query from the view definition.
Note: The Compressed field is not a reliable indicator of whether the table contains compressed
data. It typically always shows No, because the compression settings only apply during the session
that loads data and are not stored persistently with the table metadata.
For the ARRAY, STRUCT, and MAP types available in CDH 5.5 / Impala 2.3 and higher, the DESCRIBE output is formatted
to avoid excessively long lines for multiple fields within a STRUCT, or a nested sequence of complex types.
You can pass a qualified name to DESCRIBE to specify an ARRAY, STRUCT, or MAP column and visualize its structure
as if it were a table. An ARRAY is shown as a two-column table, with ITEM and POS columns. A STRUCT is shown as a
table with each field representing a column in the table. A MAP is shown as a two-column table, with KEY and VALUE
columns.
For example, here is the DESCRIBE output for a table containing a single top-level column of each complex type:
describe t1;
+------+-----------------+---------+
| name | type | comment |
+------+-----------------+---------+
| x | int | |
| a | array<int> | |
| s | struct< | |
| | f1:string, | |
| | f2:bigint | |
| | > | |
| m | map<string,int> | |
+------+-----------------+---------+
Here are examples showing how to “drill down” into the layouts of complex types, including using multi-part names
to examine the definitions of nested types. The < > delimiters identify the columns with complex types; these are the
columns where you can descend another level to see the parts that make up the complex type. This technique helps
you to understand the multi-part names you use as table references in queries involving complex types, and the
corresponding column names you refer to in the SELECT list. These tables are from the “nested TPC-H” schema, shown
in detail in Sample Schema and Data for Experimenting with Impala Complex Types on page 178.
The REGION table contains an ARRAY of STRUCT elements:
• The first DESCRIBE specifies the table name, to display the definition of each top-level column.
• The second DESCRIBE specifies the name of a complex column, REGION.R_NATIONS, showing that when you
include the name of an ARRAY column in a FROM clause, that table reference acts like a two-column table with
columns ITEM and POS.
• The final DESCRIBE specifies the fully qualified name of the ITEM field, to display the layout of its underlying
STRUCT type in table format, with the fields mapped to column names.
The CUSTOMER table contains an ARRAY of STRUCT elements, where one field in the STRUCT is another ARRAY of
STRUCT elements:
• The second DESCRIBE specifies the qualified name of the complex column, CUSTOMER.C_ORDERS, showing how
an ARRAY is represented as a two-column table with columns ITEM and POS.
• The third DESCRIBE specifies the qualified name of the ITEM of the ARRAY column, to see the structure of the
nested ARRAY. Again, it has has two parts, ITEM and POS. Because the ARRAY contains a STRUCT, the layout of
the STRUCT is shown.
• The fourth and fifth DESCRIBE statements drill down into a STRUCT field that is itself a complex type, an ARRAY
of STRUCT. The ITEM portion of the qualified name is only required when the ARRAY elements are anonymous.
The fields of the STRUCT give names to any other complex types nested inside the STRUCT. Therefore, the
DESCRIBE parameters CUSTOMER.C_ORDERS.ITEM.O_LINEITEMS and CUSTOMER.C_ORDERS.O_LINEITEMS
are equivalent. (For brevity, Cloudera recommends leaving out the ITEM portion of a qualified name when it is
not required.)
• The final DESCRIBE shows the layout of the deeply nested STRUCT type. Because there are no more complex
types nested inside this STRUCT, this is as far as you can drill down into the layout for this table.
| pos | bigint |
+------+------------------------------------+
-- #4: The ARRAY nested inside the STRUCT elements of the first ARRAY.
describe customer.c_orders.item.o_lineitems;
+------+----------------------------------+
| name | type |
+------+----------------------------------+
| item | struct< |
| | l_partkey:bigint, |
| | l_suppkey:bigint, |
... more struct fields ...
| | l_comment:string |
| | > |
| pos | bigint |
+------+----------------------------------+
-- #5: Shorter form of the previous DESCRIBE. Omits the .ITEM portion of the name
-- because O_LINEITEMS and other field names provide a way to refer to things
-- inside the ARRAY element.
describe customer.c_orders.o_lineitems;
+------+----------------------------------+
| name | type |
+------+----------------------------------+
| item | struct< |
| | l_partkey:bigint, |
| | l_suppkey:bigint, |
... more struct fields ...
| | l_comment:string |
| | > |
| pos | bigint |
+------+----------------------------------+
Usage notes:
After the impalad daemons are restarted, the first query against a table can take longer than subsequent queries,
because the metadata for the table is loaded before the query is processed. This one-time delay for each table can
cause misleading results in benchmark tests or cause unnecessary concern. To “warm up” the Impala metadata cache,
you can issue a DESCRIBE statement in advance for each table you intend to access later.
When you are dealing with data files stored in HDFS, sometimes it is important to know details such as the path of the
data files for an Impala table, and the hostname for the namenode. You can get this information from the DESCRIBE
FORMATTED output. You specify HDFS URIs or path specifications with statements such as LOAD DATA and the LOCATION
clause of CREATE TABLE or ALTER TABLE. You might also use HDFS URIs or paths with Linux commands such as
hadoop and hdfs to copy, rename, and so on, data files in HDFS.
If you connect to different Impala nodes within an impala-shell session for load-balancing purposes, you can enable
the SYNC_DDL query option to make each DDL statement wait before returning, until the new or changed metadata
has been received by all the Impala nodes. See SYNC_DDL Query Option on page 321 for details.
Each table can also have associated table statistics and column statistics. To see these categories of information, use
the SHOW TABLE STATS table_name and SHOW COLUMN STATS table_name statements. See SHOW Statement
on page 322 for details.
Important: After adding or replacing data in a table used in performance-critical queries, issue a
COMPUTE STATS statement to make sure all statistics are up-to-date. Consider updating statistics for
a table after any INSERT, LOAD DATA, or CREATE TABLE AS SELECT statement in Impala, or after
loading data through Hive and doing a REFRESH table_name in Impala. This technique is especially
important for tables that are very large, used in join queries, or both.
Examples:
The following example shows the results of both a standard DESCRIBE and DESCRIBE FORMATTED for different kinds
of schema objects:
• DESCRIBE for a table or a view returns the name, type, and comment for each of the columns. For a view, if the
column value is computed by an expression, the column name is automatically generated as _c0, _c1, and so on
depending on the ordinal number of the column.
• A table created with no special format or storage clauses is designated as a MANAGED_TABLE (an “internal table”
in Impala terminology). Its data files are stored in an HDFS directory under the default Hive data directory. By
default, it uses Text data format.
• A view is designated as VIRTUAL_VIEW in DESCRIBE FORMATTED output. Some of its properties are NULL or
blank because they are inherited from the base table. The text of the query that defines the view is part of the
DESCRIBE FORMATTED output.
• A table with additional clauses in the CREATE TABLE statement has differences in DESCRIBE FORMATTED output.
The output for T2 includes the EXTERNAL_TABLE keyword because of the CREATE EXTERNAL TABLE syntax,
and different InputFormat and OutputFormat fields to reflect the Parquet file format.
+------------------------------+--------------------------------------------+------------+
| # col_name | data_type | comment
|
| | NULL | NULL
|
| x | int | None
|
| y | int | None
|
| s | string | None
|
| | NULL | NULL
|
| # Detailed Table Information | NULL | NULL
|
| Database: | describe_formatted | NULL
|
| Owner: | cloudera | NULL
|
| CreateTime: | Mon Jul 22 17:03:16 EDT 2013 | NULL
|
| LastAccessTime: | UNKNOWN | NULL
|
| Protect Mode: | None | NULL
|
| Retention: | 0 | NULL
|
| Location: | hdfs://127.0.0.1:8020/user/hive/warehouse/ |
|
| | describe_formatted.db/t1 | NULL
|
| Table Type: | MANAGED_TABLE | NULL
|
| Table Parameters: | NULL | NULL
|
| | transient_lastDdlTime | 1374526996
|
| | NULL | NULL
|
| # Storage Information | NULL | NULL
|
| SerDe Library: | org.apache.hadoop.hive.serde2.lazy. |
|
| | LazySimpleSerDe | NULL
|
| InputFormat: | org.apache.hadoop.mapred.TextInputFormat | NULL
|
| OutputFormat: | org.apache.hadoop.hive.ql.io. |
|
| | HiveIgnoreKeyTextOutputFormat | NULL
|
| Compressed: | No | NULL
|
| Num Buckets: | 0 | NULL
|
| Bucket Columns: | [] | NULL
|
| Sort Columns: | [] | NULL
|
+------------------------------+--------------------------------------------+------------+
Returned 26 row(s) in 0.03s
[localhost:21000] > create view v1 as select x, upper(s) from t1;
Query: create view v1 as select x, upper(s) from t1
[localhost:21000] > describe v1;
Query: describe v1
Query finished, fetching results ...
+------+--------+---------+
| name | type | comment |
+------+--------+---------+
| x | int | |
| _c1 | string | |
+------+--------+---------+
Returned 2 row(s) in 0.10s
Related information:
Overview of Impala Tables on page 208, CREATE TABLE Statement on page 235, SHOW TABLES Statement on page 327,
SHOW CREATE TABLE Statement on page 329
When you do not use the CASCADE clause, drop or move all the objects inside the database manually before dropping
the database itself:
• Use the SHOW TABLES statement to locate all tables and views in the database, and issue DROP TABLE and DROP
VIEW statements to remove them all.
• Use the SHOW FUNCTIONS and SHOW AGGREGATE FUNCTIONS statements to locate all user-defined functions
in the database, and issue DROP FUNCTION and DROP AGGREGATE FUNCTION statements to remove them all.
• To keep tables or views contained by a database while removing the database itself, use ALTER TABLE and ALTER
VIEW to move the relevant objects to a different database before dropping the original database.
You cannot drop the current database, that is, the database your session connected to either through the USE statement
or the -d option of impala-shell. Issue a USE statement to switch to a different database first. Because the default
database is always available, issuing USE default is a convenient way to leave the current database before dropping
it.
Hive considerations:
When you drop a database in Impala, the database can no longer be used by Hive.
Examples:
See CREATE DATABASE Statement on page 231 for examples covering CREATE DATABASE, USE, and DROP DATABASE.
Cancellation: Cannot be cancelled.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have write permission for the
directory associated with the database.
Examples:
-- Before dropping a database, first drop all the tables inside it,
-- or in CDH 5.5 and higher use the CASCADE clause.
drop database temp;
ERROR: ImpalaRuntimeException: Error making 'dropDatabase' RPC to Hive Metastore:
CAUSED BY: InvalidOperationException: Database temp is not empty
show tables in temp;
+------+
| name |
+------+
| t3 |
+------+
Related information:
Overview of Impala Databases on page 205, CREATE DATABASE Statement on page 231, USE Statement on page 337,
SHOW DATABASES on page 326, DROP TABLE Statement on page 258
Syntax:
Required privileges:
Only administrative users (initially, a predefined set of users specified in the Sentry service configuration file) can use
this statement.
Compatibility:
Impala makes use of any roles and privileges specified by the GRANT and REVOKE statements in Hive, and Hive makes
use of any roles and privileges specified by the GRANT and REVOKE statements in Impala. The Impala GRANT and REVOKE
statements for privileges do not require the ROLE keyword to be repeated before each role name, unlike the equivalent
Hive statements.
Related information:
Enabling Sentry Authorization for Impala on page 95, GRANT Statement (CDH 5.2 or higher only) on page 263 REVOKE
Statement (CDH 5.2 or higher only) on page 279, CREATE ROLE Statement (CDH 5.2 or higher only) on page 235, SHOW
Statement on page 322
Cancellation: Cannot be cancelled.
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
The PARTITION clause is only allowed in combination with the INCREMENTAL clause. It is optional for COMPUTE
INCREMENTAL STATS, and required for DROP INCREMENTAL STATS. Whenever you specify partitions through the
PARTITION (partition_spec) clause in a COMPUTE INCREMENTAL STATS or DROP INCREMENTAL STATS
statement, you must include all the partitioning columns in the specification, and specify constant values for all the
partition key columns.
DROP STATS removes all statistics from the table, whether created by COMPUTE STATS or COMPUTE INCREMENTAL
STATS.
DROP INCREMENTAL STATS only affects incremental statistics for a single partition, specified through the PARTITION
clause. The incremental stats are marked as outdated, so that they are recomputed by the next COMPUTE INCREMENTAL
STATS statement.
Usage notes:
You typically use this statement when the statistics for a table or a partition have become stale due to data files being
added to or removed from the associated HDFS data directories, whether by manual HDFS operations or INSERT,
INSERT OVERWRITE, or LOAD DATA statements, or adding or dropping partitions.
When a table or partition has no associated statistics, Impala treats it as essentially zero-sized when constructing the
execution plan for a query. In particular, the statistics influence the order in which tables are joined in a join query. To
ensure proper query planning and good query performance and scalability, make sure to run COMPUTE STATS or
COMPUTE INCREMENTAL STATS on the table or partition after removing any stale statistics.
Dropping the statistics is not required for an unpartitioned table or a partitioned table covered by the original type of
statistics. A subsequent COMPUTE STATS statement replaces any existing statistics with new ones, for all partitions,
regardless of whether the old ones were outdated. Therefore, this statement was rarely used before the introduction
of incremental statistics.
Dropping the statistics is required for a partitioned table containing incremental statistics, to make a subsequent
COMPUTE INCREMENTAL STATS statement rescan an existing partition. See Table and Column Statistics on page 489
for information about incremental statistics, a new feature available in Impala 2.1.0 and higher.
Statement type: DDL
To remove statistics for particular partitions, use the DROP INCREMENTAL STATS statement. After removing statistics
for two partitions, the table-level statistics reflect that change in the #Rows and Incremental stats fields. The
counts, maximums, and averages of the column-level statistics are unaffected.
Note: (It is possible that the row count might be preserved in future after a DROP INCREMENTAL
STATS statement. Check the resolution of the issue IMPALA-1615.)
To remove all statistics from the table, whether produced by COMPUTE STATS or COMPUTE INCREMENTAL STATS,
use the DROP STATS statement without the INCREMENTAL clause). Now, both table-level and column-level statistics
are reset.
Related information:
COMPUTE STATS Statement on page 226, SHOW TABLE STATS Statement on page 330, SHOW COLUMN STATS Statement
on page 332, Table and Column Statistics on page 489
IF EXISTS clause:
The optional IF EXISTS clause makes the statement succeed whether or not the table exists. If the table does exist,
it is dropped; if it does not exist, the statement has no effect. This capability is useful in standardized setup scripts that
remove existing schema objects and create new ones. By using some combination of IF EXISTS for the DROP statements
and IF NOT EXISTS clauses for the CREATE statements, the script can run successfully the first time you run it (when
the objects do not exist yet) and subsequent times (when some or all of the objects do already exist).
PURGE clause:
The optional PURGE keyword, available in CDH 5.5 / Impala 2.3 and higher, causes Impala to remove the associated
HDFS data files immediately, rather than going through the HDFS trashcan mechanism. Use this keyword when dropping
a table if it is crucial to remove the data as quickly as possible to free up space, or if there is a problem with the trashcan,
such as the trashcan not being configured or being in a different HDFS encryption zone than the data files.
Statement type: DDL
Usage notes:
By default, Impala removes the associated HDFS directory and data files for the table. If you issue a DROP TABLE and
the data files are not deleted, it might be for the following reasons:
• If the table was created with the EXTERNAL clause, Impala leaves all files and directories untouched. Use external
tables when the data is under the control of other Hadoop components, and Impala is only used to query the data
files from their original locations.
• Impala might leave the data files behind unintentionally, if there is no HDFS location available to hold the HDFS
trashcan for the impala user. See User Account Requirements on page 24 for the procedure to set up the required
HDFS home directory.
Make sure that you are in the correct database before dropping a table, either by issuing a USE statement first or by
using a fully qualified name db_name.table_name.
If you intend to issue a DROP DATABASE statement, first issue DROP TABLE statements to remove all the tables in
that database.
Examples:
For other tips about managing and reclaiming Impala disk space, see Managing Disk Space for Impala Data on page
89.
Amazon S3 considerations:
Although Impala cannot write new data to a table stored in the Amazon S3 filesystem, the DROP TABLE statement
can remove data files from S3 if the associated S3 table is an internal table. See Using Impala to Query the Amazon S3
Filesystem on page 566 for details about working with S3 tables.
Cancellation: Cannot be cancelled.
HDFS permissions:
For an internal table, the user ID that the impalad daemon runs under, typically the impala user, must have write
permission for all the files and directories that make up the table.
For an external table, dropping the table only involves changes to metadata in the metastore database. Because Impala
does not remove any HDFS files or directories when external tables are dropped, no particular permissions are needed
for the associated HDFS files or directories.
Related information:
Overview of Impala Tables on page 208, ALTER TABLE Statement on page 215, CREATE TABLE Statement on page 235,
Partitioning for Impala Tables on page 522, Internal Tables on page 208, External Tables on page 209
USE db1;
-- Create a view in a different database.
CREATE VIEW db2.v1 AS SELECT * FROM db2.foo;
-- Switch into the other database and drop the view.
USE db2;
DROP VIEW v1;
USE db1;
-- Create a view in a different database.
CREATE VIEW db2.v1 AS SELECT * FROM db2.foo;
-- Drop a view in the other database.
DROP VIEW db2.v1;
Related information:
Overview of Impala Views on page 210, CREATE VIEW Statement on page 243, ALTER VIEW Statement on page 224
EXPLAIN Statement
Returns the execution plan for a statement, showing the low-level mechanisms that Impala will use to read the data,
divide the work among nodes in the cluster, and transmit intermediate and final results across the network. Use
explain followed by a complete SELECT query. For example:
Syntax:
The select_query is a SELECT statement, optionally prefixed by a WITH clause. See SELECT Statement on page 280 for
details.
The insert_stmt is an INSERT statement that inserts into or overwrites an existing table. It can use either the INSERT
... SELECT or INSERT ... VALUES syntax. See INSERT Statement on page 263 for details.
The ctas_stmt is a CREATE TABLE statement using the AS SELECT clause, typically abbreviated as a “CTAS” operation.
See CREATE TABLE Statement on page 235 for details.
Usage notes:
You can interpret the output to judge whether the query is performing efficiently, and adjust the query and/or the
schema if not. For example, you might change the tests in the WHERE clause, add hints to make join operations more
efficient, introduce subqueries, change the order of tables in a join, add or change partitioning for a table, collect
column statistics and/or table statistics in Hive, or any other performance tuning steps.
The EXPLAIN output reminds you if table or column statistics are missing from any table involved in the query. These
statistics are important for optimizing queries involving large tables or multi-table joins. See COMPUTE STATS Statement
on page 226 for how to gather statistics, and Table and Column Statistics on page 489 for how to use this information
for query tuning.
These examples show how the extended EXPLAIN output becomes more accurate and informative as statistics are
gathered by the COMPUTE STATS statement. Initially, much of the information about data size and distribution is
marked “unavailable”. Impala can determine the raw data size, but not the number of rows or number of distinct
values for each column without additional analysis. The COMPUTE STATS statement performs this analysis, so a
subsequent EXPLAIN statement has additional information to use in deciding how to optimize the distributed query.
Security considerations:
If these statements in your environment contain sensitive literal values such as credit card numbers or tax identifiers,
Impala can redact this sensitive information when displaying the statements in log files and other administrative
contexts. See http://www.cloudera.com/content/cloudera/en/documentation/core/latest/topics/sg_redaction.html
for details.
Cancellation: Cannot be cancelled.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read and execute permissions
for all applicable directories in all source tables for the query that is being explained. (A SELECT operation could read
files from multiple different HDFS directories if the source table is partitioned.)
Related information:
SELECT Statement on page 280, INSERT Statement on page 263, CREATE TABLE Statement on page 235, Understanding
Impala Query Performance - EXPLAIN Plans and Query Profiles on page 507
INSERT Statement
Impala supports inserting into tables and partitions that you create with the Impala CREATE TABLE statement, or
pre-defined tables and partitions created through Hive.
Syntax:
[with_clause]
INSERT { INTO | OVERWRITE } [TABLE] table_name
[(column_list)]
[ PARTITION (partition_clause)]
{
[hint_clause] select_statement
| VALUES (value [, value ...]) [, (value [, value ...]) ...]
}
hint_clause ::= [SHUFFLE] | [NOSHUFFLE] (Note: the square brackets are part of the
syntax.)
• An optional hint clause immediately before the SELECT keyword, to fine-tune the behavior when doing an INSERT
... SELECT operation into partitioned Parquet tables. The hint keywords are [SHUFFLE] and [NOSHUFFLE],
including the square brackets. Inserting into partitioned Parquet tables can be a resource-intensive operation
because it potentially involves many files being written to HDFS simultaneously, and separate large memory
buffers being allocated to buffer the data for each partition. For usage details, see Loading Data into Parquet
Tables on page 536.
Note:
• Insert commands that partition or add files result in changes to Hive metadata. Because Impala
uses Hive metadata, such changes may necessitate a metadata refresh. For more information,
see the REFRESH function.
• Currently, Impala can only insert data into tables that use the text and Parquet formats. For other
file formats, insert the data using Hive and use Impala to query it.
• As an alternative to the INSERT statement, if you have existing data files elsewhere in HDFS, the
LOAD DATA statement can move those files into a table. This statement works with tables of any
file format.
Important: After adding or replacing data in a table used in performance-critical queries, issue a
COMPUTE STATS statement to make sure all statistics are up-to-date. Consider updating statistics for
a table after any INSERT, LOAD DATA, or CREATE TABLE AS SELECT statement in Impala, or after
loading data through Hive and doing a REFRESH table_name in Impala. This technique is especially
important for tables that are very large, used in join queries, or both.
Examples:
The following example sets up new tables with the same definition as the TAB1 table from the Tutorial section, using
different file formats, and demonstrates inserting data into the tables created with the STORED AS TEXTFILE and
STORED AS PARQUET clauses:
With the INSERT INTO TABLE syntax, each new set of inserted rows is appended to any existing data in the table.
This is how you would record small amounts of data that arrive continuously, or ingest new batches of data alongside
the existing data. For example, after running 2 INSERT INTO TABLE statements with 5 rows each, the table contains
10 rows total:
With the INSERT OVERWRITE TABLE syntax, each new set of inserted rows replaces any existing data in the table.
This is how you load data to query in a data warehousing scenario where you analyze just the data for a particular day,
quarter, and so on, discarding the previous data each time. You might keep the entire set of data in one raw table, and
transfer and transform certain rows into a more compact and efficient form to perform intensive analysis on that
subset.
For example, here we insert 5 rows into a table using the INSERT INTO clause, then replace the data by inserting 3
rows with the INSERT OVERWRITE clause. Afterward, the table only contains the 3 rows from the final INSERT
statement.
The VALUES clause lets you insert one or more rows by specifying constant values for all the columns. The number,
types, and order of the expressions must match the table definition.
Note: The INSERT ... VALUES technique is not suitable for loading large quantities of data into
HDFS-based tables, because the insert operations cannot be parallelized, and each one produces a
separate data file. Use it for setting up small dimension tables or tiny amounts of data for experimenting
with SQL syntax, or with HBase tables. Do not use it for large ETL jobs or benchmark tests for load
operations. Do not run scripts with thousands of INSERT ... VALUES statements that insert a single
row each time. If you do run INSERT ... VALUES operations to load data into a staging table as
one stage in an ETL pipeline, include multiple row values if possible within each VALUES clause, and
use a separate database to make cleanup easier if the operation does produce many tiny files.
The following example shows how to insert one row or multiple rows, with expressions of different types, using literal
values, expressions, and function return values:
These examples show the type of “not implemented” error that you see when attempting to insert data into a table
with a file format that Impala currently does not write to:
Inserting data into partitioned tables requires slightly different syntax that divides the partitioning columns from the
others:
The following examples show how you can copy the data in all the columns from one table to another, copy the data
from only some columns, or specify the columns in the select list in a different order than they actually appear in the
table:
-- The column names can be entirely different in the source and destination tables.
-- You can copy any columns, not just the corresponding ones, from the source table.
-- But the number and type of selected columns must match the columns mentioned in the
() part.
alter table t2 replace columns (x int, y int);
insert into t2 (y) select c1 from t1;
-- For partitioned tables, all the partitioning columns must be mentioned in the ()
column list
-- or a PARTITION clause; these columns cannot be defaulted to NULL.
create table pt1 (x int, y int) partitioned by (z int);
-- The values from c1 are copied into the column x in the new table,
-- all in the same partition based on a constant value for z.
-- The values of y in the new table are all NULL.
insert into pt1 (x) partition (z=5) select c1 from t1;
-- Again we omit the values for column y so they are all NULL.
-- The inserted x values can go into different partitions, based on
-- the different values inserted into the partitioning column z.
insert into pt1 (x,z) select x, z from t2;
SELECT * for a partitioned table requires that all partition key columns in the source table be declared as the last
columns in the CREATE TABLE statement. You still include a PARTITION BY clause listing all the partition key columns.
These partition columns are automatically mapped to the last columns from the SELECT * list.
create table source (x int, y int, year int, month int, day int);
create table destination (x int, y int) partitioned by (year int, month int, day int);
...load some data into the unpartitioned source table...
-- Insert a single partition of data.
-- The SELECT * means you cannot specify partition (year=2014, month, day).
insert overwrite destination partition (year, month, day) select * from source where
year=2014;
-- Insert the data for all year/month/day combinations.
insert overwrite destination partition (year, month, day) select * from source;
Sorting considerations: Although you can specify an ORDER BY clause in an INSERT ... SELECT statement, any
ORDER BY clause is ignored and the results are not necessarily sorted. An INSERT ... SELECT operation potentially
creates many different data files, prepared on different data nodes, and therefore the notion of the data being stored
in sorted order is impractical.
Concurrency considerations: Each INSERT operation creates new data files with unique names, so you can run multiple
INSERT INTO statements simultaneously without filename conflicts. While data is being inserted into an Impala table,
the data is staged temporarily in a subdirectory inside the data directory; during this period, you cannot issue queries
against that table in Hive. If an INSERT operation fails, the temporary data file and the subdirectory could be left
behind in the data directory. If so, remove the relevant subdirectory and any data files it contains manually, by issuing
an hdfs dfs -rm -r command, specifying the full path of the work subdirectory, whose name ends in _dir.
VALUES Clause
The VALUES clause is a general-purpose way to specify the columns of one or more rows, typically within an INSERT
statement.
Note: The INSERT ... VALUES technique is not suitable for loading large quantities of data into
HDFS-based tables, because the insert operations cannot be parallelized, and each one produces a
separate data file. Use it for setting up small dimension tables or tiny amounts of data for experimenting
with SQL syntax, or with HBase tables. Do not use it for large ETL jobs or benchmark tests for load
operations. Do not run scripts with thousands of INSERT ... VALUES statements that insert a single
row each time. If you do run INSERT ... VALUES operations to load data into a staging table as
one stage in an ETL pipeline, include multiple row values if possible within each VALUES clause, and
use a separate database to make cleanup easier if the operation does produce many tiny files.
When used in an INSERT statement, the Impala VALUES clause can specify some or all of the columns in the destination
table, and the columns can be specified in a different order than they actually appear in the table. To specify a different
set or order of columns than in the table, use the syntax:
Any columns in the table that are not listed in the INSERT statement are set to NULL.
To use a VALUES clause like a table in other statements, wrap it in parentheses and use AS clauses to specify aliases
for the entire object and any columns you need to refer to:
For example, you might use a tiny table constructed like this from constant literals or function return values as part of
a longer statement involving joins or UNION ALL.
HDFS considerations:
Impala physically writes all inserted files under the ownership of its default user, typically impala. Therefore, this user
must have HDFS write permission in the corresponding table directory.
The permission requirement is independent of the authorization performed by the Sentry framework. (If the connected
user is not authorized to insert into a table, Sentry blocks that operation immediately, regardless of the privileges
available to the impala user.) Files created by Impala are not owned by and do not inherit permissions from the
connected user.
The number of data files produced by an INSERT statement depends on the size of the cluster, the number of data
blocks that are processed, the partition key columns in a partitioned table, and the mechanism Impala uses for dividing
the work in parallel. Do not assume that an INSERT statement will produce some particular number of output files.
In case of performance issues with data written by Impala, check that the output files do not suffer from issues such
as many tiny files or many tiny partitions. (In the Hadoop context, even files or partitions of a few tens of megabytes
are considered “tiny”.)
The INSERT statement has always left behind a hidden work directory inside the data directory of the table. Formerly,
this hidden work directory was named .impala_insert_staging . In Impala 2.0.1 and later, this directory name is
changed to _impala_insert_staging . (While HDFS tools are expected to treat names beginning either with
underscore and dot as hidden, in practice names beginning with an underscore are more widely supported.) If you
have any scripts, cleanup jobs, and so on that rely on the name of this work directory, adjust them to use the new
name.
HBase considerations:
You can use the INSERT statement with HBase tables as follows:
• You can insert a single row or a small set of rows into an HBase table with the INSERT ... VALUES syntax. This
is a good use case for HBase tables with Impala, because HBase tables are not subject to the same kind of
fragmentation from many small insert operations as HDFS tables are.
• You can insert any number of rows at once into an HBase table using the INSERT ... SELECT syntax.
• If more than one inserted row has the same value for the HBase key column, only the last inserted row with that
value is visible to Impala queries. You can take advantage of this fact with INSERT ... VALUES statements to
effectively update rows one at a time, by inserting new rows with the same key values as existing rows. Be aware
that after an INSERT ... SELECT operation copying from an HDFS table, the HBase table might contain fewer
rows than were inserted, if the key column in the source table contained duplicate values.
• You cannot INSERT OVERWRITE into an HBase table. New rows are always appended.
• When you create an Impala or Hive table that maps to an HBase table, the column order you specify with the
INSERT statement might be different than the order you declare with the CREATE TABLE statement. Behind the
scenes, HBase arranges the columns based on how they are divided into column families. This might cause a
mismatch during insert operations, especially if you use the syntax INSERT INTO hbase_table SELECT *
FROM hdfs_table. Before inserting data, verify the column order by issuing a DESCRIBE statement for the table,
and adjust the order of the select list in the INSERT statement.
See Using Impala to Query HBase Tables on page 557 for more details about using Impala with HBase.
Amazon S3 considerations:
Currently, Impala cannot insert or load data into a table or partition that resides in the Amazon Simple Storage Service
(S3). Bring data into S3 using the normal S3 transfer mechanisms, then use Impala to query the S3 data. See Using
Impala to Query the Amazon S3 Filesystem on page 566 for details about using Impala with S3.
Security considerations:
If these statements in your environment contain sensitive literal values such as credit card numbers or tax identifiers,
Impala can redact this sensitive information when displaying the statements in log files and other administrative
contexts. See http://www.cloudera.com/content/cloudera/en/documentation/core/latest/topics/sg_redaction.html
for details.
Cancellation: Can be cancelled. To cancel this statement, use Ctrl-C from the impala-shell interpreter, the Cancel
button from the Watch page in Hue, Actions > Cancel from the Queries list in Cloudera Manager, or Cancel from the
list of in-flight queries (for a particular node) on the Queries tab in the Impala web UI (port 25000).
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read permission for the files
in the source directory of an INSERT ... SELECT operation, and write permission for all affected directories in the
destination table. (An INSERT operation could write files to multiple different HDFS directories if the destination table
is partitioned.) This user must also have write permission to create a temporary work directory in the top-level HDFS
directory of the destination table. An INSERT OVERWRITE operation does not require write permission on the original
data files in the table, only on the table directories themselves.
Restrictions:
For INSERT operations into CHAR or VARCHAR columns, you must cast all STRING literals or expressions returning
STRING to to a CHAR or VARCHAR type with the appropriate length.
By default, the cached metadata for all tables is flushed. If you specify a table name, only the metadata for that one
table is flushed. Even for a single table, INVALIDATE METADATA is more expensive than REFRESH, so prefer REFRESH
in the common case where you add new data files for an existing table.
Internal details:
To accurately respond to queries, Impala must have current metadata about those databases and tables that clients
query directly. Therefore, if some other entity modifies information used by Impala in the metastore that Impala and
Hive share, the information cached by Impala must be updated. However, this does not mean that all metadata updates
require an Impala update.
Note:
In Impala 1.2.4 and higher, you can specify a table name with INVALIDATE METADATA after the table
is created in Hive, allowing you to make individual tables visible to Impala without doing a full reload
of the catalog metadata. Impala 1.2.4 also includes other changes to make the metadata broadcast
mechanism faster and more responsive, especially during Impala startup. See New Features in Impala
Version 1.2.4 on page 617 for details.
In Impala 1.2 and higher, a dedicated daemon (catalogd) broadcasts DDL changes made through
Impala to all Impala nodes. Formerly, after you created a database or table while connected to one
Impala node, you needed to issue an INVALIDATE METADATA statement on another Impala node
before accessing the new database or table from the other node. Now, newly created or altered
objects are picked up automatically by all Impala nodes. You must still use the INVALIDATE METADATA
technique after creating or altering objects through Hive. See The Impala Catalog Service on page 19
for more information on the catalog service.
The INVALIDATE METADATA statement is new in Impala 1.1 and higher, and takes over some of the
use cases of the Impala 1.0 REFRESH statement. Because REFRESH now requires a table name
parameter, to flush the metadata for all tables at once, use the INVALIDATE METADATA statement.
Because REFRESH table_name only works for tables that the current Impala node is already aware
of, when you create a new table in the Hive shell, enter INVALIDATE METADATA new_table before
you can see the new table in impala-shell. Once the table is known by Impala, you can issue
REFRESH table_name after you add data files for that table.
INVALIDATE METADATA and REFRESH are counterparts: INVALIDATE METADATA waits to reload the metadata when
needed for a subsequent query, but reloads all the metadata for the table, which can be an expensive operation,
especially for large tables with many partitions. REFRESH reloads the metadata immediately, but only loads the block
location data for newly added data files, making it a less expensive operation overall. If data was altered in some more
extensive way, such as being reorganized by the HDFS balancer, use INVALIDATE METADATA to avoid a performance
penalty from reduced local reads. If you used Impala version 1.0, the INVALIDATE METADATA statement works just
like the Impala 1.0 REFRESH statement did, while the Impala 1.1 REFRESH is optimized for the common use case of
adding new data files to an existing table, thus the table name argument is now required.
Usage notes:
A metadata update for an impalad instance is required if:
• A metadata change occurs.
• and the change is made from another impalad instance in your cluster, or through Hive.
• and the change is made to a metastore database to which clients such as the Impala shell or ODBC directly connect.
A metadata update for an Impala node is not required when you issue queries from the same Impala node where you
ran ALTER TABLE, INSERT, or other table-modifying statement.
Database and table metadata is typically modified by:
• Hive - via ALTER, CREATE, DROP or INSERT operations.
• Impalad - via CREATE TABLE, ALTER TABLE, and INSERT operations.
INVALIDATE METADATA causes the metadata for that table to be marked as stale, and reloaded the next time the
table is referenced. For a huge table, that process could take a noticeable amount of time; thus you might prefer to
use REFRESH where practical, to avoid an unpredictable delay later, for example if the next reference to the table is
during a benchmark test.
Examples:
The following example shows how you might use the INVALIDATE METADATA statement after creating new tables
(such as SequenceFile or HBase tables) through the Hive shell. Before the INVALIDATE METADATA statement was
issued, Impala would give a “table not found” error if you tried to refer to those table names. The DESCRIBE statements
cause the latest metadata to be immediately loaded for the tables, avoiding a delay the next time those tables are
queried.
For more examples of using REFRESH and INVALIDATE METADATA with a combination of Impala and Hive operations,
see Switching Back and Forth Between Impala and Hive on page 60.
If you need to ensure that the metadata is up-to-date when you start an impala-shell session, run impala-shell
with the -r or --refresh_after_connect command-line option. Because this operation adds a delay to the next
query against each table, potentially expensive for large tables with many partitions, try to avoid using this option for
day-to-day operations in a production environment.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have execute permissions for all
the relevant directories holding table data. (A table could have data spread across multiple directories, or in unexpected
paths, if it uses partitioning or specifies a LOCATION attribute for individual partitions or the entire table.) Issues with
permissions might not cause an immediate error for this statement, but subsequent statements such as SELECT or
SHOW TABLE STATS could fail.
HDFS considerations:
By default, the INVALIDATE METADATA command checks HDFS permissions of the underlying data files and directories,
caching this information so that a statement can be cancelled immediately if for example the impala user does not
have permission to write to the data directory for the table. (This checking does not apply if you have set the catalogd
configuration option --load_catalog_in_background=false.) Impala reports any lack of write permissions as
an INFO message in the log file, in case that represents an oversight. If you change HDFS permissions to make data
readable or writeable by the Impala user, issue another INVALIDATE METADATA to make Impala aware of the change.
Usage notes:
This example illustrates creating a new database and new table in Hive, then doing an INVALIDATE METADATA
statement in Impala using the fully qualified table name, after which both the new table and the new database are
visible to Impala. The ability to specify INVALIDATE METADATA table_name for a table created in Hive is a new
capability in Impala 1.2.4. In earlier releases, that statement would have returned an error indicating an unknown
table, requiring you to do INVALIDATE METADATA with no table name, a more expensive operation that reloaded
metadata for all tables and databases.
$ hive
hive> create database new_db_from_hive;
OK
Time taken: 4.118 seconds
hive> create table new_db_from_hive.new_table_from_hive (x int);
OK
Time taken: 0.618 seconds
hive> quit;
$ impala-shell
[localhost:21000] > show databases like 'new*';
[localhost:21000] > refresh new_db_from_hive.new_table_from_hive;
Amazon S3 considerations:
The REFRESH and INVALIDATE METADATA statements also cache metadata for tables where the data resides in the
Amazon Simple Storage Service (S3). In particular, issue a REFRESH for a table after adding or removing files in the
associated S3 data directory. See Using Impala to Query the Amazon S3 Filesystem on page 566 for details about working
with S3 tables.
Cancellation: Cannot be cancelled.
Related information:
Overview of Impala Metadata and the Metastore on page 21, REFRESH Statement on page 277
When the LOAD DATA statement operates on a partitioned table, it always operates on one partition at a time. Specify
the PARTITION clauses and list all the partition key columns, with a constant value specified for each.
Statement type: DML (but still affected by SYNC_DDL query option)
Usage notes:
• The loaded data files are moved, not copied, into the Impala data directory.
• You can specify the HDFS path of a single file to be moved, or the HDFS path of a directory to move all the files
inside that directory. You cannot specify any sort of wildcard to take only some of the files from a directory. When
loading a directory full of data files, keep all the data files at the top level, with no nested directories underneath.
• Currently, the Impala LOAD DATA statement only imports files from HDFS, not from the local filesystem. It does
not support the LOCAL keyword of the Hive LOAD DATA statement. You must specify a path, not an hdfs://
URI.
• In the interest of speed, only limited error checking is done. If the loaded files have the wrong file format, different
columns than the destination table, or other kind of mismatch, Impala does not raise any error for the LOAD DATA
statement. Querying the table afterward could produce a runtime error or unexpected results. Currently, the only
checking the LOAD DATA statement does is to avoid mixing together uncompressed and LZO-compressed text
files in the same table.
• When you specify an HDFS directory name as the LOAD DATA argument, any hidden files in that directory (files
whose names start with a .) are not moved to the Impala data directory.
• The loaded data files retain their original names in the new location, unless a name conflicts with an existing data
file, in which case the name of the new file is modified slightly to be unique. (The name-mangling is a slight
difference from the Hive LOAD DATA statement, which replaces identically named files.)
• By providing an easy way to transport files from known locations in HDFS into the Impala data directory structure,
the LOAD DATA statement lets you avoid memorizing the locations and layout of HDFS directory tree containing
the Impala databases and tables. (For a quick way to check the location of the data files for an Impala table, issue
the statement DESCRIBE FORMATTED table_name.)
• The PARTITION clause is especially convenient for ingesting new data for a partitioned table. As you receive new
data for a time period, geographic region, or other division that corresponds to one or more partitioning columns,
you can load that data straight into the appropriate Impala data directory, which might be nested several levels
down if the table is partitioned by multiple columns. When the table is partitioned, you must specify constant
values for all the partitioning columns.
Complex type considerations:
Because Impala currently cannot create Parquet data files containing complex types (ARRAY, STRUCT, and MAP), the
LOAD DATA statement is especially important when working with tables containing complex type columns. You create
the Parquet data files outside Impala, then use either LOAD DATA, an external table, or HDFS-level file operations
followed by REFRESH to associate the data files with the corresponding table. See Complex Types (CDH 5.5 or higher
only) on page 156 for details about using complex types.
If you connect to different Impala nodes within an impala-shell session for load-balancing purposes, you can enable
the SYNC_DDL query option to make each DDL statement wait before returning, until the new or changed metadata
has been received by all the Impala nodes. See SYNC_DDL Query Option on page 321 for details.
Important: After adding or replacing data in a table used in performance-critical queries, issue a
COMPUTE STATS statement to make sure all statistics are up-to-date. Consider updating statistics for
a table after any INSERT, LOAD DATA, or CREATE TABLE AS SELECT statement in Impala, or after
loading data through Hive and doing a REFRESH table_name in Impala. This technique is especially
important for tables that are very large, used in join queries, or both.
Examples:
First, we use a trivial Python script to write different numbers of strings (one per line) into files stored in the cloudera
HDFS user account. (Substitute the path for your own HDFS user account when doing hdfs dfs operations like these.)
Next, we create a table and load an initial set of data into it. Remember, unless you specify a STORED AS clause, Impala
tables default to TEXTFILE format with Ctrl-A (hex 01) as the field delimiter. This example uses a single-column table,
so the delimiter is not significant. For large-scale ETL jobs, you would typically use binary format data files such as
Parquet or Avro, and load them into Impala tables that use the corresponding file format.
As indicated by the message at the end of the previous example, the data file was moved from its original location.
The following example illustrates how the data file was moved into the Impala data directory for the destination table,
keeping its original filename:
The following example demonstrates the difference between the INTO TABLE and OVERWRITE TABLE clauses. The
table already contains 1000 rows. After issuing the LOAD DATA statement with the INTO TABLE clause, the table
contains 100 more rows, for a total of 1100. After issuing the LOAD DATA statement with the OVERWRITE INTO
TABLE clause, the former contents are gone, and now the table only contains the 10 rows from the just-loaded data
file.
Amazon S3 considerations:
Currently, Impala cannot insert or load data into a table or partition that resides in the Amazon Simple Storage Service
(S3). Bring data into S3 using the normal S3 transfer mechanisms, then use Impala to query the S3 data. See Using
Impala to Query the Amazon S3 Filesystem on page 566 for details about using Impala with S3.
Cancellation: Cannot be cancelled.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read and write permissions
for the files in the source directory, and write permission for the destination directory.
HBase considerations:
REFRESH Statement
To accurately respond to queries, the Impala node that acts as the coordinator (the node to which you are connected
through impala-shell, JDBC, or ODBC) must have current metadata about those databases and tables that are
referenced in Impala queries. If you are not familiar with the way Impala uses metadata and how it shares the same
metastore database as Hive, see Overview of Impala Metadata and the Metastore on page 21 for background
information.
Syntax:
REFRESH [db_name.]table_name
Usage notes:
Use the REFRESH statement to load the latest metastore metadata and block location data for a particular table in
these scenarios:
• After loading new data files into the HDFS data directory for the table. (Once you have set up an ETL pipeline to
bring data into Impala on a regular basis, this is typically the most frequent reason why metadata needs to be
refreshed.)
• After issuing ALTER TABLE, INSERT, LOAD DATA, or other table-modifying SQL statement in Hive.
Note:
In CDH 5.5 / Impala 2.3 and higher, the syntax ALTER TABLE table_name RECOVER PARTITIONS
is a faster alternative to REFRESH when the only change to the table data is the addition of new
partition directories through Hive or manual HDFS operations. See ALTER TABLE Statement on page
215 for details.
You only need to issue the REFRESH statement on the node to which you connect to issue queries. The coordinator
node divides the work among all the Impala nodes in a cluster, and sends read requests for the correct HDFS blocks
without relying on the metadata on the other nodes.
REFRESH reloads the metadata for the table from the metastore database, and does an incremental reload of the
low-level block location data to account for any new data files added to the HDFS data directory for the table. It is a
low-overhead, single-table operation, specifically tuned for the common scenario where new data files are added to
HDFS.
Only the metadata for the specified table is flushed. The table must already exist and be known to Impala, either
because the CREATE TABLE statement was run in Impala rather than Hive, or because a previous INVALIDATE
METADATA statement caused Impala to reload its entire metadata catalog.
Note:
The catalog service broadcasts any changed metadata as a result of Impala ALTER TABLE, INSERT
and LOAD DATA statements to all Impala nodes. Thus, the REFRESH statement is only required if you
load data through Hive or by manipulating data files in HDFS directly. See The Impala Catalog Service
on page 19 for more information on the catalog service.
Another way to avoid inconsistency across nodes is to enable the SYNC_DDL query option before
performing a DDL statement or an INSERT or LOAD DATA.
The table name is a required parameter. To flush the metadata for all tables, use the INVALIDATE
METADATA command.
Because REFRESH table_name only works for tables that the current Impala node is already aware
of, when you create a new table in the Hive shell, enter INVALIDATE METADATA new_table before
you can see the new table in impala-shell. Once the table is known by Impala, you can issue
REFRESH table_name after you add data files for that table.
INVALIDATE METADATA and REFRESH are counterparts: INVALIDATE METADATA waits to reload the metadata when
needed for a subsequent query, but reloads all the metadata for the table, which can be an expensive operation,
especially for large tables with many partitions. REFRESH reloads the metadata immediately, but only loads the block
location data for newly added data files, making it a less expensive operation overall. If data was altered in some more
extensive way, such as being reorganized by the HDFS balancer, use INVALIDATE METADATA to avoid a performance
penalty from reduced local reads. If you used Impala version 1.0, the INVALIDATE METADATA statement works just
like the Impala 1.0 REFRESH statement did, while the Impala 1.1 REFRESH is optimized for the common use case of
adding new data files to an existing table, thus the table name argument is now required.
A metadata update for an impalad instance is required if:
• A metadata change occurs.
• and the change is made through Hive.
• and the change is made to a metastore database to which clients such as the Impala shell or ODBC directly connect.
A metadata update for an Impala node is not required after you run ALTER TABLE, INSERT, or other table-modifying
statement in Impala rather than Hive. Impala handles the metadata synchronization automatically through the catalog
service.
Database and table metadata is typically modified by:
• Hive - through ALTER, CREATE, DROP or INSERT operations.
• Impalad - through CREATE TABLE, ALTER TABLE, and INSERT operations. Such changes are propagated to all
Impala nodes by the Impala catalog service.
REFRESH causes the metadata for that table to be immediately reloaded. For a huge table, that process could take a
noticeable amount of time; but doing the refresh up front avoids an unpredictable delay later, for example if the next
reference to the table is during a benchmark test.
If you connect to different Impala nodes within an impala-shell session for load-balancing purposes, you can enable
the SYNC_DDL query option to make each DDL statement wait before returning, until the new or changed metadata
has been received by all the Impala nodes. See SYNC_DDL Query Option on page 321 for details.
Examples:
The following example shows how you might use the REFRESH statement after manually adding new HDFS data files
to the Impala data directory for a table:
For more examples of using REFRESH and INVALIDATE METADATA with a combination of Impala and Hive operations,
see Switching Back and Forth Between Impala and Hive on page 60.
Related impala-shell options:
The impala-shell option -r issues an INVALIDATE METADATA statement when starting up the shell, effectively
performing a REFRESH of all tables. Due to the expense of reloading the metadata for all tables, the impala-shell
-r option is not recommended for day-to-day use in a production environment. (This option was mainly intended as
a workaround for synchronization issues in very old Impala versions.)
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have execute permissions for all
the relevant directories holding table data. (A table could have data spread across multiple directories, or in unexpected
paths, if it uses partitioning or specifies a LOCATION attribute for individual partitions or the entire table.) Issues with
permissions might not cause an immediate error for this statement, but subsequent statements such as SELECT or
SHOW TABLE STATS could fail.
HDFS considerations:
The REFRESH command checks HDFS permissions of the underlying data files and directories, caching this information
so that a statement can be cancelled immediately if for example the impala user does not have permission to write
to the data directory for the table. Impala reports any lack of write permissions as an INFO message in the log file, in
case that represents an oversight. If you change HDFS permissions to make data readable or writeable by the Impala
user, issue another REFRESH to make Impala aware of the change.
Important: After adding or replacing data in a table used in performance-critical queries, issue a
COMPUTE STATS statement to make sure all statistics are up-to-date. Consider updating statistics for
a table after any INSERT, LOAD DATA, or CREATE TABLE AS SELECT statement in Impala, or after
loading data through Hive and doing a REFRESH table_name in Impala. This technique is especially
important for tables that are very large, used in join queries, or both.
Amazon S3 considerations:
The REFRESH and INVALIDATE METADATA statements also cache metadata for tables where the data resides in the
Amazon Simple Storage Service (S3). In particular, issue a REFRESH for a table after adding or removing files in the
associated S3 data directory. See Using Impala to Query the Amazon S3 Filesystem on page 566 for details about working
with S3 tables.
Cancellation: Cannot be cancelled.
Related information:
Overview of Impala Metadata and the Metastore on page 21, INVALIDATE METADATA Statement on page 271
The ability to grant or revoke SELECT privilege on specific columns is available in CDH 5.5 / Impala 2.3 and higher. See
https://www.cloudera.com/documentation/enterprise/latest/topics/sg_hive_sql.html for details.
Required privileges:
Only administrative users (those with ALL privileges on the server, defined in the Sentry policy file) can use this
statement.
Compatibility:
• The Impala GRANT and REVOKE statements are available in CDH 5.2 and higher.
• In CDH 5.1 and higher, Impala makes use of any roles and privileges specified by the GRANT and REVOKE statements
in Hive, when your system is configured to use the Sentry service instead of the file-based policy mechanism.
• The Impala GRANT and REVOKE statements do not require the ROLE keyword to be repeated before each role
name, unlike the equivalent Hive statements.
• Currently, each Impala GRANT or REVOKE statement can only grant or revoke a single privilege to or from a single
role.
Cancellation: Cannot be cancelled.
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Related information:
Enabling Sentry Authorization for Impala on page 95, GRANT Statement (CDH 5.2 or higher only) on page 263 CREATE
ROLE Statement (CDH 5.2 or higher only) on page 235, DROP ROLE Statement (CDH 5.2 or higher only) on page 254,
SHOW Statement on page 322
SELECT Statement
The SELECT statement performs queries, retrieving data from one or more tables and producing result sets consisting
of rows and columns.
The Impala INSERT statement also typically ends with a SELECT statement, to define data to copy from one table to
another.
Syntax:
• An optional WITH clause before the SELECT keyword, to define a subquery whose name or column names can
be referenced from later in the main query. This clause lets you abstract repeated clauses, such as aggregation
functions, that are referenced multiple times in the same query.
• By default, one DISTINCT clause per query. See DISTINCT Operator on page 301 for details. See
APPX_COUNT_DISTINCT Query Option (CDH 5.2 or higher only) on page 306 for a query option to allow multiple
COUNT(DISTINCT) impressions in the same query.
• Subqueries in a FROM clause. In CDH 5.2 / Impala 2.0 and higher, subqueries can also go in the WHERE clause, for
example with the IN(), EXISTS, and NOT EXISTS operators.
• WHERE, GROUP BY, HAVING clauses.
• ORDER BY. Prior to Impala 1.4.0, Impala required that queries using an ORDER BY clause also include a LIMIT
clause. In Impala 1.4.0 and higher, this restriction is lifted; sort operations that would exceed the Impala memory
limit automatically use a temporary disk work area to perform the sort.
• Impala supports a wide variety of JOIN clauses. Left, right, semi, full, and outer joins are supported in all Impala
versions. The CROSS JOIN operator is available in Impala 1.2.2 and higher. During performance tuning, you can
override the reordering of join clauses that Impala does internally by including the keyword STRAIGHT_JOIN
immediately after the SELECT and any DISTINCT or ALL keywords.
See Joins in Impala SELECT Statements on page 282 for details and examples of join queries.
• UNION ALL.
• LIMIT.
• External tables.
• Relational operators such as greater than, less than, or equal to.
• Arithmetic operators such as addition or subtraction.
• Logical/Boolean operators AND, OR, and NOT. Impala does not support the corresponding symbols &&, ||, and !.
• Common SQL built-in functions such as COUNT, SUM, CAST, LIKE, IN, BETWEEN, and COALESCE. Impala specifically
supports built-ins described in Impala Built-In Functions on page 338.
Impala queries ignore files with extensions commonly used for temporary work files by Hadoop tools. Any files with
extensions .tmp or .copying are not considered part of the Impala table. The suffix matching is case-insensitive, so
for example Impala ignores both .copying and .COPYING suffixes.
Security considerations:
If these statements in your environment contain sensitive literal values such as credit card numbers or tax identifiers,
Impala can redact this sensitive information when displaying the statements in log files and other administrative
contexts. See http://www.cloudera.com/content/cloudera/en/documentation/core/latest/topics/sg_redaction.html
for details.
Cancellation: Can be cancelled. To cancel this statement, use Ctrl-C from the impala-shell interpreter, the Cancel
button from the Watch page in Hue, Actions > Cancel from the Queries list in Cloudera Manager, or Cancel from the
list of in-flight queries (for a particular node) on the Queries tab in the Impala web UI (port 25000).
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read permissions for the files
in all applicable directories in all source tables, and read and execute permissions for the relevant data directories. (A
SELECT operation could read files from multiple different HDFS directories if the source table is partitioned.) If a query
attempts to read a data file and is unable to because of an HDFS permission error, the query halts and does not return
any further results.
Related information:
The SELECT syntax is so extensive that it forms its own category of statements: queries. The other major classifications
of SQL statements are data definition language (see DDL Statements on page 214) and data manipulation language (see
DML Statements on page 215).
Because the focus of Impala is on fast queries with interactive response times over huge data sets, query performance
and scalability are important considerations. See Tuning Impala for Performance on page 479 and Scalability
Considerations for Impala on page 516 for details.
The ON clause is a general way to compare columns across the two tables, even if the column names are different. The
USING clause is a shorthand notation for specifying the join columns, when the column names are the same in both
tables. You can code equivalent WHERE clauses that compare the columns, instead of ON or USING clauses, but that
practice is not recommended because mixing the join comparisons with other filtering clauses is typically less readable
and harder to maintain.
Queries with a comma-separated list of tables and subqueries are known as SQL-89 style joins. In these queries, the
equality comparisons between columns of the joined tables go in the WHERE clause alongside other kinds of comparisons.
This syntax is easy to learn, but it is also easy to accidentally remove a WHERE clause needed for the join to work
correctly.
Self-joins:
Impala can do self-joins, for example to join on two different columns in the same table to represent parent-child
relationships or other tree-structured data. There is no explicit syntax for this; just use the same table name for both
the left-hand and right-hand table, and assign different table aliases to use when referring to the fully qualified column
names:
Cartesian joins:
To avoid producing huge result sets by mistake, Impala does not allow Cartesian joins of the form:
If you intend to join the tables based on common values, add ON or WHERE clauses to compare columns across the
tables. If you truly intend to do a Cartesian join, use the CROSS JOIN keyword as the join operator. The CROSS JOIN
form does not use any ON clause, because it produces a result set with all combinations of rows from the left-hand and
right-hand tables. The result set can still be filtered by subsequent WHERE clauses. For example:
An outer join retrieves all rows from the left-hand table, or the right-hand table, or both; wherever there is no matching
data in the table on the other side of the join, the corresponding columns in the result set are set to NULL. To perform
an outer join, include the OUTER keyword in the join operator, along with either LEFT, RIGHT, or FULL:
For outer joins, Impala requires SQL-92 syntax; that is, the JOIN keyword instead of comma-separated table names.
Impala does not support vendor extensions such as (+) or *= notation for doing outer joins with SQL-89 query syntax.
Equijoins and Non-Equijoins:
By default, Impala requires an equality comparison between the left-hand and right-hand tables, either through ON,
USING, or WHERE clauses. These types of queries are classified broadly as equijoins. Inner, outer, full, and semi joins
can all be equijoins based on the presence of equality tests between columns in the left-hand and right-hand tables.
In Impala 1.2.2 and higher, non-equijoin queries are also possible, with comparisons such as != or < between the join
columns. These kinds of queries require care to avoid producing huge result sets that could exceed resource limits.
Once you have planned a non-equijoin query that produces a result set of acceptable size, you can code the query
using the CROSS JOIN operator, and add the extra comparisons in the WHERE clause:
In CDH 5.5 / Impala 2.3 and higher, additional non-equijoin queries are possible due to the addition of nested loop
joins. These queries typically involve SEMI JOIN, ANTI JOIN, or FULL OUTER JOIN clauses. Impala sometimes also
uses nested loop joins internally when evaluating OUTER JOIN queries involving complex type columns. Query phases
involving nested loop joins do not use the spill-to-disk mechanism if they exceed the memory limit. Impala decides
internally when to use each join mechanism; you cannot specify any query hint to choose between the nested loop
join or the original hash join algorithm.
Semi-joins:
Semi-joins are a relatively rarely used variation. With the left semi-join, only data from the left-hand table is returned,
for rows where there is matching data in the right-hand table, based on comparisons between join columns in ON or
WHERE clauses. Only one instance of each row from the left-hand table is returned, regardless of how many matching
rows exist in the right-hand table. A right semi-join (available in Impala 2.0 and higher) reverses the comparison and
returns data from the right-hand table.
SELECT t1.c1, t1.c2, t1.c2 FROM t1 LEFT SEMI JOIN t2 ON t1.id = t2.id;
-- 'NATURAL' is interpreted as an alias for 't1' and Impala attempts an inner join,
-- resulting in an error because inner joins require explicit comparisons between columns.
SELECT t1.c1, t2.c2 FROM t1 NATURAL JOIN t2;
ERROR: NotImplementedException: Join with 't2' requires at least one conjunctive equality
predicate.
To perform a Cartesian product between two tables, use a CROSS JOIN.
-- If you expect the tables to have identically named columns with matching values,
-- list the corresponding column names in a USING clause.
SELECT t1.c1, t2.c2 FROM t1 JOIN t2 USING (id, type_flag, name, address);
Note: Impala can join tables of different file formats, including Impala-managed tables and HBase
tables. For example, you might keep small dimension tables in HBase, for convenience of single-row
lookups and updates, and for the larger fact tables use Parquet or other binary file format
optimized for scan operations. Then, you can issue a join query to cross-reference the fact tables
with the dimension tables.
• When data is normalized, a technique for reducing data duplication by dividing it across multiple tables. This kind
of organization is often found in data that comes from traditional relational database systems. For example,
instead of repeating some long string such as a customer name in multiple tables, each table might contain a
numeric customer ID. Queries that need to display the customer name could “join” the table that specifies which
customer ID corresponds to which name.
• When certain columns are rarely needed for queries, so they are moved into separate tables to reduce overhead
for common queries. For example, a biography field might be rarely needed in queries on employee data. Putting
that field in a separate table reduces the amount of I/O for common queries on employee addresses or phone
numbers. Queries that do need the biography column can retrieve it by performing a join with that separate
table.
• In CDH 5.5 / Impala 2.3 or higher, when referring to complex type columns in queries. See Complex Types (CDH
5.5 or higher only) on page 156 for details.
When comparing columns with the same names in ON or WHERE clauses, use the fully qualified names such as
db_name.table_name, or assign table aliases, column aliases, or both to make the code more compact and
understandable:
Note:
Performance for join queries is a crucial aspect for Impala, because complex join queries are
resource-intensive operations. An efficient join query produces much less network traffic and CPU
overhead than an inefficient one. For best results:
• Make sure that both table and column statistics are available for all the tables involved in a join
query, and especially for the columns referenced in any join conditions. Impala uses the statistics
to automatically deduce an efficient join order. Use SHOW TABLE STATS table_name and
SHOW COLUMN STATS table_name to check if statistics are already present. Issue the COMPUTE
STATS table_name for a nonpartitioned table, or (in Impala 2.1.0 and higher) COMPUTE
INCREMENTAL STATS table_name for a partitioned table, to collect the initial statistics at both
the table and column levels, and to keep the statistics up to date after any substantial INSERT
or LOAD DATA operations.
• If table or column statistics are not available, join the largest table first. You can check the existence
of statistics with the SHOW TABLE STATS table_name and SHOW COLUMN STATS table_name
statements.
• If table or column statistics are not available, join subsequent tables according to which table
has the most selective filter, based on overall size and WHERE clauses. Joining the table with the
most selective filter results in the fewest number of rows being returned.
For more information and examples of performance for join queries, see Performance Considerations
for Join Queries on page 482.
To control the result set from a join query, include the names of corresponding column names in both tables in an ON
or USING clause, or by coding equality comparisons for those columns in the WHERE clause.
One potential downside of joins is the possibility of excess resource usage in poorly constructed queries. Impala imposes
restrictions on join queries to guard against such issues. To minimize the chance of runaway queries on large data sets,
Impala requires every join query to contain at least one equality predicate between the columns of the various tables.
For example, if T1 contains 1000 rows and T2 contains 1,000,000 rows, a query SELECT columns FROM t1 JOIN
t2 could return up to 1 billion rows (1000 * 1,000,000); Impala requires that the query include a clause such as ON
t1.c1 = t2.c2 or WHERE t1.c1 = t2.c2.
Because even with equality clauses, the result set can still be large, as we saw in the previous example, you might use
a LIMIT clause to return a subset of the results:
Or you might use additional comparison operators or aggregation functions to condense a large result set into a smaller
set of values:
[localhost:21000] > -- Find the names of customers who live in one particular town.
[localhost:21000] > select distinct c_last_name from customer, customer_address where
c_customer_sk = ca_address_sk
and ca_city = "Green Acres";
+---------------+
| c_last_name |
+---------------+
| Hensley |
| Pearson |
| Mayer |
| Montgomery |
| Ricks |
...
| Barrett |
| Price |
| Hill |
| Hansen |
| Meeks |
+---------------+
Returned 332 row(s) in 0.97s
[localhost:21000] > -- See how many different customers in this town have names starting
with "A".
[localhost:21000] > select count(distinct c_last_name) from customer, customer_address
where
c_customer_sk = ca_address_sk
and ca_city = "Green Acres"
and substr(c_last_name,1,1) = "A";
+-----------------------------+
| count(distinct c_last_name) |
+-----------------------------+
| 12 |
+-----------------------------+
Returned 1 row(s) in 1.00s
Because a join query can involve reading large amounts of data from disk, sending large amounts of data across the
network, and loading large amounts of data into memory to do the comparisons and filtering, you might do
benchmarking, performance analysis, and query tuning to find the most efficient join queries for your data set, hardware
capacity, network configuration, and cluster workload.
The two categories of joins in Impala are known as partitioned joins and broadcast joins. If inaccurate table or column
statistics, or some quirk of the data distribution, causes Impala to choose the wrong mechanism for a particular join,
consider using query hints as a temporary workaround. For details, see Query Hints in Impala SELECT Statements on
page 301.
Examples:
The following examples refer to these simple tables containing small sets of integers:
The following example demonstrates an anti-join, returning the values from T1 that do not exist in T2 (in this case,
the odd numbers 1, 3, and 5):
Related information:
See these tutorials for examples of different kinds of joins:
• Cross Joins and Cartesian Products with the CROSS JOIN Operator on page 61
ORDER BY Clause
The familiar ORDER BY clause of a SELECT statement sorts the result set based on the values from one or more
columns.
For distributed queries, this is a relatively expensive operation, because the entire result set must be produced and
transferred to one node before the sorting can happen. This can require more memory capacity than a query without
ORDER BY. Even if the query takes approximately the same time to finish with or without the ORDER BY clause,
subjectively it can appear slower because no results are available until all processing is finished, rather than results
coming back gradually as rows matching the WHERE clause are found. Therefore, if you only need the first N results
from the sorted result set, also include the LIMIT clause, which reduces network overhead and the memory requirement
on the coordinator node.
Note:
In Impala 1.4.0 and higher, the LIMIT clause is now optional (rather than required) for queries that
use the ORDER BY clause. Impala automatically uses a temporary disk work area to perform the sort
if the sort operation would otherwise exceed the Impala memory limit for a particular DataNode.
Syntax:
The full syntax for the ORDER BY clause is:
ORDER BY col_ref [, col_ref ...] [ASC | DESC] [NULLS FIRST | NULLS LAST]
Although the most common usage is ORDER BY column_name, you can also specify ORDER BY 1 to sort by the first
column of the result set, ORDER BY 2 to sort by the second column, and so on. The number must be a numeric literal,
not some other kind of constant expression. (If the argument is some other expression, even a STRING value, the query
succeeds but the order of results is undefined.)
ORDER BY column_number can only be used when the query explicitly lists the columns in the SELECT list, not with
SELECT * queries.
CREATE TABLE games (id BIGINT, score ARRAY <BIGINT>) STORED AS PARQUET;
...use LOAD DATA to load externally created Parquet files into the table...
SELECT id FROM games ORDER BY score DESC;
ERROR: AnalysisException: ORDER BY expression 'score' with complex type 'ARRAY<BIGINT>'
is not supported.
Examples:
The following query retrieves the user ID and score, only for scores greater than one million, with the highest scores
for each user listed first. Because the individual array elements are now represented as separate rows in the result
set, they can be used in the ORDER BY clause, referenced using the ITEM pseudocolumn that represents each array
element.
The following queries use similar ORDER BY techniques with variations of the GAMES table, where the complex type
is an ARRAY containing STRUCT or MAP elements to represent additional details about each game that was played. For
an array of structures, the fields of the structure are referenced as ITEM.field_name. For an array of maps, the keys
and values within each array element are referenced as ITEM.KEY and ITEM.VALUE.
CREATE TABLE games2 (id BIGINT, play array < struct <game_name: string, score: BIGINT,
high_score: boolean> >) STORED AS PARQUET
...use LOAD DATA to load externally created Parquet files into the table...
SELECT id, item.game_name, item.score FROM games2, games2.play
WHERE item.score > 1000000
ORDER BY id, item.score DESC;
CREATE TABLE games3 (id BIGINT, play ARRAY < MAP <STRING, BIGINT> >) STORED AS PARQUET;
...use LOAD DATA to load externally created Parquet files into the table...
SELECT id, info.key AS k, info.value AS v from games3, games3.play AS plays,
games3.play.item AS info
WHERE info.KEY = 'score' AND info.VALUE > 1000000
ORDER BY id, info.value desc;
Usage notes:
Although the LIMIT clause is now optional on ORDER BY queries, if your query only needs some number of rows that
you can predict in advance, use the LIMIT clause to reduce unnecessary processing. For example, if the query has a
clause LIMIT 10, each data node sorts its portion of the relevant result set and only returns 10 rows to the coordinator
node. The coordinator node picks the 10 highest or lowest row values out of this small intermediate result set.
If an ORDER BY clause is applied to an early phase of query processing, such as a subquery or a view definition, Impala
ignores the ORDER BY clause. To get ordered results from a subquery or view, apply an ORDER BY clause to the
outermost or final SELECT level.
ORDER BY is often used in combination with LIMIT to perform “top-N” queries:
ORDER BY is sometimes used in combination with OFFSET and LIMIT to paginate query results, although it is relatively
inefficient to issue multiple queries like this against the large tables typically used with Impala:
Internal details:
Impala sorts the intermediate results of an ORDER BY clause in memory whenever practical. In a cluster of N DataNodes,
each node sorts roughly 1/Nth of the result set, the exact proportion varying depending on how the data matching
the query is distributed in HDFS.
If the size of the sorted intermediate result set on any DataNode would cause the query to exceed the Impala memory
limit, Impala sorts as much as practical in memory, then writes partially sorted data to disk. (This technique is known
in industry terminology as “external sorting” and “spilling to disk”.) As each 8 MB batch of data is written to disk, Impala
frees the corresponding memory to sort a new 8 MB batch of data. When all the data has been processed, a final merge
sort operation is performed to correctly order the in-memory and on-disk results as the result set is transmitted back
to the coordinator node. When external sorting becomes necessary, Impala requires approximately 60 MB of RAM at
a minimum for the buffers needed to read, write, and sort the intermediate results. If more RAM is available on the
DataNode, Impala will use the additional RAM to minimize the amount of disk I/O for sorting.
This external sort technique is used as appropriate on each DataNode (possibly including the coordinator node) to sort
the portion of the result set that is processed on that node. When the sorted intermediate results are sent back to the
coordinator node to produce the final result set, the coordinator node uses a merge sort technique to produce a final
sorted result set without using any extra resources on the coordinator node.
Configuration for disk usage:
By default, intermediate files used during large sort, join, aggregation, or analytic function operations are stored in
the directory /tmp/impala-scratch . These files are removed when the operation finishes. (Multiple concurrent
queries can perform operations that use the “spill to disk” technique, without any name conflicts for these temporary
files.) You can specify a different location by starting the impalad daemon with the
--scratch_dirs="path_to_directory" configuration option or the equivalent configuration option in the Cloudera
Manager user interface. You can specify a single directory, or a comma-separated list of directories. The scratch
directories must be on the local filesystem, not in HDFS. You might specify different directory paths for different hosts,
depending on the capacity and speed of the available storage devices. In CDH 5.5 / Impala 2.3 or higher, Impala
successfully starts (with a warning written to the log) if it cannot create or read and write files in one of the scratch
directories. If there is less than 1 GB free on the filesystem where that directory resides, Impala still runs, but writes a
warning message to its log. If Impala encounters an error reading or writing files in a scratch directory during a query,
Impala logs the error and the query fails.
Sorting considerations: Although you can specify an ORDER BY clause in an INSERT ... SELECT statement, any
ORDER BY clause is ignored and the results are not necessarily sorted. An INSERT ... SELECT operation potentially
creates many different data files, prepared on different data nodes, and therefore the notion of the data being stored
in sorted order is impractical.
An ORDER BY clause without an additional LIMIT clause is ignored in any view definition. If you need to sort the entire
result set from a view, use an ORDER BY clause in the SELECT statement that queries the view. You can still make a
simple “top 10” report by combining the ORDER BY and LIMIT clauses in the same view definition:
| 5 |
| 6 |
| 7 |
| 8 |
| 9 |
+---+
[localhost:21000] > create view top_3_view as select x from unsorted order by x limit
3;
[localhost:21000] > select x from top_3_view; -- ORDER BY and LIMIT together in view
definition are preserved.
+---+
| x |
+---+
| 1 |
| 2 |
| 3 |
+---+
With the lifting of the requirement to include a LIMIT clause in every ORDER BY query (in Impala 1.4 and higher):
• Now the use of scratch disk space raises the possibility of an “out of disk space” error on a particular DataNode,
as opposed to the previous possibility of an “out of memory” error. Make sure to keep at least 1 GB free on the
filesystem used for temporary sorting work.
• The query options DEFAULT_ORDER_BY_LIMIT and ABORT_ON_DEFAULT_LIMIT_EXCEEDED, which formerly
controlled the behavior of ORDER BY queries with no limit specified, are now ignored.
In Impala 1.2.1 and higher, all NULL values come at the end of the result set for ORDER BY ... ASC queries, and at
the beginning of the result set for ORDER BY ... DESC queries. In effect, NULL is considered greater than all other
values for sorting purposes. The original Impala behavior always put NULL values at the end, even for ORDER BY ...
DESC queries. The new behavior in Impala 1.2.1 makes Impala more compatible with other popular database systems.
In Impala 1.2.1 and higher, you can override or specify the sorting behavior for NULL by adding the clause NULLS
FIRST or NULLS LAST at the end of the ORDER BY clause.
+------+
| 3 |
| 2 |
| 1 |
| NULL |
| NULL |
+------+
Related information:
See SELECT Statement on page 280 for further examples of queries with the ORDER BY clause.
Analytic functions use the ORDER BY clause in a different context to define the sequence in which rows are analyzed.
See Impala Analytic Functions on page 429 for details.
GROUP BY Clause
Specify the GROUP BY clause in queries that use aggregation functions, such as COUNT(), SUM(), AVG(), MIN(), and
MAX(). Specify in the GROUP BY clause the names of all the columns that do not participate in the aggregation operation.
select
ss_item_sk as Item,
count(ss_item_sk) as Times_Purchased,
sum(ss_quantity) as Total_Quantity_Purchased
from store_sales
group by ss_item_sk
order by sum(ss_quantity) desc
limit 5;
+-------+-----------------+--------------------------+
| item | times_purchased | total_quantity_purchased |
+-------+-----------------+--------------------------+
| 9325 | 372 | 19072 |
| 4279 | 357 | 18501 |
| 7507 | 371 | 18475 |
| 5953 | 369 | 18451 |
| 16753 | 375 | 18446 |
+-------+-----------------+--------------------------+
The HAVING clause lets you filter the results of aggregate functions, because you cannot refer to those expressions in
the WHERE clause. For example, to find the 5 lowest-selling items that were included in at least 100 sales transactions,
we could use this query:
select
ss_item_sk as Item,
count(ss_item_sk) as Times_Purchased,
sum(ss_quantity) as Total_Quantity_Purchased
from store_sales
group by ss_item_sk
having times_purchased >= 100
order by sum(ss_quantity)
limit 5;
+-------+-----------------+--------------------------+
| item | times_purchased | total_quantity_purchased |
+-------+-----------------+--------------------------+
| 13943 | 105 | 4087 |
| 2992 | 101 | 4176 |
| 4773 | 107 | 4204 |
| 14350 | 103 | 4260 |
| 11956 | 102 | 4275 |
+-------+-----------------+--------------------------+
When performing calculations involving scientific or financial data, remember that columns with type FLOAT or DOUBLE
are stored as true floating-point numbers, which cannot precisely represent every possible fractional value. Thus, if
you include a FLOAT or DOUBLE column in a GROUP BY clause, the results might not precisely match literal values in
your query or from an original Text data file. Use rounding operations, the BETWEEN operator, or another arithmetic
technique to match floating-point values that are “near” literal values you expect. For example, this query on the
ss_wholesale_cost column returns cost values that are close but not identical to the original figures that were
entered as decimal fractions.
Notice how wholesale cost values originally entered as decimal fractions such as 96.94 and 98.38 are slightly larger
or smaller in the result set, due to precision limitations in the hardware floating-point types. The imprecise representation
of FLOAT and DOUBLE values is why financial data processing systems often store currency using data types that are
less space-efficient but avoid these types of rounding errors.
Related information:
SELECT Statement on page 280, Impala Aggregate Functions on page 403
HAVING Clause
Performs a filter operation on a SELECT query, by examining the results of aggregation functions rather than testing
each individual table row. Therefore, it is always used in conjunction with a function such as COUNT(), SUM(), AVG(),
MIN(), or MAX(), and typically with the GROUP BY clause also.
Restrictions:
The filter expression in the HAVING clause cannot include a scalar subquery.
Related information:
SELECT Statement on page 280, GROUP BY Clause on page 292, Impala Aggregate Functions on page 403
LIMIT Clause
The LIMIT clause in a SELECT query sets a maximum number of rows for the result set. Pre-selecting the maximum
size of the result set helps Impala to optimize memory usage while processing a distributed query.
Syntax:
LIMIT constant_integer_expression
The argument to the LIMIT clause must evaluate to a constant value. It can be a numeric literal, or another kind of
numeric expression involving operators, casts, and function return values. You cannot refer to a column or use a
subquery.
Usage notes:
This clause is useful in contexts such as:
• To return exactly N items from a top-N query, such as the 10 highest-rated items in a shopping category or the
50 hostnames that refer the most traffic to a web site.
• To demonstrate some sample values from a table or a particular query. (To display some arbitrary items, use a
query with no ORDER BY clause. An ORDER BY clause causes additional memory and/or disk usage during the
query.)
• To keep queries from returning huge result sets by accident if a table is larger than expected, or a WHERE clause
matches more rows than expected.
Originally, the value for the LIMIT clause had to be a numeric literal. In Impala 1.2.1 and higher, it can be a numeric
expression.
Prior to Impala 1.4.0, Impala required any query including an ORDER BY clause to also use a LIMIT clause. In Impala
1.4.0 and higher, the LIMIT clause is optional for ORDER BY queries. In cases where sorting a huge result set requires
enough memory to exceed the Impala memory limit for a particular node, Impala automatically uses a temporary disk
work area to perform the sort operation.
See ORDER BY Clause on page 287 for details.
In Impala 1.2.1 and higher, you can combine a LIMIT clause with an OFFSET clause to produce a small result set that
is different from a top-N query, for example, to return items 11 through 20. This technique can be used to simulate
“paged” results. Because Impala queries typically involve substantial amounts of I/O, use this technique only for
compatibility in cases where you cannot rewrite the application logic. For best performance and scalability, wherever
practical, query as many items as you expect to need, cache them on the application side, and display small groups of
results to users using application logic.
Restrictions:
Correlated subqueries used in EXISTS and IN operators cannot include a LIMIT clause.
Examples:
The following example shows how the LIMIT clause caps the size of the result set, with the limit being applied after
any other clauses such as WHERE.
For top-N and bottom-N queries, you use the ORDER BY and LIMIT clauses together:
[localhost:21000] > select x as "Top 3" from numbers order by x desc limit 3;
+-------+
| top 3 |
+-------+
| 5 |
| 4 |
| 3 |
+-------+
[localhost:21000] > select x as "Bottom 3" from numbers order by x limit 3;
+----------+
| bottom 3 |
+----------+
| 1 |
| 2 |
| 3 |
+----------+
You can use constant values besides integer literals as the LIMIT argument:
OFFSET Clause
The OFFSET clause in a SELECT query causes the result set to start some number of rows after the logical first item.
The result set is numbered starting from zero, so OFFSET 0 produces the same result as leaving out the OFFSET clause.
Always use this clause in combination with ORDER BY (so that it is clear which item should be first, second, and so on)
and LIMIT (so that the result set covers a bounded range, such as items 0-9, 100-199, and so on).
In Impala 1.2.1 and higher, you can combine a LIMIT clause with an OFFSET clause to produce a small result set that
is different from a top-N query, for example, to return items 11 through 20. This technique can be used to simulate
“paged” results. Because Impala queries typically involve substantial amounts of I/O, use this technique only for
compatibility in cases where you cannot rewrite the application logic. For best performance and scalability, wherever
practical, query as many items as you expect to need, cache them on the application side, and display small groups of
results to users using application logic.
Examples:
The following example shows how you could run a “paging” query originally written for a traditional database application.
Because typical Impala queries process megabytes or gigabytes of data and read large data files from disk each time,
it is inefficient to run a separate query to retrieve each small group of items. Use this technique only for compatibility
while porting older applications, then rewrite the application code to use a single query with a large result set, and
display pages of results from the cached result set.
| 2 |
| 3 |
| 4 |
| 5 |
+----+
[localhost:21000] > select x from numbers order by x limit 5 offset 5;
+----+
| x |
+----+
| 6 |
| 7 |
| 8 |
| 9 |
| 10 |
+----+
UNION Clause
The UNION clause lets you combine the result sets of multiple queries. By default, the result sets are combined as if
the DISTINCT operator was applied.
Syntax:
Usage notes:
The UNION keyword by itself is the same as UNION DISTINCT. Because eliminating duplicates can be a memory-intensive
process for a large result set, prefer UNION ALL where practical. (That is, when you know the different queries in the
union will not produce any duplicates, or where the duplicate values are acceptable.)
When an ORDER BY clause applies to a UNION ALL or UNION query, in Impala 1.4 and higher, the LIMIT clause is no
longer required. To make the ORDER BY and LIMIT clauses apply to the entire result set, turn the UNION query into
a subquery, SELECT from the subquery, and put the ORDER BY clause at the end, outside the subquery.
Examples:
First, set up some sample data, including duplicate 1 values:
This example shows how UNION ALL returns all rows from both queries, without any additional filtering to eliminate
duplicates. For the large result sets common with Impala queries, this is the most memory-efficient technique.
+---+
Returned 8 row(s) in 0.42s
[localhost:21000] > select * from (select x from few_ints union all select x from
few_ints) as t1 order by x;
+---+
| x |
+---+
| 1 |
| 1 |
| 1 |
| 1 |
| 2 |
| 2 |
| 3 |
| 3 |
+---+
Returned 8 row(s) in 0.53s
[localhost:21000] > select x from few_ints union all select 10;
+----+
| x |
+----+
| 10 |
| 1 |
| 1 |
| 2 |
| 3 |
+----+
Returned 5 row(s) in 0.38s
This example shows how the UNION clause without the ALL keyword condenses the result set to eliminate all duplicate
values, making the query take more time and potentially more memory. The extra processing typically makes this
technique not recommended for queries that return result sets with millions or billions of values.
[localhost:21000] > select x from few_ints union select x+1 from few_ints;
+---+
| x |
+---+
| 3 |
| 4 |
| 1 |
| 2 |
+---+
Returned 4 row(s) in 0.51s
[localhost:21000] > select x from few_ints union select 10;
+----+
| x |
+----+
| 2 |
| 10 |
| 1 |
| 3 |
+----+
Returned 4 row(s) in 0.49s
[localhost:21000] > select * from (select x from few_ints union select x from few_ints)
as t1 order by x;
+---+
| x |
+---+
| 1 |
| 2 |
| 3 |
+---+
Returned 3 row(s) in 0.53s
A scalar subquery produces a result set with a single row containing a single column, typically produced by an aggregation
function such as MAX() or SUM(). This single result value can be substituted in scalar contexts such as arguments to
comparison operators. If the result set is empty, the value of the scalar subquery is NULL. For example, the following
query finds the maximum value of T2.Y and then substitutes that value into the WHERE clause of the outer block that
queries T1:
Uncorrelated subqueries do not refer to any tables from the outer block of the query. The same value or set of values
produced by the subquery is used when evaluating each row from the outer query block. In this example, the subquery
returns an arbitrary number of values from T2.Y, and each value of T1.X is tested for membership in that same set
of values:
Correlated subqueries compare one or more values from the outer query block to values referenced in the WHERE
clause of the subquery. Each row evaluated by the outer WHERE clause can be evaluated using a different set of values.
These kinds of subqueries are restricted in the kinds of comparisons they can do between columns of the inner and
outer tables. (See the following Restrictions item.)
For example, the following query finds all the employees with salaries that are higher than average for their department.
The subquery potentially computes a different AVG() value for each employee.
Syntax:
Subquery in the FROM clause:
comparison_operator is a numeric comparison such as =, <=, !=, and so on, or a string comparison operator such
as LIKE or REGEXP.
Although you can use non-equality comparison operators such as < or >=, the subquery must include at least one
equality comparison between the columns of the inner and outer query blocks.
All syntax is available for both correlated and uncorrelated queries, except that the NOT EXISTS clause cannot be
used with an uncorrelated subquery.
Impala subqueries can be nested arbitrarily deep.
Standards compliance: Introduced in SQL:1999.
Examples:
This example illustrates how subqueries can be used in the FROM clause to organize the table names, column names,
and column values by producing intermediate result sets, especially for join queries.
These examples show how a query can test for the existence of values in a separate table using the EXISTS() operator
with a subquery.
The following examples show how a value can be compared against a set of values returned by a subquery.
SELECT count(x) FROM t1 WHERE EXISTS(SELECT 1 FROM t2 WHERE t1.x = t2.y * 10);
The following examples demonstrate scalar subqueries. When a subquery is known to return a single value, you can
substitute it where you would normally put a constant value.
Usage notes:
If the same table is referenced in both the outer and inner query blocks, construct a table alias in the outer query block
and use a fully qualified name to distinguish the inner and outer table references:
SELECT * FROM t1 one WHERE id IN (SELECT parent FROM t1 two WHERE t1.parent = t2.id);
The STRAIGHT_JOIN hint affects the join order of table references in the query block containing the hint. It does not
affect the join order of nested queries, such as views, inline views, or WHERE-clause subqueries. To use this hint for
performance tuning of complex queries, apply the hint to all query blocks that need a fixed join order.
Internal details:
Internally, subqueries involving IN, NOT IN, EXISTS, or NOT EXISTS clauses are rewritten into join queries. Depending
on the syntax, the subquery might be rewritten to an outer join, semi join, cross join, or anti join.
A query is processed differently depending on whether the subquery calls any aggregation functions. There are correlated
and uncorrelated forms, with and without calls to aggregation functions. Each of these four categories is rewritten
differently.
Column statistics considerations:
Because queries that include correlated and uncorrelated subqueries in the WHERE clause are written into join queries,
to achieve best performance, follow the same guidelines for running the COMPUTE STATS statement as you do for
tables involved in regular join queries. Run the COMPUTE STATS statement for each associated tables after loading or
substantially changing the data in that table. See Table and Column Statistics on page 489 for details.
Added in: Subqueries are substantially enhanced starting in Impala 2.0 for CDH 4, and CDH 5.2.0. Now, they can be
used in the WHERE clause, in combination with clauses such as EXISTS and IN, rather than just in the FROM clause.
Restrictions:
The initial Impala support for nested subqueries addresses the most common use cases. Some restrictions remain:
• Although you can use subqueries in a query involving UNION or UNION ALL in Impala 2.1.0 and higher, currently
you cannot construct a union of two subqueries (for example, in the argument of an IN or EXISTS operator).
• Subqueries returning scalar values cannot be used with the operators ANY or ALL. (Impala does not currently have
a SOME operator, but if it did, the same restriction would apply.)
• For the EXISTS and NOT EXISTS clauses, any subquery comparing values from the outer query block to another
table must use at least one equality comparison, not exclusively other kinds of comparisons such as less than,
greater than, BETWEEN, or !=.
• Currently, a scalar subquery cannot be used as the first or second argument to the BETWEEN operator.
• A subquery cannot be used inside an OR conjunction. Expressions inside a subquery, for example in the WHERE
clause, can use OR conjunctions; the restriction only applies to parts of the query “above” the subquery.
• Scalar subqueries are only supported in numeric contexts. You cannot use a scalar subquery as an argument to
the LIKE, REGEXP, or RLIKE operators, or compare it to a value of a non-numeric type such as TIMESTAMP or
BOOLEAN.
• You cannot use subqueries with the CASE function to generate the comparison value, the values to be compared
against, or the return value.
• A subquery is not allowed in the filter condition for the HAVING clause. (Strictly speaking, a subquery cannot
appear anywhere outside the WITH, FROM, and WHERE clauses.)
• You must use a fully qualified name (table_name.column_name or
database_name.table_name.column_name) when referring to any column from the outer query block within
a subquery.
WITH Clause
A clause that can be added before a SELECT statement, to define aliases for complicated expressions that are referenced
multiple times within the body of the SELECT. Similar to CREATE VIEW, except that the table and column names
defined in the WITH clause do not persist after the query finishes, and do not conflict with names used in actual tables
or views. Also known as “subquery factoring”.
You can rewrite a query using subqueries to work the same as with the WITH clause. The purposes of the WITH clause
are:
• Convenience and ease of maintenance from less repetition with the body of the query. Typically used with queries
involving UNION, joins, or aggregation functions where the similar complicated expressions are referenced multiple
times.
• SQL code that is easier to read and understand by abstracting the most complex part of the query into a separate
block.
• Improved compatibility with SQL from other database systems that support the same clause (primarily Oracle
Database).
Note:
The Impala WITH clause does not support recursive queries in the WITH, which is supported in
some other database systems.
-- Define 2 subqueries that can be referenced from the body of a longer query.
with t1 as (select 1), t2 as (select 2) insert into tab select * from t1 union all select
* from t2;
-- Define one subquery at the outer level, and another at the inner level as part of
the
-- initial stage of the UNION ALL query.
with t1 as (select 1) (with t2 as (select 2) select * from t2) union all select * from
t1;
DISTINCT Operator
The DISTINCT operator in a SELECT statement filters the result set to remove duplicates:
You can use DISTINCT in combination with an aggregation function, typically COUNT(), to find how many different
values a column contains:
One construct that Impala SQL does not support is using DISTINCT in more than one aggregation function in the same
query. For example, you could not have a single query with both COUNT(DISTINCT c_first_name) and
COUNT(DISTINCT c_last_name) in the SELECT list.
Zero-length strings: For purposes of clauses such as DISTINCT and GROUP BY, Impala considers zero-length strings
(""), NULL, and space to all be different values.
Note:
By default, Impala only allows a single COUNT(DISTINCT columns) expression in each query.
If you do not need precise accuracy, you can produce an estimate of the distinct values for a column
by specifying NDV(column); a query can contain multiple instances of NDV(column). To make Impala
automatically rewrite COUNT(DISTINCT) expressions to NDV(), enable the APPX_COUNT_DISTINCT
query option.
To produce the same result as multiple COUNT(DISTINCT) expressions, you can use the following
technique for queries involving a single table:
Because CROSS JOIN is an expensive operation, prefer to use the NDV() technique wherever practical.
Note:
In contrast with some database systems that always return DISTINCT values in sorted order, Impala
does not do any ordering of DISTINCT values. Always include an ORDER BY clause if you need the
values in alphabetical or numeric sorted order.
Hints are most often used for the most resource-intensive kinds of Impala queries:
• Join queries involving large tables, where intermediate result sets are transmitted across the network to evaluate
the join conditions.
• Inserting into partitioned Parquet tables, where many memory buffers could be allocated on each host to hold
intermediate results for each partition.
Syntax:
You can represent the hints as keywords surrounded by [] square brackets; include the brackets in the text of the SQL
statement.
INSERT insert_clauses
[{SHUFFLE|NOSHUFFLE}]
SELECT remainder_of_query;
In Impala 2.0 and higher, or CDH 5.2 and higher, you can also specify the hints inside comments that use either the /*
*/ or -- notation. Specify a + symbol immediately before the hint name.
INSERT insert_clauses
/* +SHUFFLE|NOSHUFFLE */
SELECT remainder_of_query;
INSERT insert_clauses
-- +SHUFFLE|NOSHUFFLE
SELECT remainder_of_query;
Usage notes:
With both forms of hint syntax, include the STRAIGHT_JOIN keyword immediately after the SELECT and any DISTINCT
or ALL keywords to prevent Impala from reordering the tables in a way that makes the join-related hints ineffective.
The STRAIGHT_JOIN hint affects the join order of table references in the query block containing the hint. It does not
affect the join order of nested queries, such as views, inline views, or WHERE-clause subqueries. To use this hint for
performance tuning of complex queries, apply the hint to all query blocks that need a fixed join order.
To reduce the need to use hints, run the COMPUTE STATS statement against all tables involved in joins, or used as the
source tables for INSERT ... SELECT operations where the destination is a partitioned Parquet table. Do this
operation after loading data or making substantial changes to the data within each table. Having up-to-date statistics
helps Impala choose more efficient query plans without the need for hinting. See Table and Column Statistics on page
489 for details and examples.
To see which join strategy is used for a particular query, examine the EXPLAIN output for that query. See Using the
EXPLAIN Plan for Performance Tuning on page 507 for details and examples.
Hints for join queries:
The [BROADCAST] and [SHUFFLE] hints control the execution strategy for join queries. Specify one of the following
constructs immediately after the JOIN keyword in a query:
• [SHUFFLE] - Makes that join operation use the “partitioned” technique, which divides up corresponding rows
from both tables using a hashing algorithm, sending subsets of the rows to other nodes for processing. (The
keyword SHUFFLE is used to indicate a “partitioned join”, because that type of join is not related to “partitioned
tables”.) Since the alternative “broadcast” join mechanism is the default when table and index statistics are
unavailable, you might use this hint for queries where broadcast joins are unsuitable; typically, partitioned joins
are more efficient for joins between large tables of similar size.
• [BROADCAST] - Makes that join operation use the “broadcast” technique that sends the entire contents of the
right-hand table to all nodes involved in processing the join. This is the default mode of operation when table and
index statistics are unavailable, so you would typically only need it if stale metadata caused Impala to mistakenly
choose a partitioned join operation. Typically, broadcast joins are more efficient in cases where one table is much
smaller than the other. (Put the smaller table on the right side of the JOIN operator.)
Hints for INSERT ... SELECT queries:
When inserting into partitioned tables, especially using the Parquet file format, you can include a hint in the INSERT
statement to fine-tune the overall performance of the operation and its resource usage:
• These hints are available in Impala 1.2.2 and higher.
• You would only use these hints if an INSERT into a partitioned Parquet table was failing due to capacity limits, or
if such an INSERT was succeeding but with less-than-optimal performance.
• To use these hints, put the hint keyword [SHUFFLE] or [NOSHUFFLE] (including the square brackets) after the
PARTITION clause, immediately before the SELECT keyword.
• [SHUFFLE] selects an execution plan that minimizes the number of files being written simultaneously to HDFS,
and the number of memory buffers holding data for individual partitions. Thus it reduces overall resource usage
for the INSERT operation, allowing some INSERT operations to succeed that otherwise would fail. It does involve
some data transfer between the nodes so that the data files for a particular partition are all constructed on the
same node.
• [NOSHUFFLE] selects an execution plan that might be faster overall, but might also produce a larger number of
small data files or exceed capacity limits, causing the INSERT operation to fail. Use [SHUFFLE] in cases where
an INSERT statement fails or runs inefficiently due to all nodes attempting to construct data for all partitions.
• Impala automatically uses the [SHUFFLE] method if any partition key column in the source table, mentioned in
the INSERT ... SELECT query, does not have column statistics. In this case, only the [NOSHUFFLE] hint would
have any effect.
• If column statistics are available for all partition key columns in the source table mentioned in the INSERT ...
SELECT query, Impala chooses whether to use the [SHUFFLE] or [NOSHUFFLE] technique based on the estimated
number of distinct values in those columns and the number of nodes involved in the INSERT operation. In this
case, you might need the [SHUFFLE] or the [NOSHUFFLE] hint to override the execution plan selected by Impala.
Suggestions versus directives:
In early Impala releases, hints were always obeyed and so acted more like directives. Once Impala gained join order
optimizations, sometimes join queries were automatically reordered in a way that made a hint irrelevant. Therefore,
the hints act more like suggestions in Impala 1.2.2 and higher.
To force Impala to follow the hinted execution mechanism for a join query, include the STRAIGHT_JOIN keyword in
the SELECT statement. See Overriding Join Reordering with STRAIGHT_JOIN on page 483 for details. When you use
this technique, Impala does not reorder the joined tables at all, so you must be careful to arrange the join order to put
the largest table (or subquery result set) first, then the smallest, second smallest, third smallest, and so on. This ordering
lets Impala do the most I/O-intensive parts of the query using local reads on the DataNodes, and then reduce the size
of the intermediate result set as much as possible as each subsequent table or subquery result set is joined.
Restrictions:
Queries that include subqueries in the WHERE clause can be rewritten internally as join queries. Currently, you cannot
apply hints to the joins produced by these types of queries.
Because hints can prevent queries from taking advantage of new metadata or improvements in query planning, use
them only when required to work around performance issues, and be prepared to remove them when they are no
longer required, such as after a new Impala release or bug fix.
In particular, the [BROADCAST] and [SHUFFLE] hints are expected to be needed much less frequently in Impala 1.2.2
and higher, because the join order optimization feature in combination with the COMPUTE STATS statement now
automatically choose join order and join mechanism without the need to rewrite the query and add hints. See
Performance Considerations for Join Queries on page 482 for details.
Compatibility:
The hints embedded within -- comments are compatible with Hive queries. The hints embedded within /* */
comments or [ ] square brackets are not recognized by or not compatible with Hive. For example, Hive raises an error
for Impala hints within /* */ comments because it does not recognize the Impala hint names.
Considerations for views:
If you use a hint in the query that defines a view, the hint is preserved when you query the view. Impala internally
rewrites all hints in views to use the -- comment notation, so that Hive can query such views without errors due to
unrecognized hint names.
Examples:
For example, this query joins a large customer table with a small lookup table of less than 100 rows. The right-hand
table can be broadcast efficiently to all nodes involved in the join. Thus, you would use the [broadcast] hint to force
a broadcast join strategy:
This query joins two large tables of unpredictable size. You might benchmark the query with both kinds of hints and
find that it is more efficient to transmit portions of each table to other nodes for processing. Thus, you would use the
[shuffle] hint to force a partitioned join strategy:
For joins involving three or more tables, the hint applies to the tables on either side of that specific JOIN keyword.
The STRAIGHT_JOIN keyword ensures that joins are processed in a predictable order from left to right. For example,
this query joins t1 and t2 using a partitioned join, then joins that result set to t3 using a broadcast join:
Related information:
For more background information about join queries, see Joins in Impala SELECT Statements on page 282. For performance
considerations, see Performance Considerations for Join Queries on page 482.
SET Statement
Specifies values for query options that control the runtime behavior of other statements within the same session.
Syntax:
SET [query_option=option_value]
SET with no arguments returns a result set consisting of all available query options and their current values.
The query option name and any string argument values are case-insensitive.
Each query option has a specific allowed notation for its arguments. Boolean options can be enabled and disabled by
assigning values of either true and false, or 1 and 0. Some numeric options accept a final character signifying the
unit, such as 2g for 2 gigabytes or 100m for 100 megabytes. See Query Options for the SET Statement on page 305 for
the details of each query option.
Usage notes:
MEM_LIMIT is probably the most commonly used query option. You can specify a high value to allow a resource-intensive
query to complete. For testing how queries would work on memory-constrained systems, you might specify an artificially
low value.
Complex type considerations:
Examples:
The following example sets some numeric and some Boolean query options to control usage of memory, disk space,
and timeout periods, then runs a query whose success could depend on the options in effect:
set mem_limit=64g;
set DISABLE_UNSAFE_SPILLS=true;
set parquet_file_size=400m;
set RESERVATION_REQUEST_TIMEOUT=900000;
insert overwrite parquet_table select c1, c2, count(c3) from text_table group by c1,
c2, c3;
Note:
In Impala 2.0 and later, you can set query options directly through the JDBC and ODBC interfaces by
using the SET statement. Formerly, SET was only available as a command within the impala-shell
interpreter.
Related information:
SET Statement on page 304
ABORT_ON_DEFAULT_LIMIT_EXCEEDED Query Option
Now that the ORDER BY clause no longer requires an accompanying LIMIT clause in Impala 1.4.0 and higher, this
query option is deprecated and has no effect.
Type: Boolean; recognized values are 1 and 0, or true and false; any other value interpreted as false
Default: false (shown as 0 in output of SET statement)
When you enable the APPX_COUNT_DISTINCT query option, now the query with multiple COUNT(DISTINCT) works.
The reason this behavior requires a query option is that each COUNT(DISTINCT) is rewritten internally to use the
NDV() function instead, which provides an approximate result rather than a precise count.
| 10 | 7 |
+-------------------+--------------------------+
Related information:
COUNT Function on page 409, DISTINCT Operator on page 301, NDV Function on page 421
BATCH_SIZE Query Option
Number of rows evaluated at a time by SQL operators. Unspecified or a size of 0 uses a predefined default size. Using
a large number improves responsiveness, especially for scan operations, at the cost of a higher memory footprint.
This option is primarily for Cloudera testing, or for use under the direction of Cloudera Support.
Type: numeric
Default: 0 (meaning the predefined default of 1024)
COMPRESSION_CODEC Query Option (CDH 5.2 or higher only)
When Impala writes Parquet data files using the INSERT statement, the underlying compression is controlled by the
COMPRESSION_CODEC query option.
Note: Prior to Impala 2.0, this option was named PARQUET_COMPRESSION_CODEC. In Impala 2.0 and
later, the PARQUET_COMPRESSION_CODEC name is not recognized. Use the more general name
COMPRESSION_CODEC for new code.
Syntax:
SET COMPRESSION_CODEC=codec_name;
The allowed values for this query option are SNAPPY (the default), GZIP, and NONE.
Note: A Parquet file created with COMPRESSION_CODEC=NONE is still typically smaller than the original
data, due to encoding schemes such as run-length encoding and dictionary encoding that are applied
separately from compression.
set compression_codec=gzip;
insert into parquet_table_highly_compressed select * from t1;
set compression_codec=snappy;
insert into parquet_table_compression_plus_fast_queries select * from t1;
set compression_codec=none;
insert into parquet_table_no_compression select * from t1;
set compression_codec=foo;
select * from t1 limit 5;
ERROR: Invalid compression codec: foo
Related information:
For information about how compressing Parquet data files affects query performance, see Snappy and GZip Compression
for Parquet Data Files on page 539.
involving tiny amounts of data. Reducing the overhead for small queries allows Impala to complete them more quickly,
keeping YARN resources, admission control slots, and so on available for data-intensive queries.
Syntax:
SET EXEC_SINGLE_NODE_ROWS_THRESHOLD=number_of_rows
Type: numeric
Default: 100
Usage notes: Typically, you increase the default value to make this optimization apply to more queries. If incorrect or
corrupted table and column statistics cause Impala to apply this optimization incorrectly to queries that actually involve
substantial work, you might see the queries being slower as a result of remote reads. In that case, recompute statistics
with the COMPUTE STATS or COMPUTE INCREMENTAL STATS statement. If there is a problem collecting accurate
statistics, you can turn this feature off by setting the value to -1.
Internal details:
This setting applies to query fragments where the amount of data to scan can be accurately determined, either through
table and column statistics, or by the presence of a LIMIT clause. If Impala cannot accurately estimate the size of the
input data, this setting does not apply.
In CDH 5.5 / Impala 2.3 and higher, where Impala supports the complex data types STRUCT, ARRAY, and MAP, if a query
refers to any column of those types, the small-query optimization is turned off for that query regardless of the
EXEC_SINGLE_NODE_ROWS_THRESHOLD setting.
For a query that is determined to be “small”, all work is performed on the coordinator node. This might result in some
I/O being performed by remote reads. The savings from not distributing the query work and not generating native
code are expected to outweigh any overhead from the remote reads.
Added in: CDH 5.3.0 (Impala 2.1.0)
Examples:
A common use case is to query just a few rows from a table to inspect typical data values. In this example, Impala does
not parallelize the query or perform native code generation because the result set is guaranteed to be smaller than
the threshold value from this query option:
SET EXEC_SINGLE_NODE_ROWS_THRESHOLD=500;
SELECT * FROM enormous_table LIMIT 300;
• 3 or VERBOSE: The maximum level of detail, showing how work is split up within each node into “query fragments”
that are connected in a pipeline. This extra detail is primarily useful for low-level performance testing and tuning
within Impala itself, rather than for rewriting the SQL code at the user level.
Note: Prior to Impala 1.3, the allowed argument range for EXPLAIN_LEVEL was 0 to 1: level 0 had
the mnemonic NORMAL, and level 1 was VERBOSE. In Impala 1.3 and higher, NORMAL is not a valid
mnemonic value, and VERBOSE still applies to the highest level of detail but now corresponds to level
3. You might need to adjust the values if you have any older impala-shell script files that set the
EXPLAIN_LEVEL query option.
Changing the value of this option controls the amount of detail in the output of the EXPLAIN statement. The extended
information from level 2 or 3 is especially useful during performance tuning, when you need to confirm whether the
work for the query is distributed the way you expect, particularly for the most resource-intensive operations such as
join queries against large tables, queries against tables with large numbers of partitions, and insert operations for
Parquet tables. The extended information also helps to check estimated resource usage when you use the admission
control or resource management features explained in Integrated Resource Management with YARN on page 83. See
EXPLAIN Statement on page 260 for the syntax of the EXPLAIN statement, and Using the EXPLAIN Plan for Performance
Tuning on page 507 for details about how to use the extended information.
Usage notes:
As always, read the EXPLAIN output from bottom to top. The lowest lines represent the initial work of the query
(scanning data files), the lines in the middle represent calculations done on each node and how intermediate results
are transmitted from one node to another, and the topmost lines represent the final results being sent back to the
coordinator node.
The numbers in the left column are generated internally during the initial planning phase and do not represent the
actual order of operations, so it is not significant if they appear out of order in the EXPLAIN output.
At all EXPLAIN levels, the plan contains a warning if any tables in the query are missing statistics. Use the COMPUTE
STATS statement to gather statistics for each table and suppress this warning. See Table and Column Statistics on page
489 for details about how the statistics help query performance.
The PROFILE command in impala-shell always starts with an explain plan showing full detail, the same as with
EXPLAIN_LEVEL=3. After the explain plan comes the executive summary, the same output as produced by the SUMMARY
command in impala-shell.
Examples:
These examples use a trivial, empty table to illustrate how the essential aspects of query planning are shown in EXPLAIN
output:
As the warning message demonstrates, most of the information needed for Impala to do efficient query planning, and
for you to understand the performance characteristics of the query, requires running the COMPUTE STATS statement
for the table:
| Explain String |
+------------------------------------------------------------------------+
| Estimated Per-Host Requirements: Memory=-9223372036854775808B VCores=0 |
| |
| F01:PLAN FRAGMENT [PARTITION=UNPARTITIONED] |
| 01:EXCHANGE [PARTITION=UNPARTITIONED] |
| hosts=0 per-host-mem=unavailable |
| tuple-ids=0 row-size=20B cardinality=0 |
| |
| F00:PLAN FRAGMENT [PARTITION=RANDOM] |
| DATASTREAM SINK [FRAGMENT=F01, EXCHANGE=01, PARTITION=UNPARTITIONED] |
| 00:SCAN HDFS [explain_plan.t1, PARTITION=RANDOM] |
| partitions=1/1 size=0B |
| table stats: 0 rows total |
| column stats: all |
| hosts=0 per-host-mem=0B |
| tuple-ids=0 row-size=20B cardinality=0 |
+------------------------------------------------------------------------+
Joins and other complicated, multi-part queries are the ones where you most commonly need to examine the EXPLAIN
output and customize the amount of detail in the output. This example shows the default EXPLAIN output for a
three-way join query, then the equivalent output with a [SHUFFLE] hint to change the join mechanism between the
first two tables from a broadcast join to a shuffle join.
For a join involving many different tables, the default EXPLAIN output might stretch over several pages, and the only
details you care about might be the join order and the mechanism (broadcast or shuffle) for joining each pair of tables.
In that case, you might set EXPLAIN_LEVEL to its lowest value of 0, to focus on just the join order and join mechanism
for each stage. The following example shows how the rows from the first and second joined tables are hashed and
divided among the nodes of the cluster for further filtering; then the entire contents of the third table are broadcast
to all nodes for the final stage of join processing.
Type: Boolean; recognized values are 1 and 0, or true and false; any other value interpreted as false
Default: false (shown as 0 in output of SET statement)
Command-line equivalent:
You can enable this query option within impala-shell by starting the shell with the --live_progress command-line
option. You can still turn this setting off and on again within the shell through the SET command.
Usage notes:
The output from this query option is printed to standard error. The output is only displayed in interactive mode, that
is, not when the -q or -f options are used.
For a more detailed way of tracking the progress of an interactive query through all phases of processing, see
LIVE_SUMMARY Query Option (CDH 5.5 or higher only) on page 315.
Restrictions:
Because the percentage complete figure is calculated using the number of issued and completed “scan ranges”, which
occur while reading the table data, the progress bar might reach 100% before the query is entirely finished. For example,
the query might do work to perform aggregations after all the table data has been read. If many of your queries fall
into this category, consider using the LIVE_SUMMARY option instead for more granular progress reporting.
The LIVE_PROGRESS and LIVE_SUMMARY query options currently do not produce any output during COMPUTE STATS
operations.
Because the LIVE_PROGRESS and LIVE_SUMMARY query options are available only within the impala-shell
interpreter:
• You cannot change these query options through the SQL SET statement using the JDBC or ODBC interfaces. The
SET command in impala-shell recognizes these names as shell-only options.
• Be careful when using impala-shell on a pre-CDH 5.5 system to connect to Impala running on a CDH 5.5 or
higher system. The older impala-shell does not recognize these query option names. Upgrade impala-shell
on the systems where you intend to use these query options.
• Likewise, the impala-shell command relies on some information only available in Impala 2.3 / CDH 5.5 and
higher to prepare live progress reports and query summaries. The LIVE_PROGRESS and LIVE_SUMMARY query
options have no effect when impala-shell connects to a cluster running an older version of Impala.
To see how the LIVE_PROGRESS and LIVE_SUMMARY query options work in real time, see this animated demo.
• Be careful when using impala-shell on a pre-CDH 5.5 system to connect to Impala running on a CDH 5.5 or
higher system. The older impala-shell does not recognize these query option names. Upgrade impala-shell
on the systems where you intend to use these query options.
• Likewise, the impala-shell command relies on some information only available in Impala 2.3 / CDH 5.5 and
higher to prepare live progress reports and query summaries. The LIVE_PROGRESS and LIVE_SUMMARY query
options have no effect when impala-shell connects to a cluster running an older version of Impala.
+---------------------+--------+----------+----------+---------+------------+----------+---------------+-----------------------+
| 06:AGGREGATE | 0 | 0ns | 0ns | 0 | 1 | 0 B | -1 B | FINALIZE |
| 05:EXCHANGE | 0 | 0ns | 0ns | 0 | 1 | 0 B | -1 B | UNPARTITIONED |
| 03:AGGREGATE | 0 | 0ns | 0ns | 0 | 1 | 0 B | 10.00 MB | |
| 02:NESTED LOOP JOIN | 0 | 0ns | 0ns | 0 | 22.50B | 0 B | 0 B | CROSS JOIN, BROADCAST |
| |--04:EXCHANGE | 0 | 0ns | 0ns | 0 | 150.00K | 0 B | 0 B | BROADCAST |
| | 01:SCAN HDFS | 1 | 503.57ms | 503.57ms | 150.00K | 150.00K | 24.09 MB | 64.00 MB | tpch.customer t2 |
| 00:SCAN HDFS | 0 | 0ns | 0ns | 0 | 150.00K | 0 B | 64.00 MB | tpch.customer t1 |
+---------------------+--------+----------+----------+---------+------------+----------+---------------+-----------------------+
+---------------------+--------+----------+----------+---------+------------+----------+---------------+-----------------------+
| Operator | #Hosts | Avg Time | Max Time | #Rows | Est. #Rows | Peak Mem | Est. Peak Mem | Detail |
+---------------------+--------+----------+----------+---------+------------+----------+---------------+-----------------------+
| 06:AGGREGATE | 0 | 0ns | 0ns | 0 | 1 | 0 B | -1 B | FINALIZE |
| 05:EXCHANGE | 0 | 0ns | 0ns | 0 | 1 | 0 B | -1 B | UNPARTITIONED |
| 03:AGGREGATE | 1 | 0ns | 0ns | 0 | 1 | 20.00 KB | 10.00 MB | |
| 02:NESTED LOOP JOIN | 1 | 17.62s | 17.62s | 81.14M | 22.50B | 3.23 MB | 0 B | CROSS JOIN, BROADCAST |
| |--04:EXCHANGE | 1 | 26.29ms | 26.29ms | 150.00K | 150.00K | 0 B | 0 B | BROADCAST |
| | 01:SCAN HDFS | 1 | 503.57ms | 503.57ms | 150.00K | 150.00K | 24.09 MB | 64.00 MB | tpch.customer t2 |
| 00:SCAN HDFS | 1 | 247.53ms | 247.53ms | 1.02K | 150.00K | 24.39 MB | 64.00 MB | tpch.customer t1 |
+---------------------+--------+----------+----------+---------+------------+----------+---------------+-----------------------+
+---------------------+--------+----------+----------+---------+------------+----------+---------------+-----------------------+
| Operator | #Hosts | Avg Time | Max Time | #Rows | Est. #Rows | Peak Mem | Est. Peak Mem | Detail |
+---------------------+--------+----------+----------+---------+------------+----------+---------------+-----------------------+
| 06:AGGREGATE | 0 | 0ns | 0ns | 0 | 1 | 0 B | -1 B | FINALIZE |
| 05:EXCHANGE | 0 | 0ns | 0ns | 0 | 1 | 0 B | -1 B | UNPARTITIONED |
| 03:AGGREGATE | 1 | 0ns | 0ns | 0 | 1 | 20.00 KB | 10.00 MB | |
| 02:NESTED LOOP JOIN | 1 | 61.85s | 61.85s | 283.43M | 22.50B | 3.23 MB | 0 B | CROSS JOIN, BROADCAST |
| |--04:EXCHANGE | 1 | 26.29ms | 26.29ms | 150.00K | 150.00K | 0 B | 0 B | BROADCAST |
| | 01:SCAN HDFS | 1 | 503.57ms | 503.57ms | 150.00K | 150.00K | 24.09 MB | 64.00 MB | tpch.customer t2 |
| 00:SCAN HDFS | 1 | 247.59ms | 247.59ms | 2.05K | 150.00K | 24.39 MB | 64.00 MB | tpch.customer t1 |
+---------------------+--------+----------+----------+---------+------------+----------+---------------+-----------------------+
To see how the LIVE_PROGRESS and LIVE_SUMMARY query options work in real time, see this animated demo.
MAX_ERRORS Query Option
Maximum number of non-fatal errors for any particular query that are recorded in the Impala log file. For example, if
a billion-row table had a non-fatal data error in every row, you could diagnose the problem without all billion errors
being logged. Unspecified or 0 indicates the built-in default value of 1000.
This option only controls how many errors are reported. To specify whether Impala continues or halts when it encounters
such errors, use the ABORT_ON_ERROR option.
Type: numeric
Default: 0 (meaning 1000 errors)
Related information:
ABORT_ON_ERROR Query Option on page 306, Using Impala Logging on page 574
MAX_IO_BUFFERS Query Option
Deprecated query option. Currently has no effect.
Type: numeric
Default: 0
MAX_SCAN_RANGE_LENGTH Query Option
Maximum length of the scan range. Interacts with the number of HDFS blocks in the table to determine how many
CPU cores across the cluster are involved with the processing for a query. (Each core processes one scan range.)
Lowering the value can sometimes increase parallelism if you have unused CPU capacity, but a too-small value can
limit query performance because each scan range involves extra overhead.
Only applicable to HDFS tables. Has no effect on Parquet tables. Unspecified or 0 indicates backend default, which is
the same as the HDFS block size for each table.
Although the scan range can be arbitrarily long, Impala internally uses an 8 MB read buffer so that it can query tables
with huge block sizes without allocating equivalent blocks of memory.
Type: numeric
Default: 0
The following examples show how unrecognized MEM_LIMIT values lead to errors for subsequent queries.
The following examples shows the automatic query cancellation when the MEM_LIMIT value is exceeded on any host
involved in the Impala query. First it runs a successful query and checks the largest amount of memory used on any
node for any stage of the query. Then it sets an artificially low MEM_LIMIT setting so that the same query cannot run.
If you are diagnosing a problem that you suspect is due to a timing issue due to distributed query processing, you can
set NUM_NODES=1 to verify if the problem still occurs when all the work is done on a single node.
You might set the NUM_NODES option to 1 briefly, during INSERT or CREATE TABLE AS SELECT statements. Normally,
those statements produce one or more data files per data node. If the write operation involves small amounts of data,
a Parquet table, and/or a partitioned table, the default behavior could produce many small files when intuitively you
might expect only a single output file. SET NUM_NODES=1 turns off the “distributed” aspect of the write operation,
making it more likely to produce only one or a few data files.
Warning:
Because this option results in increased resource utilization on a single host, it could cause problems
due to contention with other Impala statements or high resource usage. Symptoms could include
queries running slowly, exceeding the memory limit, or appearing to hang. Use it only in a single-user
development/test environment; do not use it in a production environment or in a cluster with a
high-concurrency or high-volume or performance-critical workload.
-- 128 megabytes.
set PARQUET_FILE_SIZE=134217728
INSERT OVERWRITE parquet_table SELECT * FROM text_table;
-- 512 megabytes.
set PARQUET_FILE_SIZE=512m;
INSERT OVERWRITE parquet_table SELECT * FROM text_table;
-- 1 gigabyte.
set PARQUET_FILE_SIZE=1g;
INSERT OVERWRITE parquet_table SELECT * FROM text_table;
Usage notes:
With tables that are small or finely partitioned, the default Parquet block size (formerly 1 GB, now 256 MB in Impala
2.0 and later) could be much larger than needed for each data file. For INSERT operations into such tables, you can
increase parallelism by specifying a smaller PARQUET_FILE_SIZE value, resulting in more HDFS blocks that can be
processed by different nodes.
Type: numeric, with optional unit specifier
Important:
Currently, the maximum value for this setting is 1 gigabyte (1g). Setting a value higher than 1 gigabyte
could result in errors during an INSERT operation.
Default: 0 (produces files with a target size of 256 MB; files might be larger for very wide tables)
Isilon considerations:
Because the EMC Isilon storage devices use a global value for the block size rather than a configurable value for each
file, the PARQUET_FILE_SIZE query option has no effect when Impala inserts data into a table or partition residing
on Isilon storage. Use the isi command to set the default block size globally on the Isilon device. For example, to set
the Isilon default block size to 256 MB, the recommended size for Parquet data files for Impala, issue the following
command:
Related information:
For information about the Parquet file format, and how the number and size of data files affects query performance,
see Using the Parquet File Format with Impala Tables on page 535.
QUERY_TIMEOUT_S Query Option (CDH 5.2 or higher only)
Sets the idle query timeout value for the session, in seconds. Queries that sit idle for longer than the timeout value
are automatically cancelled. If the system administrator specified the --idle_query_timeout startup option,
QUERY_TIMEOUT_S must be smaller than or equal to the --idle_query_timeout value.
Note:
The timeout clock for queries and sessions only starts ticking when the query or session is idle. For
queries, this means the query has results ready but is waiting for a client to fetch the data. A query
can run for an arbitrary time without triggering a timeout, because the query is computing results
rather than sitting idle waiting for the results to be fetched. The timeout period is intended to prevent
unclosed queries from consuming resources and taking up slots in the admission count of running
queries, potentially preventing other queries from starting.
For sessions, this means that no query has been submitted for some period of time.
Syntax:
SET QUERY_TIMEOUT_S=seconds;
Type: numeric
Default: 0 (no timeout if --idle_query_timeout not in effect; otherwise, use --idle_query_timeout value)
Added in: CDH 5.2.0 (Impala 2.0.0)
Related information:
Setting Timeout Periods for Daemons, Queries, and Sessions on page 84
REQUEST_POOL Query Option
The pool or queue name that queries should be submitted to. Only applies when you enable the Impala admission
control feature (CDH 4 or CDH 5; see Admission Control and Query Queuing on page 75), or the YARN resource
management feature (CDH 5 only; see Integrated Resource Management with YARN on page 83). Specifies the name
of the pool used by requests from Impala to the resource manager.
Formerly known as YARN_POOL during the CDH 5 beta period. Renamed to reflect that it can be used both with YARN
and with the lightweight admission control feature introduced in Impala 1.3.
Type: STRING
Default: empty (use the user-to-pool mapping defined by an impalad startup option in the Impala configuration file)
RESERVATION_REQUEST_TIMEOUT Query Option (CDH 5 only)
Note:
This query option is no longer supported, because it affects interaction between Impala and Llama.
The use of the Llama component for integrated resource management within YARN is no longer
supported with CDH 5.5 / Impala 2.3 and higher.
Maximum number of milliseconds Impala will wait for a reservation to be completely granted or denied. Used in
conjunction with the Impala resource management feature in Impala 1.2 and higher with CDH 5.
Type: numeric
Default: 300000 (5 minutes)
SUPPORT_START_OVER Query Option
Leave this setting at its default value. It is a read-only setting, tested by some client applications such as Hue.
If you accidentally change it through impala-shell, subsequent queries encounter errors until you undo the change
by issuing UNSET support_start_over.
Type: Boolean; recognized values are 1 and 0, or true and false; any other value interpreted as false
Default: false
SYNC_DDL Query Option
When enabled, causes any DDL operation such as CREATE TABLE or ALTER TABLE to return only when the changes
have been propagated to all other Impala nodes in the cluster by the Impala catalog service. That way, if you issue a
subsequent CONNECT statement in impala-shell to connect to a different node in the cluster, you can be sure that
other node will already recognize any added or changed tables. (The catalog service automatically broadcasts the DDL
changes to all nodes automatically, but without this option there could be a period of inconsistency if you quickly
switched to another node, such as by issuing a subsequent query through a load-balancing proxy.)
Although INSERT is classified as a DML statement, when the SYNC_DDL option is enabled, INSERT statements also
delay their completion until all the underlying data and metadata changes are propagated to all Impala nodes. Internally,
Impala inserts have similarities with DDL statements in traditional database systems, because they create metadata
needed to track HDFS block locations for new files and they potentially add new partitions to partitioned tables.
Note: Because this option can introduce a delay after each write operation, if you are running a
sequence of CREATE DATABASE, CREATE TABLE, ALTER TABLE, INSERT, and similar statements
within a setup script, to minimize the overall delay you can enable the SYNC_DDL query option only
near the end, before the final DDL statement.
Type: Boolean; recognized values are 1 and 0, or true and false; any other value interpreted as false
Default: false (shown as 0 in output of SET statement)
Related information:
DDL Statements on page 214
Note:
This query option is no longer supported, because it affects interaction between Impala and Llama.
The use of the Llama component for integrated resource management within YARN is no longer
supported with CDH 5.5 / Impala 2.3 and higher.
The number of per-host virtual CPU cores to request from YARN. If set, the query option overrides the automatic
estimate from Impala. Used in conjunction with the Impala resource management feature in Impala 1.2 and higher
and CDH 5.
Type: numeric
Default: 0 (use automatic estimates)
SHOW Statement
The SHOW statement is a flexible way to get information about different types of Impala objects.
Syntax:
SHOW ROLES
SHOW CURRENT ROLES
SHOW ROLE GRANT GROUP group_name
SHOW GRANT ROLE role_name
Issue a SHOW object_type statement to see the appropriate objects in the current database, or SHOW object_type
IN database_name to see objects in a specific database.
The optional pattern argument is a quoted string literal, using Unix-style * wildcards and allowing | for alternation.
The preceding LIKE keyword is also optional. All object names are stored in lowercase, so use all lowercase letters in
the pattern string. For example:
Note: This statement applies to tables and partitions stored on HDFS, or in the Amazon Simple Storage
System (S3). It does not apply to views. It does not apply to tables mapped onto HBase, because HBase
does not use the same file-based storage layout.
Usage notes:
You can use this statement to verify the results of your ETL process: that is, that the expected files are present, with
the expected sizes. You can examine the file information to detect conditions such as empty files, missing files, or
inefficient layouts due to a large number of small files. When you use INSERT statements to copy from one table to
another, you can see how the file layout changes due to file format conversions, compaction of small input files into
large data blocks, and multiple output files from parallel queries and partitioned inserts.
The output from this statement does not include files that Impala considers to be hidden or invisible, such as those
whose names start with a dot or an underscore, or that end with the suffixes .copying or .tmp.
The information for partitioned tables complements the output of the SHOW PARTITIONS statement, which summarizes
information about each partition. SHOW PARTITIONS produces some output for each partition, while SHOW FILES
does not produce any output for empty partitions because they do not include any data files.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read permission for all the
table files, read and execute permission for all the directories that make up the table, and execute permission for the
database directory and all its parent directories.
Examples:
The following example shows a SHOW FILES statement for an unpartitioned table using text format:
This example illustrates how, after issuing some INSERT ... VALUES statements, the table now contains some tiny
files of just a few bytes. Such small files could cause inefficient processing of parallel queries that are expecting
multi-megabyte input files. The example shows how you might compact the small files by doing an INSERT ...
SELECT into a different table, possibly converting the data to Parquet in the process:
The following example shows a SHOW FILES statement for a partitioned text table with data in two different partitions,
and two empty partitions. The partitions with no data are not represented in the SHOW FILES output.
The following example shows a SHOW FILES statement for a partitioned Parquet table. The number and sizes of files
are different from the equivalent partitioned text table used in the previous example, because INSERT operations for
Parquet tables are parallelized differently than for text tables. (Also, the amount of data is so small that it can be written
to Parquet without involving all the hosts in this 4-node cluster.)
The following example shows output from the SHOW FILES statement for a table where the data files are stored in
Amazon S3:
Security considerations:
When authorization is enabled, the output of the SHOW statement is limited to those objects for which you have some
privilege. There might be other database, tables, and so on, but their names are concealed. If you believe an object
exists but you cannot see it in the SHOW output, check with the system administrator if you need to be granted a new
privilege for that object. See Enabling Sentry Authorization for Impala on page 95 for how to set up authorization and
add privileges for specific kinds of objects.
Examples:
Depending on the roles set up within your organization by the CREATE ROLE statement, the output might look
something like this:
show roles;
+-----------+
| role_name |
+-----------+
| analyst |
| role1 |
| sales |
| superuser |
| test_role |
+-----------+
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Related information:
Enabling Sentry Authorization for Impala on page 95
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Related information:
Enabling Sentry Authorization for Impala on page 95
SHOW DATABASES
The SHOW DATABASES statement is often the first one you issue when connecting to an instance for the first time.
You typically issue SHOW DATABASES to see the names you can specify in a USE db_name statement, then after
switching to a database you issue SHOW TABLES to see the names you can specify in SELECT and INSERT statements.
The output of SHOW DATABASES includes the special _impala_builtins database, which lets you view definitions
of built-in functions, as described under SHOW FUNCTIONS.
Security considerations:
When authorization is enabled, the output of the SHOW statement is limited to those objects for which you have some
privilege. There might be other database, tables, and so on, but their names are concealed. If you believe an object
exists but you cannot see it in the SHOW output, check with the system administrator if you need to be granted a new
privilege for that object. See Enabling Sentry Authorization for Impala on page 95 for how to set up authorization and
add privileges for specific kinds of objects.
Examples:
This example shows how you might locate a particular table on an unfamiliar system. The DEFAULT database is the
one you initially connect to; a database with that name is present on every system. You can issue SHOW TABLES IN
db_name without going into a database, or SHOW TABLES once you are inside a particular database.
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Related information:
Overview of Impala Databases on page 205, CREATE DATABASE Statement on page 231, DROP DATABASE Statement on
page 252, USE Statement on page 337 SHOW TABLES Statement on page 327, SHOW FUNCTIONS Statement on page 335
Examples:
The following examples demonstrate the SHOW TABLES statement. If the database contains no tables, the result set
is empty. If the database does contain tables, SHOW TABLES IN db_name lists all the table names. SHOW TABLES
with no qualifiers lists all the table names in the current database.
use full_db;
show tables;
+------+
| name |
+------+
| t1 |
| t2 |
+------+
This example demonstrates how SHOW TABLES LIKE 'wildcard_pattern' lists table names that match a pattern,
or multiple alternative patterns. The ability to do wildcard matches for table names makes it helpful to establish naming
conventions for tables to conveniently locate a group of related tables.
/* Asterisk is the wildcard character. Only 2 out of the 3 just-created tables are
returned. */
show tables like 'dim*';
+-----------+
| name |
+-----------+
| dim_tbl_1 |
| dim_tbl_2 |
+-----------+
/* We are already in the FULL_DB database, but just to be sure we can specify the database
name also. */
show tables in full_db like 'dim*';
+-----------+
| name |
+-----------+
| dim_tbl_1 |
| dim_tbl_2 |
+-----------+
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Related information:
Overview of Impala Tables on page 208, CREATE TABLE Statement on page 235, ALTER TABLE Statement on page 215,
DROP TABLE Statement on page 258, DESCRIBE Statement on page 245, SHOW CREATE TABLE Statement on page 329,
SHOW TABLE STATS Statement on page 330, SHOW DATABASES on page 326, SHOW FUNCTIONS Statement on page
335
Security considerations:
When authorization is enabled, the output of the SHOW statement is limited to those objects for which you have some
privilege. There might be other database, tables, and so on, but their names are concealed. If you believe an object
exists but you cannot see it in the SHOW output, check with the system administrator if you need to be granted a new
privilege for that object. See Enabling Sentry Authorization for Impala on page 95 for how to set up authorization and
add privileges for specific kinds of objects.
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Examples:
The following example shows how various clauses from the CREATE TABLE statement are represented in the output
of SHOW CREATE TABLE.
create table show_create_table_demo (id int comment "Unique ID", y double, s string)
partitioned by (year smallint)
stored as parquet;
|
| LOCATION 'hdfs://127.0.0.1:8020/user/hive/warehouse/scratch.db/show_create_table_demo'
|
| TBLPROPERTIES ('transient_lastDdlTime'='1418152582')
|
+----------------------------------------------------------------------------------------+
The following example shows how, after a sequence of ALTER TABLE statements, the output from SHOW CREATE
TABLE represents the current state of the table. This output could be used to create a matching table rather than
executing the original CREATE TABLE and sequence of ALTER TABLE statements.
Related information:
CREATE TABLE Statement on page 235, DESCRIBE Statement on page 245, SHOW TABLES Statement on page 327
The output of SHOW COLUMN STATS is primarily only useful after the COMPUTE STATS statement has been run on
the table. A -1 in the #Distinct Values output column indicates that the COMPUTE STATS statement has never
been run for this table. Currently, Impala always leaves the #Nulls column as -1, even after COMPUTE STATS has
been run.
These SHOW statements work on actual tables only, not on views.
Security considerations:
When authorization is enabled, the output of the SHOW statement is limited to those objects for which you have some
privilege. There might be other database, tables, and so on, but their names are concealed. If you believe an object
exists but you cannot see it in the SHOW output, check with the system administrator if you need to be granted a new
privilege for that object. See Enabling Sentry Authorization for Impala on page 95 for how to set up authorization and
add privileges for specific kinds of objects.
Examples:
The following examples show how the SHOW TABLE STATS statement displays physical information about a table
and the associated data files:
The following example shows how, after a COMPUTE STATS or COMPUTE INCREMENTAL STATS statement, the #Rows
field is now filled in. Because the STORE_SALES table in this example is not partitioned, the COMPUTE INCREMENTAL
STATS statement produces regular stats rather than incremental stats, therefore the Incremental stats field
remains false.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read and execute permissions
for all directories that are part of the table. (A table could span multiple different HDFS directories if it is partitioned.
The directories could be widely scattered because a partition can reside in an arbitrary HDFS directory based on its
LOCATION attribute.) The Impala user must also have execute permission for the database directory, and any parent
directories of the database directory in HDFS.
Related information:
COMPUTE STATS Statement on page 226, SHOW COLUMN STATS Statement on page 332
See Table and Column Statistics on page 489 for usage information and examples.
| ss_quantity | INT | -1 | -1 | 4 | 4 |
| ss_wholesale_cost | FLOAT | -1 | -1 | 4 | 4 |
| ss_list_price | FLOAT | -1 | -1 | 4 | 4 |
| ss_sales_price | FLOAT | -1 | -1 | 4 | 4 |
| ss_ext_discount_amt | FLOAT | -1 | -1 | 4 | 4 |
| ss_ext_sales_price | FLOAT | -1 | -1 | 4 | 4 |
| ss_ext_wholesale_cost | FLOAT | -1 | -1 | 4 | 4 |
| ss_ext_list_price | FLOAT | -1 | -1 | 4 | 4 |
| ss_ext_tax | FLOAT | -1 | -1 | 4 | 4 |
| ss_coupon_amt | FLOAT | -1 | -1 | 4 | 4 |
| ss_net_paid | FLOAT | -1 | -1 | 4 | 4 |
| ss_net_paid_inc_tax | FLOAT | -1 | -1 | 4 | 4 |
| ss_net_profit | FLOAT | -1 | -1 | 4 | 4 |
+-----------------------+-------+------------------+--------+----------+----------+
The following examples show the output of the SHOW COLUMN STATS statement for some tables, after the COMPUTE
STATS statement is run. Now most of the -1 values are changed to reflect the actual table data. The #Nulls column
remains -1 because Impala does not use the number of NULL values to influence query planning.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read and execute permissions
for all directories that are part of the table. (A table could span multiple different HDFS directories if it is partitioned.
The directories could be widely scattered because a partition can reside in an arbitrary HDFS directory based on its
LOCATION attribute.) The Impala user must also have execute permission for the database directory, and any parent
directories of the database directory in HDFS.
Related information:
COMPUTE STATS Statement on page 226, SHOW TABLE STATS Statement on page 330
See Table and Column Statistics on page 489 for usage information and examples.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have read and execute permissions
for all directories that are part of the table. (A table could span multiple different HDFS directories if it is partitioned.
The directories could be widely scattered because a partition can reside in an arbitrary HDFS directory based on its
LOCATION attribute.) The Impala user must also have execute permission for the database directory, and any parent
directories of the database directory in HDFS.
Related information:
See Table and Column Statistics on page 489 for usage information and examples.
SHOW TABLE STATS Statement on page 330, Partitioning for Impala Tables on page 522
Related information:
Overview of Impala Functions on page 206, Impala Built-In Functions on page 338, Impala User-Defined Functions (UDFs)
on page 447, SHOW DATABASES on page 326, SHOW TABLES Statement on page 327
Often used to empty tables that are used during ETL cycles, after the data has been copied to another table for the
next stage of processing. This statement is a low-overhead alternative to dropping and recreating the table, or using
INSERT OVERWRITE to replace the data during the next ETL cycle.
This statement removes all the data and associated data files in the table. It can remove data files from internal tables,
external tables, partitioned tables, and tables mapped to HBase or the Amazon Simple Storage Service (S3). The data
removal applies to the entire table, including all partitions of a partitioned table.
Any statistics produced by the COMPUTE STATS statement are reset when the data is removed.
Make sure that you are in the correct database before truncating a table, either by issuing a USE statement first or by
using a fully qualified name db_name.table_name.
Any HDFS data files removed by this statement go into the HDFS trashcan, from which you can recover them within a
defined time interval if this operation turns out to be a mistake.
For other tips about managing and reclaiming Impala disk space, see Managing Disk Space for Impala Data on page
89.
Amazon S3 considerations:
Although Impala cannot write new data to a table stored in the Amazon S3 filesystem, the TRUNCATE TABLE statement
can remove data files from S3. See Using Impala to Query the Amazon S3 Filesystem on page 566 for details about
working with S3 tables.
Cancellation: Cannot be cancelled.
HDFS permissions:
The user ID that the impalad daemon runs under, typically the impala user, must have write permission for all the
files and directories that make up the table.
Examples:
The following example shows a table containing some data and with table and column statistics. After the TRUNCATE
TABLE statement, the data is removed and the statistics are reset.
-- After this statement, the data and the table/column stats will be gone.
TRUNCATE TABLE truncate_demo;
+----------+
| count(*) |
+----------+
| 0 |
+----------+
SHOW TABLE STATS truncate_demo;
+-------+--------+------+--------------+-------------------+--------+-------------------+
| #Rows | #Files | Size | Bytes Cached | Cache Replication | Format | Incremental stats
|
+-------+--------+------+--------------+-------------------+--------+-------------------+
| -1 | 0 | 0B | NOT CACHED | NOT CACHED | TEXT | false
|
+-------+--------+------+--------------+-------------------+--------+-------------------+
SHOW COLUMN STATS truncate_demo;
+--------+------+------------------+--------+----------+----------+
| Column | Type | #Distinct Values | #Nulls | Max Size | Avg Size |
+--------+------+------------------+--------+----------+----------+
| x | INT | -1 | -1 | 4 | 4 |
+--------+------+------------------+--------+----------+----------+
Related information:
Overview of Impala Tables on page 208, ALTER TABLE Statement on page 215, CREATE TABLE Statement on page 235,
Partitioning for Impala Tables on page 522, Internal Tables on page 208, External Tables on page 209
USE Statement
Switches the current session to a specified database. The current database is where any CREATE TABLE, INSERT,
SELECT, or other statements act when you specify a table or other object name, without prefixing it with a database
name. The new current database applies for the duration of the session or unti another USE statement is executed.
Syntax:
USE db_name
By default, when you connect to an Impala instance, you begin in a database named default.
Usage notes:
Switching the default database is convenient in the following situations:
• To avoid qualifying each reference to a table with the database name. For example, SELECT * FROM t1 JOIN
t2 rather than SELECT * FROM db.t1 JOIN db.t2.
• To do a sequence of operations all within the same database, such as creating a table, inserting data, and querying
the table.
To start the impala-shell interpreter and automatically issue a USE statement for a particular database, specify the
option -d db_name for the impala-shell command. The -d option is useful to run SQL scripts, such as setup or
test scripts, against multiple databases without hardcoding a USE statement into the SQL source.
Examples:
See CREATE DATABASE Statement on page 231 for examples covering CREATE DATABASE, USE, and DROP DATABASE.
Cancellation: Cannot be cancelled.
HDFS permissions: This statement does not touch any HDFS files or directories, therefore no HDFS permissions are
required.
Related information:
CREATE DATABASE Statement on page 231, DROP DATABASE Statement on page 252, SHOW DATABASES on page 326
select abs(-1);
+---------+
| abs(-1) |
+---------+
| 1 |
+---------+
select power(2,5);
+-------------+
| power(2, 5) |
+-------------+
| 32 |
+-------------+
When you use a FROM clause and specify a column name as a function argument, the function is applied for each item
in the result set:
Typically, if any argument to a built-in function is NULL, the result value is also NULL:
select cos(null);
+-----------+
| cos(null) |
+-----------+
| NULL |
+-----------+
select power(2,null);
+----------------+
| power(2, null) |
+----------------+
| NULL |
+----------------+
select concat('a',null,'b');
+------------------------+
| concat('a', null, 'b') |
+------------------------+
| NULL |
+------------------------+
Aggregate functions are a special category with different rules. These functions calculate a return value across all the
items in a result set, so they require a FROM clause in the query:
Aggregate functions also ignore NULL values rather than returning a NULL result. For example, if some rows have NULL
for a particular column, those rows are ignored when computing the AVG() for that column. Likewise, specifying
COUNT(col_name) in a query counts only those rows where col_name contains a non-NULL value.
Aggregate functions are a special category with different rules. These functions calculate a return value across all the
items in a result set, so they do require a FROM clause in the query:
Aggregate functions also ignore NULL values rather than returning a NULL result. For example, if some rows have NULL
for a particular column, those rows are ignored when computing the AVG() for that column. Likewise, specifying
COUNT(col_name) in a query counts only those rows where col_name contains a non-NULL value.
Analytic functions are a variation on aggregate functions. Instead of returning a single value, or an identical value for
each group of rows, they can compute values that vary based on a “window” consisting of other rows around them in
the result set.
Note: In Impala, exponentiation uses the pow() function rather than an exponentiation operator
such as **.
Related information:
The mathematical functions operate mainly on these data types: INT Data Type on page 135, BIGINT Data Type on page
120, SMALLINT Data Type on page 140, TINYINT Data Type on page 153, DOUBLE Data Type on page 133, FLOAT Data
Type on page 134, and DECIMAL Data Type (CDH 5.1 or higher only) on page 125. For the operators that perform the
standard operations such as addition, subtraction, multiplication, and division, see Arithmetic Operators on page 188.
Functions that perform bitwise operations are explained in Impala Bit Functions on page 351.
Function reference:
Impala supports the following mathematical functions:
abs(numeric_type a)
Purpose: Returns the absolute value of the argument.
Return type: Same as the input value
Usage notes: Use this function to ensure all return values are positive. This is different than the positive()
function, which returns its argument unchanged (even if the argument was negative).
acos(double a)
Purpose: Returns the arccosine of the argument.
Return type: double
asin(double a)
Purpose: Returns the arcsine of the argument.
Return type: double
atan(double a)
Purpose: Returns the arctangent of the argument.
Return type: double
atan2(double a, double b)
Purpose: Returns the arctangent of the two arguments, with the signs of the arguments used to determine the
quadrant of the result.
Return type: double
bin(bigint a)
Purpose: Returns the binary representation of an integer value, that is, a string of 0 and 1 digits.
Return type: string
ceil(double a), ceil(decimal(p,s) a), ceiling(double a), ceiling(decimal(p,s) a), dceil(double
a), dceil(decimal(p,s) a)
Purpose: Returns the smallest integer that is greater than or equal to the argument.
Return type: bigint or decimal(p,s) depending on the type of the input argument
conv(bigint num, int from_base, int to_base), conv(string num, int from_base, int to_base)
Purpose: Returns a string representation of an integer value in a particular base. The input value can be a string,
for example to convert a hexadecimal number such as fce2 to decimal. To use the return value as a number (for
example, when converting to base 10), use CAST() to convert to the appropriate type.
Return type: string
cos(double a)
Purpose: Returns the cosine of the argument.
Return type: double
cosh(double a)
Purpose: Returns the hyperbolic cosine of the argument.
Return type: double
cot(double a)
Purpose: Returns the cotangent of the argument.
Return type: double
Added in: CDH 5.5.0 (Impala 2.3.0)
degrees(double a)
Purpose: Converts argument value from radians to degrees.
Return type: double
e()
Purpose: Returns the mathematical constant e.
Return type: double
select factorial(5);
+--------------+
| factorial(5) |
+--------------+
| 120 |
+--------------+
select 5!;
+-----+
| 5! |
+-----+
| 120 |
+-----+
select factorial(0);
+--------------+
| factorial(0) |
+--------------+
| 1 |
+--------------+
select factorial(-100);
+-----------------+
| factorial(-100) |
+-----------------+
| 1 |
+-----------------+
The following examples show equivalent operations with the fmod() function and the % arithmetic operator, for
values not subject to any rounding error.
select fmod(10,3);
+-------------+
| fmod(10, 3) |
+-------------+
| 1 |
+-------------+
select fmod(5.5,2);
+--------------+
| fmod(5.5, 2) |
+--------------+
| 1.5 |
+--------------+
select 10 % 3;
+--------+
| 10 % 3 |
+--------+
| 1 |
+--------+
select 5.5 % 2;
+---------+
| 5.5 % 2 |
+---------+
| 1.5 |
+---------+
The following examples show operations with the fmod() function for values that cannot be represented precisely
by the DOUBLE or FLOAT types, and thus are subject to rounding error. fmod(9.9,3.0) returns a value slightly
different than the expected 0.9 because of rounding. fmod(9.9,3.3) returns a value quite different from the
expected value of 0 because of rounding error during intermediate calculations.
select fmod(9.9,3.0);
+--------------------+
| fmod(9.9, 3.0) |
+--------------------+
| 0.8999996185302734 |
+--------------------+
select fmod(9.9,3.3);
+-------------------+
| fmod(9.9, 3.3) |
+-------------------+
| 3.299999713897705 |
+-------------------+
fnv_hash(type v),
Purpose: Returns a consistent 64-bit value derived from the input argument, for convenience of implementing
hashing logic in an application.
Return type: BIGINT
Usage notes:
You might use the return value in an application where you perform load balancing, bucketing, or some other
technique to divide processing or storage.
Because the result can be any 64-bit value, to restrict the value to a particular range, you can use an expression
that includes the ABS() function and the % (modulo) operator. For example, to produce a hash value in the range
0-9, you could use the expression ABS(FNV_HASH(x)) % 10.
This function implements the same algorithm that Impala uses internally for hashing, on systems where the CRC32
instructions are not available.
This function implements the Fowler–Noll–Vo hash function, in particular the FNV-1a variation. This is not a perfect
hash function: some combinations of values could produce the same result value. It is not suitable for cryptographic
use.
Similar input values of different types could produce different hash values, for example the same numeric value
represented as SMALLINT or BIGINT, FLOAT or DOUBLE, or DECIMAL(5,2) or DECIMAL(20,5).
Examples:
For short argument values, the high-order bits of the result have relatively low entropy:
is_nan(double a),
Purpose: Tests whether a value is equal to the special value “NaN”, signifying “not a number”.
Return type: boolean
Usage notes:
Infinity and NaN can be specified in text data files as inf and nan respectively, and Impala interprets them as these
special values. They can also be produced by certain arithmetic expressions; for example, 1/0 returns Infinity
and pow(-1, 0.5) returns NaN. Or you can cast the literal values, such as CAST('nan' AS DOUBLE) or
CAST('inf' AS DOUBLE).
least(bigint a[, bigint b ...]), least(double a[, double b ...]), least(decimal(p,s) a[,
decimal(p,s) b ...]), least(string a[, string b ...]), least(timestamp a[, timestamp b
...])
Purpose: Returns the smallest value from a list of expressions.
Return type: same as the initial argument value, except that integer values are promoted to BIGINT and floating-point
values are promoted to DOUBLE; use CAST() when inserting into a smaller numeric column
ln(double a), dlog1(double a)
Purpose: Returns the natural logarithm of the argument.
Return type: double
log(double base, double a)
Purpose: Returns the logarithm of the second argument to the specified base.
Return type: double
log10(double a), dlog10(double a)
Purpose: Returns the logarithm of the argument to the base 10.
Return type: double
log2(double a)
Purpose: Returns the logarithm of the argument to the base 2.
Return type: double
max_int(), max_tinyint(), max_smallint(), max_bigint()
Purpose: Returns the largest value of the associated integral type.
Return type: The same as the integral type being checked.
Usage notes: Use the corresponding min_ and max_ functions to check if all values in a column are within the
allowed range, before copying data or altering column definitions. If not, switch to the next higher integral type or
to a DECIMAL with sufficient precision.
min_int(), min_tinyint(), min_smallint(), min_bigint()
Purpose: Returns the smallest value of the associated integral type (a negative number).
Return type: The same as the integral type being checked.
Usage notes: Use the corresponding min_ and max_ functions to check if all values in a column are within the
allowed range, before copying data or altering column definitions. If not, switch to the next higher integral type or
to a DECIMAL with sufficient precision.
mod(numeric_type a, same_type b)
Purpose: Returns the modulus of a number. Equivalent to the % arithmetic operator. Works with any size integer
type, any size floating-point type, and DECIMAL with any precision and scale.
Return type: Same as the input value
Added in: CDH 5.4.0 (Impala 2.2.0)
Usage notes:
Because this function works with DECIMAL values, prefer it over fmod() when working with fractional values. It is
not subject to the rounding errors that make fmod() problematic with floating-point numbers. The % arithmetic
operator now uses the mod() function in cases where its arguments can be interpreted as DECIMAL values, increasing
the accuracy of that operator.
Examples:
The following examples show how the mod() function works for whole numbers and fractional values, and how
the % operator works the same way. In the case of mod(9.9,3), the type conversion for the second argument
results in the first argument being interpreted as DOUBLE, so to produce an accurate DECIMAL result requires casting
the second argument or writing it as a DECIMAL literal, 3.0.
select mod(10,3);
+-------------+
| fmod(10, 3) |
+-------------+
| 1 |
+-------------+
select mod(5.5,2);
+--------------+
| fmod(5.5, 2) |
+--------------+
| 1.5 |
+--------------+
select 10 % 3;
+--------+
| 10 % 3 |
+--------+
| 1 |
+--------+
select 5.5 % 2;
+---------+
| 5.5 % 2 |
+---------+
| 1.5 |
+---------+
select mod(9.9,3.3);
+---------------+
| mod(9.9, 3.3) |
+---------------+
| 0.0 |
+---------------+
select mod(9.9,3);
+--------------------+
| mod(9.9, 3) |
+--------------------+
| 0.8999996185302734 |
+--------------------+
select mod(9.9,3.0);
+---------------+
| mod(9.9, 3.0) |
+---------------+
| 0.9 |
+---------------+
negative(numeric_type a)
Purpose: Returns the argument with the sign reversed; returns a positive value if the argument was already negative.
Return type: Same as the input value
Usage notes: Use -abs(a) instead if you need to ensure all return values are negative.
pi()
Purpose: Returns the constant pi.
Return type: double
pmod(bigint a, bigint b), pmod(double a, double b)
Purpose: Returns the positive modulus of a number. Primarily for HiveQL compatibility.
Return type: int or double, depending on type of arguments
Examples:
The following examples show how the fmod() function sometimes returns a negative value depending on the sign
of its arguments, and the pmod() function returns the same value as fmod(), but sometimes with the sign flipped.
select fmod(-5,2);
+-------------+
| fmod(-5, 2) |
+-------------+
| -1 |
+-------------+
select pmod(-5,2);
+-------------+
| pmod(-5, 2) |
+-------------+
| 1 |
+-------------+
select fmod(-5,-2);
+--------------+
| fmod(-5, -2) |
+--------------+
| -1 |
+--------------+
select pmod(-5,-2);
+--------------+
| pmod(-5, -2) |
+--------------+
| -1 |
+--------------+
select fmod(5,-2);
+-------------+
| fmod(5, -2) |
+-------------+
| 1 |
+-------------+
select pmod(5,-2);
+-------------+
| pmod(5, -2) |
+-------------+
| -1 |
+-------------+
positive(numeric_type a)
Purpose: Returns the original argument unchanged (even if the argument is negative).
Return type: Same as the input value
Usage notes: Use abs() instead if you need to ensure all return values are positive.
pow(double a, double p), power(double a, double p), dpow(double a, double p), fpow(double
a, double p)
Purpose: Returns the first argument raised to the power of the second argument.
Return type: double
precision(numeric_expression)
Purpose: Computes the precision (number of decimal digits) needed to represent the type of the argument expression
as a DECIMAL value.
Usage notes:
Typically used in combination with the scale() function, to determine the appropriate
DECIMAL(precision,scale) type to declare in a CREATE TABLE statement or CAST() function.
| x | unix_timestamp(now()) | rand(unix_timestamp(now())) |
+---+-----------------------+-----------------------------+
| 1 | 1440777752 | 0.002051228658320023 |
| 2 | 1440777752 | 0.5098743483004506 |
| 3 | 1440777752 | 0.9517714925817081 |
+---+-----------------------+-----------------------------+
round(double a), round(double a, int d), round(decimal a, int_type d), dround(double a),
dround(double a, int d), dround(decimal(p,s) a, int_type d)
Purpose: Rounds a floating-point value. By default (with a single argument), rounds to the nearest integer. Values
ending in .5 are rounded up for positive numbers, down for negative numbers (that is, away from zero). The optional
second argument specifies how many digits to leave after the decimal point; values greater than zero produce a
floating-point return value rounded to the requested number of digits to the right of the decimal point.
Return type: bigint for single double argument. double for two-argument signature when second argument
greater than zero. For DECIMAL values, the smallest DECIMAL(p,s) type with appropriate precision and scale.
scale(numeric_expression)
Purpose: Computes the scale (number of decimal digits to the right of the decimal point) needed to represent the
type of the argument expression as a DECIMAL value.
Usage notes:
Typically used in combination with the precision() function, to determine the appropriate
DECIMAL(precision,scale) type to declare in a CREATE TABLE statement or CAST() function.
sign(double a)
Purpose: Returns -1, 0, or 1 to indicate the signedness of the argument value.
Return type: int
sin(double a)
Purpose: Returns the sine of the argument.
Return type: double
sinh(double a)
Purpose: Returns the hyperbolic sine of the argument.
Return type: double
sqrt(double a), dsqrt(double a)
Purpose: Returns the square root of the argument.
Return type: double
tan(double a)
Purpose: Returns the tangent of the argument.
Return type: double
tanh(double a)
Purpose: Returns the hyperbolic tangent of the argument.
Return type: double
truncate(double_or_decimal a[, digits_to_leave]), dtrunc(double_or_decimal a[,
digits_to_leave])
Purpose: Removes some or all fractional digits from a numeric value. With no argument, removes all fractional
digits, leaving an integer value. The optional argument specifies the number of fractional digits to include in the
return value, and only applies with the argument type is DECIMAL. truncate() and dtrunc() are aliases for the
same function.
Return type: decimal for DECIMAL arguments; bigint for DOUBLE arguments
Examples:
select truncate(3.45)
+----------------+
| truncate(3.45) |
+----------------+
| 3 |
+----------------+
select truncate(-3.45)
+-----------------+
| truncate(-3.45) |
+-----------------+
| -3 |
+-----------------+
select truncate(3.456,1)
+--------------------+
| truncate(3.456, 1) |
+--------------------+
| 3.4 |
+--------------------+
select dtrunc(3.456,1)
+------------------+
| dtrunc(3.456, 1) |
+------------------+
| 3.4 |
+------------------+
select truncate(3.456,2)
+--------------------+
| truncate(3.456, 2) |
+--------------------+
| 3.45 |
+--------------------+
select truncate(3.456,7)
+--------------------+
| truncate(3.456, 7) |
+--------------------+
| 3.4560000 |
+--------------------+
unhex(string a)
Purpose: Returns a string of characters with ASCII values corresponding to pairs of hexadecimal digits in the argument.
Return type: string
| 255 |
+--------------------+
bitnot(integer_type a)
Purpose: Inverts all the bits of the input argument.
Usage notes: The bitnot() function is equivalent to the ~ unary operator.
Return type: Same as the input value
Added in: CDH 5.5.0 (Impala 2.3.0)
Examples:
These examples illustrate what happens when you flip all the bits of an integer value. The sign always changes. The
decimal representation is one different between the positive and negative values.
| 127 |
+--------------+
bitor(integer_type a, same_type b)
Purpose: Returns an integer value representing the bits that are set to 1 in either of the arguments. If the arguments
are of different sizes, the smaller is promoted to the type of the larger.
Usage notes: The bitor() function is equivalent to the | binary operator.
Return type: Same as the input value
Added in: CDH 5.5.0 (Impala 2.3.0)
Examples:
The following examples show the results of ORing integer values.
bitxor(integer_type a, same_type b)
Purpose: Returns an integer value representing the bits that are set to 1 in one but not both of the arguments. If
the arguments are of different sizes, the smaller is promoted to the type of the larger.
Usage notes: The bitxor() function is equivalent to the ^ binary operator.
Return type: Same as the input value
Added in: CDH 5.5.0 (Impala 2.3.0)
Examples:
The following examples show the results of XORing integer values. XORing a non-zero value with zero returns the
non-zero value. XORing two identical values returns zero, because all the 1 bits from the first argument are also 1
bits in the second argument. XORing different non-zero values turns off some bits and leaves others turned on,
based on whether the same bit is set in both arguments.
| 5 |
+----------------+
number of 1 bits as the original value, just in different positions. In computer science terms, this operation is a
“circular shift”.
Usage notes:
Specifying a second argument of zero leaves the original value unchanged. Rotating a -1 value by any number of
positions still returns -1, because the original value has all 1 bits and all the 1 bits are preserved during rotation.
Similarly, rotating a 0 value by any number of positions still returns 0. Rotating a value by the same number of bits
as in the value returns the same value. Because this is a circular operation, the number of positions is not limited
to the number of bits in the input value. For example, rotating an 8-bit value by 1, 9, 17, and so on positions returns
an identical result in each case.
Return type: Same as the input value
Added in: CDH 5.5.0 (Impala 2.3.0)
Examples:
Examples:
+--------------+
+-----------------+
| shiftleft(8, 2) |
+-----------------+
| 32 |
+-----------------+
| shiftright(-1, 1) |
+-------------------+
| 127 |
+-------------------+
Note: Although in CDH 5.5.0, the SHOW FUNCTIONS output for database _IMPALA_BUILTINS contains
some function signatures matching the pattern castto*, these functions are not intended for public
use and are expected to be hidden in future.
Function reference:
Impala supports the following type conversion functions:
cast(expr AS type)
Purpose: Converts the value of an expression to any other type. If the expression value is of a type that cannot be
converted to the target type, the result is NULL.
Usage notes: Use CAST when passing a column value or literal to a function that expects a parameter with a different
type. Frequently used in SQL operations such as CREATE TABLE AS SELECT and INSERT ... VALUES to ensure
that values from various sources are of the appropriate type for the destination columns. Where practical, do a
one-time CAST() operation during the ingestion process to make each column into the appropriate type, rather
than using many CAST() operations in each query; doing type conversions for each row during each query can be
expensive for tables with millions or billions of rows.
The way this function deals with time zones when converting to or from TIMESTAMP values is affected by the
--use_local_tz_for_unix_timestamp_conversions startup flag for the impalad daemon. See TIMESTAMP
Data Type on page 148 for details about how Impala handles time zone considerations for the TIMESTAMP data
type.
Examples:
The following example starts with a text table where every column has a type of STRING, which might be how you
ingest data of unknown schema until you can verify the cleanliness of the underly values. Then it uses CAST() to
create a new Parquet table with the same data, but using specific numeric data types for the columns with numeric
data. Using numeric types of appropriate sizes can result in substantial space savings on disk and in memory, and
performance improvements in queries, over using strings or larger-than-necessary numeric types.
cast(z as smallint) z
from t1;
describe t2;
+------+----------+---------+
| name | type | comment |
+------+----------+---------+
| name | string | |
| x | bigint | |
| y | smallint | |
| z | tinyint | |
+------+----------+---------+
Related information:
For details of casts from each kind of data type, see the description of the appropriate type: TINYINT Data Type on
page 153, SMALLINT Data Type on page 140, INT Data Type on page 135, BIGINT Data Type on page 120, FLOAT Data
Type on page 134, DOUBLE Data Type on page 133, DECIMAL Data Type (CDH 5.1 or higher only) on page 125, STRING
Data Type on page 141, CHAR Data Type (CDH 5.2 or higher only) on page 122, VARCHAR Data Type (CDH 5.2 or
higher only) on page 154, TIMESTAMP Data Type on page 148, BOOLEAN Data Type on page 121
typeof(type value)
Purpose: Returns the name of the data type corresponding to an expression. For types with extra attributes, such
as length for CHAR and VARCHAR, or precision and scale for DECIMAL, includes the full specification of the type.
Return type: string
Usage notes: Typically used in interactive exploration of a schema, or in application code that programmatically
generates schema definitions such as CREATE TABLE statements. For example, previously, to understand the type
of an expression such as col1 / col2 or concat(col1, col2, col3), you might have created a dummy table
with a single row, using syntax such as CREATE TABLE foo AS SELECT 5 / 3.0, and then doing a DESCRIBE
to see the type of the row. Or you might have done a CREATE TABLE AS SELECT operation to create a table and
copy data into it, only learning the types of the columns by doing a DESCRIBE afterward. This technique is especially
useful for arithmetic expressions involving DECIMAL types, because the precision and scale of the result is typically
different than that of the operands.
Added in: CDH 5.5.0 (Impala 2.3.0)
Examples:
These examples show how to check the type of a simple literal or function value. Notice how adding even tiny
integers together changes the data type of the result to avoid overflow, and how the results of arithmetic operations
on DECIMAL values have specific precision and scale attributes.
select typeof(2)
+-----------+
| typeof(2) |
+-----------+
| TINYINT |
+-----------+
select typeof(2+2)
+---------------+
| typeof(2 + 2) |
+---------------+
| SMALLINT |
+---------------+
select typeof('xyz')
+---------------+
| typeof('xyz') |
+---------------+
| STRING |
+---------------+
select typeof(now())
+---------------+
| typeof(now()) |
+---------------+
| TIMESTAMP |
+---------------+
select typeof(typeof(2+2))
+-----------------------+
| typeof(typeof(2 + 2)) |
+-----------------------+
| STRING |
+-----------------------+
This example shows how even if you do not have a record of the type of a column, for example because the type
was changed by ALTER TABLE after the original CREATE TABLE, you can still find out the type in a more compact
form than examining the full DESCRIBE output. Remember to use LIMIT 1 in such cases, to avoid an identical
result value for every row in the table.
/* OK, now we have some data but the type of column A is being changed. */
insert into typeof_example values (1, 2, 3, 4);
alter table typeof_example change a a bigint;
/* We can always find out the current type of that column without doing a full DESCRIBE.
*/
select typeof(a) from typeof_example limit 1;
+-----------+
| typeof(a) |
+-----------+
| BIGINT |
+-----------+
This example shows how you might programmatically generate a CREATE TABLE statement with the appropriate
column definitions to hold the result values of arbitrary expressions. The typeof() function lets you construct a
detailed CREATE TABLE statement without actually creating the table, as opposed to CREATE TABLE AS SELECT
operations where you create the destination table but only learn the column data types afterward through DESCRIBE.
describe typeof_example;
+------+----------+---------+
| name | type | comment |
+------+----------+---------+
| a | bigint | |
| b | tinyint | |
| c | smallint | |
| d | bigint | |
+------+----------+---------+
/* An ETL or business intelligence tool might create variations on a table with different
file formats,
different sets of columns, and so on. TYPEOF() lets an application introspect the
types of the original columns. */
select concat('create table derived_table (a ', typeof(a), ', b ', typeof(b), ', c ',
+-------------------------------+-------------------------------+
| 2016-05-31 10:47:31.732298000 | 2016-04-30 10:47:31.732298000 |
+-------------------------------+-------------------------------+
current_timestamp()
Purpose: Alias for the now() function.
Return type: timestamp
Examples:
| right_now | next_week |
+-------------------------------+-------------------------------+
| 2016-05-20 11:03:48.687055000 | 2016-05-27 11:03:48.687055000 |
+-------------------------------+-------------------------------+
The following examples show the shorthand notation of an INTERVAL expression, instead of specifying the precise
number of days. The INTERVAL notation also lets you work with units smaller than a single day.
Like all date/time functions that deal with months, date_add() handles nonexistent dates past the end of a month
by setting the date to the last day of the month. The following example shows how the nonexistent date April 31st
is normalized to April 30th:
date_part(string, timestamp)
Purpose: Similar to EXTRACT(), with the argument order reversed. Supports the same date and time units as
EXTRACT(). For compatibility with SQL code containing vendor extensions.
The following example shows the simplest usage, of subtracting a specified number of days from a TIMESTAMP
value:
The following examples show the shorthand notation of an INTERVAL expression, instead of specifying the precise
number of days. The INTERVAL notation also lets you work with units smaller than a single day.
Like all date/time functions that deal with months, date_add() handles nonexistent dates past the end of a month
by setting the date to the last day of the month. The following example shows how the nonexistent date April 31st
is normalized to April 30th:
The following examples show how the return value represent the number of days between the associated dates,
regardless of the time portion of each TIMESTAMP. For example, different times on the same day produce a
date_diff() of 0, regardless of which one is earlier or later. But if the arguments represent different dates,
date_diff() returns a non-zero integer value, regardless of the time portions of the dates.
The following examples show how the day value is NULL for nonexistent dates or misformatted date strings.
dayname(timestamp date)
Purpose: Returns the day field from a TIMESTAMP value, converted to the string corresponding to that day name.
The range of return values is 'Sunday' to 'Saturday'. Used in report-generating queries, as an alternative to
calling dayofweek() and turning that numeric return value into a string using a CASE expression.
Return type: string
Examples:
The following examples show the day name associated with TIMESTAMP values representing different days.
dayofweek(timestamp date)
Purpose: Returns the day field from the date portion of a TIMESTAMP, corresponding to the day of the week. The
range of return values is 1 (Sunday) to 7 (Saturday).
Return type: int
Examples:
+-------------------------------+--------------------+-----------------+
| right_now | todays_day_of_week | todays_day_name |
+-------------------------------+--------------------+-----------------+
| 2016-05-31 10:57:03.953670000 | 3 | Tuesday |
+-------------------------------+--------------------+-----------------+
dayofyear(timestamp date)
Purpose: Returns the day field from a TIMESTAMP value, corresponding to the day of the year. The range of return
values is 1 (January 1) to 366 (December 31 of a leap year).
Return type: int
Examples:
The following examples show return values from the dayofyear() function. The same date in different years
returns a different day number for all dates after February 28, because 2016 is a leap year while 2015 is not a leap
year.
Usage notes:
Typically used in GROUP BY queries to arrange results by hour, day, month, and so on. You can also use this function
in an INSERT ... SELECT into a partitioned table to split up TIMESTAMP values into individual parts, if the
partitioned table has separate partition key columns representing year, month, day, and so on. If you need to divide
by more complex units of time, such as by week or by quarter, use the TRUNC() function instead.
Return type: int
Examples:
The following examples show different ways to format a TIMESTAMP value as a string:
Note:
The more flexible format strings allowed with the built-in functions do not change the rules about
using CAST() to convert from a string to a TIMESTAMP value. Strings being converted through
CAST() must still have the elements in the specified order and use the specified delimiter characters,
as described in TIMESTAMP Data Type on page 148.
Examples:
select from_unixtime(1392394861,"yyyy-MM-dd");
+-----------------------------------------+
| from_unixtime(1392394861, 'yyyy-mm-dd') |
+-----------------------------------------+
| 2014-02-14 |
+-----------------------------------------+
select from_unixtime(1392394861,"HH:mm:ss.SSSS");
+--------------------------------------------+
| from_unixtime(1392394861, 'hh:mm:ss.ssss') |
+--------------------------------------------+
| 16:21:01.0000 |
+--------------------------------------------+
select from_unixtime(1392394861,"HH:mm:ss");
+---------------------------------------+
| from_unixtime(1392394861, 'hh:mm:ss') |
+---------------------------------------+
| 16:21:01 |
+---------------------------------------+
unix_timestamp() and from_unixtime() are often used in combination to convert a TIMESTAMP value into
a particular string format. For example:
The following example shows how when TIMESTAMP values representing the UTC time zone are stored in a table,
a query can display the equivalent local date and time for a different time zone.
The following example shows that for a date and time when daylight savings is in effect (PDT), the UTC time is 7
hours ahead of the local California time; while when daylight savings is not in effect (PST), the UTC time is 8 hours
ahead of the local California time.
hour(timestamp date)
Purpose: Returns the hour field from a TIMESTAMP field.
Return type: int
Examples:
If either value is NULL, which could happen for example when converting a nonexistent date string such as
'2015-02-29' to a TIMESTAMP, the result is also NULL.
If the first argument represents an earlier time than the second argument, the result is negative.
Examples:
+------------------------------------------------+
| 1 |
+------------------------------------------------+
+-------------------------------+-------------------------------+
| 2016-06-01 14:30:30.067366000 | 2016-06-01 14:30:31.567366000 |
+-------------------------------+-------------------------------+
minute(timestamp date)
Purpose: Returns the minute field from a TIMESTAMP value.
Return type: int
Examples:
month(timestamp date)
Purpose: Returns the month field, represented as an integer, from the date portion of a TIMESTAMP.
Return type: int
Examples:
The following examples show how if months_add() would return a nonexistent date, due to different months
having different numbers of days, the function returns a TIMESTAMP from the last day of the relevant month. For
example, adding one month to January 31 produces a date of February 29th in the year 2016 (a leap year), and
February 28th in the year 2015 (a non-leap year).
The method of determining the number of elapsed months includes some special handling of months with different
numbers of days that creates edge cases for dates between the 28th and 31st days of certain months.
If either value is NULL, which could happen for example when converting a nonexistent date string such as
'2015-02-29' to a TIMESTAMP, the result is also NULL.
If the first argument represents an earlier time than the second argument, the result is negative.
Examples:
The following examples show how dates that are on the same day of the month are considered to be exactly N
months apart, even if the months have different numbers of days.
The following examples show how dates that are on the last day of the month are considered to be exactly N months
apart, even if the months have different numbers of days. For example, from January 28th to February 28th is
exactly one month because the day of the month is identical; January 31st to February 28th is exactly one month
because in both cases it is the last day of the month; but January 29th or 30th to February 28th is considered a
fractional month.
| 0.967741935483871 |
+--------------------------------------------+
The following examples show how dates that are not a precise number of months apart result in a fractional return
value.
The following examples show how the time portion of the TIMESTAMP values are irrelevant for calculating the
month interval. Even the fractional part of the result only depends on the number of full days between the argument
values, regardless of the time portion.
| 1 |
+--------------------------------------------------------------+
now()
Purpose: Returns the current date and time (in the local time zone) as a TIMESTAMP value.
Return type: timestamp
Usage notes:
To find a date/time value in the future or the past relative to the current date and time, add or subtract an INTERVAL
expression to the return value of now(). See TIMESTAMP Data Type on page 148 for examples.
To produce a TIMESTAMP representing the current date and time that can be shared or stored without interoperability
problems due to time zone differences, use the to_utc_timestamp() function and specify the time zone of the
server. When TIMESTAMP data is stored in UTC form, any application that queries those values can convert them
to the appropriate local time zone by calling the inverse function, from_utc_timestamp().
To determine the time zone of the server you are connected to, in CDH 5.5 / Impala 2.3 and higher you can call the
timeofday() function, which includes the time zone specifier in its return value. Remember that with cloud
computing, the server you interact with might be in a different time zone than you are, or different sessions might
connect to servers in different time zones, or a cluster might include servers in more than one time zone.
Any references to the now() function are evaluated at the start of a query. All calls to now() within the same query
return the same value, and the value does not depend on how long the query takes.
Examples:
second(timestamp date)
Purpose: Returns the second field from a TIMESTAMP value.
Return type: int
Examples:
+-------------------------------+---------------------------+
| right_now | seconds_in_current_minute |
+-------------------------------+---------------------------+
| 2016-06-01 16:03:57.006603000 | 57 |
+-------------------------------+---------------------------+
timeofday()
Purpose: Returns a string representation of the current date and time, according to the time of the local system,
including any time zone designation.
/* The TIMEOFDAY() result is formatted differently than NOW(). NOW() returns a TIMESTAMP.
*/
select now(), timeofday();
+-------------------------------+------------------------------+
| now() | timeofday() |
+-------------------------------+------------------------------+
| 2015-09-01 15:15:25.930021000 | Tue Sep 01 15:15:25 2015 PDT |
+-------------------------------+------------------------------+
/* You can strip out the time zone field to use in calls to from_utc_timestamp(). */
select regexp_replace(timeofday(), '.* ([A-Z]+)$', '\\1') as current_timezone;
+------------------+
| current_timezone |
+------------------+
| PDT |
+------------------+
time than the second argument, the result is -1. If the first and second arguments represent identical points in time,
the result is 0. If either argument is NULL, the result is NULL.
select timestamp_cmp(now() +
interval 3 days +
interval 5 hours, now())
as now_vs_days_from_now;
+----------------------+
| now_vs_days_from_now |
+----------------------+
| 1 |
+----------------------+
select timestamp_cmp
(
now() + interval 1 hour,
now() + interval 60 minutes
) as equivalent_date_times;
+-----------------------+
| equivalent_date_times |
+-----------------------+
| 0 |
+-----------------------+
to_date(timestamp)
Purpose: Returns a string representation of the date field from a timestamp value.
Return type: string
Examples:
The following examples show how to convert a BIGINT representing seconds past epoch into a TIMESTAMP value:
Once a value is converted to the UTC time zone by to_utc_timestamp(), it can be converted back to the local
time zone with from_utc_timestamp(). You can combine these functions using different time zone identifiers
to convert a TIMESTAMP between any two time zones. This example starts with a TIMESTAMP value representing
Pacific Daylight Time, converts it to UTC, and converts it to the equivalent value in Eastern Daylight Time.
• HH, HH12, HH24: Hour. A TIMESTAMP value truncated to the hour is always represented in 24-hour notation,
even for the HH12 argument string.
• MI: Minute.
Usage notes:
Typically used in GROUP BY queries to aggregate results from the same hour, day, week, month, quarter, and so
on. You can also use this function in an INSERT ... SELECT into a partitioned table to divide TIMESTAMP values
into the correct partition.
Because the return value is a TIMESTAMP, if you cast the result of TRUNC() to STRING, you will often see zeroed-out
portions such as 00:00:00 in the time field. If you only need the individual units such as hour, day, month, or year,
use the EXTRACT() function instead. If you need the individual units from a truncated TIMESTAMP value, run the
TRUNCATE() function on the original value, then run EXTRACT() on the result.
change application code that interacts with these functions, change the types of columns that store the return
values, or add CAST() calls to SQL statements that call these functions.
unix_timestamp() and from_unixtime() are often used in combination to convert a TIMESTAMP value into
a particular string format. For example:
The way this function deals with time zones when converting to or from TIMESTAMP values is affected by the
--use_local_tz_for_unix_timestamp_conversions startup flag for the impalad daemon. See TIMESTAMP
Data Type on page 148 for details about how Impala handles time zone considerations for the TIMESTAMP data
type.
Examples:
The following examples show different ways of turning the same date and time into an integer value. A format
string that Impala recognizes by default is interpreted as a UTC date and time. The trailing Z is a confirmation that
the timezone is UTC. If the date and time string is formatted differently, a second argument specifies the position
and units for each of the date and time values.
The final two examples show how to specify a timezone offset of Pacific Daylight Saving Time, which is 7 hours
earlier than UTC. You can use the numeric offset -07:00 and the equivalent suffix of -hh:mm in the format string,
or specify the mnemonic name for the time zone in a call to to_utc_timestamp(). This particular date and time
expressed in PDT translates to a different number than the same date and time expressed in UTC.
-- 2 ways of expressing the same date and time but in a different timezone.
-- The resulting integer is different from the previous examples.
select unix_timestamp
(
'2015-05-15 12:00:00-07:00',
'yyyy-MM-dd HH:mm:ss-hh:mm'
) as may_15_year_month_day;
+-----------------------+
| may_15_year_month_day |
+-----------------------+
| 1431716400 |
+-----------------------+
select unix_timestamp
(to_utc_timestamp(
'2015-05-15 12:00:00',
'PDT')
) as may_15_pdt;
+------------+
| may_15_pdt |
+------------+
| 1431716400 |
+------------+
weekofyear(timestamp date)
Purpose: Returns the corresponding week (1-53) from the date portion of a TIMESTAMP.
Return type: int
Examples:
year(timestamp date)
Purpose: Returns the year field from the date portion of a TIMESTAMP.
Return type: int
Examples:
The following example shows how if the equivalent date does not exist in the year of the result due to a leap year,
the date is changed to the last day of the appropriate month.
The following example shows how if the equivalent date does not exist in the year of the result due to a leap year,
the date is changed to the last day of the appropriate month.
| feb_29_2016 | feb_29_2015 |
+---------------------+---------------------+
| 2016-02-29 00:00:00 | 2015-02-28 00:00:00 |
+---------------------+---------------------+
select case x
when 1 then 'one'
when 2 then 'two'
when 0 then 'zero'
else 'out of range'
end
from t1;
Because this flexible form of the CASE expressions allows you to perform many comparisons and call multiple
functions when evaluating each row, be careful applying elaborate CASE expressions to queries that process large
amounts of data. For example, when practical, evaluate and transform values through CASE after applying operations
such as aggregations that reduce the size of the result set; transform numbers to strings after performing joins with
the original numeric values.
Examples:
Although this example is split across multiple lines, you can put any or all parts of a CASE expression on a single
line, with no punctuation or other separators between the WHEN, ELSE, and END clauses.
select case
when dayname(now()) in ('Saturday','Sunday') then 'result undefined on weekends'
when x > y then 'x greater than y'
when x = y then 'x and y are equal'
when x is null or y is null then 'one of the columns is null'
else null
end
from t1;
isnotfalse(boolean)
Purpose: Tests if a Boolean expression is not false (that is, either true or NULL). Returns true if so. If the argument
is NULL, returns true. Identical to istrue(), except it returns the opposite value for a NULL argument.
Return type: BOOLEAN
Usage notes: Primarily for compatibility with code containing industry extensions to SQL.
Added in: CDH 5.4.0 (Impala 2.2.0)
isnottrue(boolean)
Purpose: Tests if a Boolean expression is not true (that is, either false or NULL). Returns true if so. If the argument
is NULL, returns true. Identical to isfalse(), except it returns the opposite value for a NULL argument.
Return type: BOOLEAN
Added in: CDH 5.4.0 (Impala 2.2.0)
isnull(type a, type ifNull)
Purpose: Tests if an expression is NULL, and returns the expression result value if not. If the first argument is NULL,
returns the second argument.
Compatibility notes: Equivalent to the nvl() function from Oracle Database or ifnull() from MySQL. The nvl()
and ifnull() functions are also available in Impala.
Return type: Same as the first argument value
istrue(boolean)
Purpose: Tests if a Boolean expression is true or not. Returns true if so. If the argument is NULL, returns false.
Identical to isnotfalse(), except it returns the opposite value for a NULL argument.
Return type: BOOLEAN
Usage notes: Primarily for compatibility with code containing industry extensions to SQL.
Added in: CDH 5.4.0 (Impala 2.2.0)
nonnullvalue(expression)
Purpose: Tests if an expression (of any type) is NULL or not. Returns false if so. The converse of nullvalue().
Return type: BOOLEAN
Usage notes: Primarily for compatibility with code containing industry extensions to SQL.
Added in: CDH 5.4.0 (Impala 2.2.0)
nullif(expr1,expr2)
Purpose: Returns NULL if the two specified arguments are equal. If the specified arguments are not equal, returns
the value of expr1. The data types of the expressions must be compatible, according to the conversion rules from
Data Types on page 116. You cannot use an expression that evaluates to NULL for expr1; that way, you can distinguish
a return value of NULL from an argument value of NULL, which would never match expr2.
Usage notes: This function is effectively shorthand for a CASE expression of the form:
CASE
WHEN expr1 = expr2 THEN NULL
ELSE expr1
END
It is commonly used in division expressions, to produce a NULL result instead of a divide-by-zero error when the
divisor is equal to zero:
You might also use it for compatibility with other database systems that support the same NULLIF() function.
Return type: same as the initial argument value, except that integer values are promoted to BIGINT and floating-point
values are promoted to DOUBLE; use CAST() when inserting into a smaller numeric column
Added in: Impala 1.3.0
nullifzero(numeric_expr)
Purpose: Returns NULL if the numeric expression evaluates to 0, otherwise returns the result of the expression.
Usage notes: Used to avoid error conditions such as divide-by-zero in numeric calculations. Serves as shorthand
for a more elaborate CASE expression, to simplify porting SQL with vendor extensions to Impala.
Return type: same as the initial argument value, except that integer values are promoted to BIGINT and floating-point
values are promoted to DOUBLE; use CAST() when inserting into a smaller numeric column
Added in: Impala 1.3.0
nullvalue(expression)
Purpose: Tests if an expression (of any type) is NULL or not. Returns true if so. The converse of nonnullvalue().
Return type: BOOLEAN
Usage notes: Primarily for compatibility with code containing industry extensions to SQL.
Added in: CDH 5.4.0 (Impala 2.2.0)
nvl(type a, type ifNull)
Purpose: Alias for the isnull() function. Tests if an expression is NULL, and returns the expression result value
if not. If the first argument is NULL, returns the second argument. Equivalent to the nvl() function from Oracle
Database or ifnull() from MySQL.
Return type: Same as the first argument value
Added in: Impala 1.1
zeroifnull(numeric_expr)
Purpose: Returns 0 if the numeric expression evaluates to NULL, otherwise returns the result of the expression.
Usage notes: Used to avoid unexpected results due to unexpected propagation of NULL values in numeric calculations.
Serves as shorthand for a more elaborate CASE expression, to simplify porting SQL with vendor extensions to Impala.
Return type: same as the initial argument value, except that integer values are promoted to BIGINT and floating-point
values are promoted to DOUBLE; use CAST() when inserting into a smaller numeric column
Added in: Impala 1.3.0
Related information:
The string functions operate mainly on these data types: STRING Data Type on page 141, VARCHAR Data Type (CDH 5.2
or higher only) on page 154, and CHAR Data Type (CDH 5.2 or higher only) on page 122.
Function reference:
Impala supports the following string functions:
ascii(string str)
Purpose: Returns the numeric ASCII code of the first character of the argument.
Return type: int
btrim(string a), btrim(string a, string chars_to_trim)
Purpose: Removes all instances of one or more characters from the start and end of a STRING value. By default,
removes only spaces. If a non-NULL optional second argument is specified, the function removes all occurrences
of characters in that second argument from the beginning and end of the string.
Return type: string
Added in: CDH 5.5.0 (Impala 2.3.0)
Examples:
The following examples show the default btrim() behavior, and what changes when you specify the optional
second argument. All the examples bracket the output value with [ ] so that you can see any leading or trailing
spaces in the btrim() result. By default, the function removes and number of both leading and trailing spaces.
When the second argument is specified, any number of occurrences of any character in the second argument are
removed from the start and end of the input string; in this case, spaces are not removed (unless they are part of
the second argument) and any instances of the characters are not removed if they do not come right at the beginning
or end of the string.
Examples:
SELECT chr(65);
+---------+
| chr(65) |
+---------+
| A |
+---------+
SELECT chr(97);
+---------+
| chr(97) |
+---------+
| a |
+---------+
length(string a)
Purpose: Returns the length in characters of the argument string.
Return type: int
locate(string substr, string str[, int pos])
Purpose: Returns the position (starting from 1) of the first occurrence of a substring within a longer string, optionally
after a particular position.
Return type: int
lower(string a), lcase(string a)
Purpose: Returns the argument string converted to all-lowercase.
Return type: string
lpad(string str, int len, string pad)
Purpose: Returns a string of a specified length, based on the first argument string. If the specified string is too short,
it is padded on the left with a repeating sequence of the characters from the pad string. If the specified string is too
long, it is truncated on the right.
Return type: string
ltrim(string a)
Purpose: Returns the argument string with any leading spaces removed from the left side.
Return type: string
parse_url(string urlString, string partToExtract [, string keyToExtract])
Purpose: Returns the portion of a URL corresponding to a specified part. The part argument can be 'PROTOCOL',
'HOST', 'PATH', 'REF', 'AUTHORITY', 'FILE', 'USERINFO', or 'QUERY'. Uppercase is required for these
literal values. When requesting the QUERY portion of the URL, you can optionally specify a key to retrieve just the
associated value from the key-value pairs in the query string.
Return type: string
Usage notes: This function is important for the traditional Hadoop use case of interpreting web logs. For example,
if the web traffic data features raw URLs not divided into separate table columns, you can count visitors to a particular
page by extracting the 'PATH' or 'FILE' field, or analyze search terms by extracting the corresponding key from
the 'QUERY' field.
regexp_extract(string subject, string pattern, int index)
Purpose: Returns the specified () group from a string based on a regular expression pattern. Group 0 refers to the
entire extracted string, while group 1, 2, and so on refers to the first, second, and so on (...) portion.
Return type: string
In Impala 2.0 and later, the Impala regular expression syntax conforms to the POSIX Extended Regular Expression
syntax used by the Google RE2 library. For details, see the RE2 documentation. It has most idioms familiar from
regular expressions in Perl, Python, and so on, including .*? for non-greedy matches.
In Impala 2.0 and later, a change in the underlying regular expression library could cause changes in the way regular
expressions are interpreted by this function. Test any queries that use regular expressions and adjust the expression
patterns if necessary. See Incompatible Changes Introduced in Impala 2.0.0 / CDH 5.2.0 on page 628 for details.
Because the impala-shell interpreter uses the \ character for escaping, use \\ to represent the regular expression
escape character in any regular expressions that you submit through impala-shell . You might prefer to use the
equivalent character class names, such as [[:digit:]] instead of \d which you would have to escape as \\d.
Examples:
This example shows how group 0 matches the full pattern string, including the portion outside any () group:
This example shows how group 1 matches just the contents inside the first () group in the pattern string:
Unlike in earlier Impala releases, the regular expression library used in Impala 2.0 and later supports the .*? idiom
for non-greedy matches. This example shows how a pattern string starting with .*? matches the shortest possible
portion of the source string, returning the rightmost set of lowercase letters. A pattern string both starting and
ending with .*? finds two potential matches of equal length, and returns the first one found (the leftmost set of
lowercase letters).
Because the impala-shell interpreter uses the \ character for escaping, use \\ to represent the regular expression
escape character in any regular expressions that you submit through impala-shell . You might prefer to use the
equivalent character class names, such as [[:digit:]] instead of \d which you would have to escape as \\d.
Examples:
This example shows how regexp_like() can test for the existence of various kinds of regular expression patterns
within a source string:
-- The 3rd argument can change the matching logic, such as 'i' meaning case-insensitive.
select regexp_like('foo','F','i');
+------------------------------+
| regexp_like('foo', 'f', 'i') |
+------------------------------+
| true |
+------------------------------+
select regexp_like('foo','o$');
+--------------------------+
| regexp_like('foo', 'o$') |
+--------------------------+
| true |
+--------------------------+
select regexp_like('foooooobar','fx*y*o*b');
+---------------------------------------+
| regexp_like('foooooobar', 'fx*y*o*b') |
+---------------------------------------+
| true |
+---------------------------------------+
In Impala 2.0 and later, the Impala regular expression syntax conforms to the POSIX Extended Regular Expression
syntax used by the Google RE2 library. For details, see the RE2 documentation. It has most idioms familiar from
regular expressions in Perl, Python, and so on, including .*? for non-greedy matches.
In Impala 2.0 and later, a change in the underlying regular expression library could cause changes in the way regular
expressions are interpreted by this function. Test any queries that use regular expressions and adjust the expression
patterns if necessary. See Incompatible Changes Introduced in Impala 2.0.0 / CDH 5.2.0 on page 628 for details.
Because the impala-shell interpreter uses the \ character for escaping, use \\ to represent the regular expression
escape character in any regular expressions that you submit through impala-shell . You might prefer to use the
equivalent character class names, such as [[:digit:]] instead of \d which you would have to escape as \\d.
Examples:
These examples show how you can replace parts of a string matching a pattern with replacement text, which can
include backreferences to any () groups in the pattern string. The backreference numbers start at 1, and any \
characters must be escaped as \\.
Replace a character pattern with new text:
Replace a character pattern with substitution text that includes the original matching text:
rtrim(string a)
Purpose: Returns the argument string with any trailing spaces removed from the right side.
Return type: string
space(int n)
Purpose: Returns a concatenated string of the specified number of spaces. Shorthand for repeat(' ',n).
Return type: string
split_part(string source, string delimiter, bigint n)
Purpose: Returns the nth field within a delimited string. The fields are numbered starting from 1. The delimiter can
consist of multiple characters, not just a single character. All matching of the delimiter is done exactly, not using
any regular expression patterns.
Return type: string
In Impala 2.0 and later, the Impala regular expression syntax conforms to the POSIX Extended Regular Expression
syntax used by the Google RE2 library. For details, see the RE2 documentation. It has most idioms familiar from
regular expressions in Perl, Python, and so on, including .*? for non-greedy matches.
In Impala 2.0 and later, a change in the underlying regular expression library could cause changes in the way regular
expressions are interpreted by this function. Test any queries that use regular expressions and adjust the expression
patterns if necessary. See Incompatible Changes Introduced in Impala 2.0.0 / CDH 5.2.0 on page 628 for details.
Because the impala-shell interpreter uses the \ character for escaping, use \\ to represent the regular expression
escape character in any regular expressions that you submit through impala-shell . You might prefer to use the
equivalent character class names, such as [[:digit:]] instead of \d which you would have to escape as \\d.
Examples:
These examples show how to retrieve the nth field from a delimited string:
select split_part('x,y,z',',',1);
+-----------------------------+
| split_part('x,y,z', ',', 1) |
+-----------------------------+
| x |
+-----------------------------+
select split_part('x,y,z',',',2);
+-----------------------------+
| split_part('x,y,z', ',', 2) |
+-----------------------------+
| y |
+-----------------------------+
select split_part('x,y,z',',',3);
+-----------------------------+
| split_part('x,y,z', ',', 3) |
+-----------------------------+
| z |
+-----------------------------+
These examples show what happens for out-of-range field positions. Specifying a value less than 1 produces an
error. Specifying a value greater than the number of fields returns a zero-length string (which is not the same as
NULL).
select split_part('x,y,z',',',0);
ERROR: Invalid field position: 0
, concat('[',nonexistent_field,']')
, length(nonexistent_field);
from t1
+-------------------+-------------------------------------+---------------------------+
| nonexistent_field | concat('[', nonexistent_field, ']') | length(nonexistent_field)
|
+-------------------+-------------------------------------+---------------------------+
| | [] | 0
|
+-------------------+-------------------------------------+---------------------------+
select split_part('one***two***three','***',2);
+-------------------------------------------+
| split_part('one***two***three', '***', 2) |
+-------------------------------------------+
| two |
+-------------------------------------------+
select split_part('one\|/two\|/three','\|/',3);
+-------------------------------------------+
| split_part('one\|/two\|/three', '\|/', 3) |
+-------------------------------------------+
| three |
+-------------------------------------------+
Aggregate functions also ignore NULL values rather than returning a NULL result. For example, if some rows have NULL
for a particular column, those rows are ignored when computing the AVG() for that column. Likewise, specifying
COUNT(col_name) in a query counts only those rows where col_name contains a non-NULL value.
APPX_MEDIAN Function
An aggregate function that returns a value that is approximately the median (midpoint) of values in the set of input
values.
Syntax:
This function works with any input type, because the only requirement is that the type supports less-than and
greater-than comparison operators.
Usage notes:
Because the return value represents the estimated midpoint, it might not reflect the precise midpoint value, especially
if the cardinality of the input values is very high. If the cardinality is low (up to approximately 20,000), the result is
more accurate because the sampling considers all or almost all of the different values.
Return type: Same as the input value, except for CHAR and VARCHAR arguments which produce a STRING result
The return value is always the same as one of the input values, not an “in-between” value produced by averaging.
Restrictions:
This function cannot be used in an analytic context. That is, the OVER() clause is not allowed at all with this function.
Examples:
The following example uses a table of a million random floating-point numbers ranging up to approximately 50,000.
The average is approximately 25,000. Because of the random distribution, we would expect the median to be close to
this same number. Computing the precise median is a more intensive operation than computing the average, because
it requires keeping track of every distinct value and how many times each occurs. The APPX_MEDIAN() function uses
a sampling algorithm to return an approximate result, which in this case is close to the expected value. To make sure
that the value is not substantially out of range due to a skewed distribution, subsequent queries confirm that there
are approximately 500,000 values higher than the APPX_MEDIAN() value, and approximately 500,000 values lower
than the APPX_MEDIAN() value.
The following example computes the approximate median using a subset of the values from the table, and then confirms
that the result is a reasonable estimate for the midpoint.
[localhost:21000] > select appx_median(x) from million_numbers where x between 1000 and
5000;
+-------------------+
| appx_median(x) |
+-------------------+
| 3013.107787358159 |
+-------------------+
[localhost:21000] > select count(x) as higher from million_numbers where x between 1000
and 5000 and x > 3013.107787358159;
+--------+
| higher |
+--------+
| 37692 |
+--------+
[localhost:21000] > select count(x) as lower from million_numbers where x between 1000
and 5000 and x < 3013.107787358159;
+-------+
| lower |
+-------+
| 37089 |
+-------+
AVG Function
An aggregate function that returns the average value from a set of numbers or TIMESTAMP values. Its single argument
can be numeric column, or the numeric result of a function or expression applied to the column value. Rows with a
NULL value for the specified column are ignored. If the table is empty, or all the values supplied to AVG are NULL, AVG
returns NULL.
Syntax:
When the query contains a GROUP BY clause, returns one value for each combination of grouping values.
Return type: DOUBLE for numeric values; TIMESTAMP for TIMESTAMP values
Complex type considerations:
To access a column with a complex type (ARRAY, STRUCT, or MAP) in an aggregation function, you unpack the individual
elements using join notation in the query, and then apply the function to the final scalar item, field, key, or value at
the bottom of any nested type hierarchy in the column. See Complex Types (CDH 5.5 or higher only) on page 156 for
details about using complex types in Impala.
The following example demonstrates calls to several aggregation functions using values from a column containing
nested complex types (an ARRAY of STRUCT items). The array is unpacked inside the query using join notation. The
array elements are referenced using the ITEM pseudocolumn, and the structure fields inside the array elements are
referenced using dot notation. Numeric values such as SUM() and AVG() are computed using the numeric R_NATIONKEY
field, and the general-purpose MAX() and MIN() values are computed from the string N_NAME field.
describe region;
+-------------+-------------------------+---------+
| name | type | comment |
+-------------+-------------------------+---------+
| r_regionkey | smallint | |
| r_name | string | |
| r_comment | string | |
| r_nations | array<struct< | |
| | n_nationkey:smallint, | |
| | n_name:string, | |
| | n_comment:string | |
| | >> | |
+-------------+-------------------------+---------+
| AFRICA | 16 |
| AMERICA | 1 |
| AMERICA | 2 |
| AMERICA | 3 |
| AMERICA | 17 |
| AMERICA | 24 |
| ASIA | 8 |
| ASIA | 9 |
| ASIA | 12 |
| ASIA | 18 |
| ASIA | 21 |
| EUROPE | 6 |
| EUROPE | 7 |
| EUROPE | 19 |
| EUROPE | 22 |
| EUROPE | 23 |
| MIDDLE EAST | 4 |
| MIDDLE EAST | 10 |
| MIDDLE EAST | 11 |
| MIDDLE EAST | 13 |
| MIDDLE EAST | 20 |
+-------------+------------------+
select
r_name,
count(r_nations.item.n_nationkey) as count,
sum(r_nations.item.n_nationkey) as sum,
avg(r_nations.item.n_nationkey) as avg,
min(r_nations.item.n_name) as minimum,
max(r_nations.item.n_name) as maximum,
ndv(r_nations.item.n_nationkey) as distinct_vals
from
region, region.r_nations as r_nations
group by r_name
order by r_name;
+-------------+-------+-----+------+-----------+----------------+---------------+
| r_name | count | sum | avg | minimum | maximum | distinct_vals |
+-------------+-------+-----+------+-----------+----------------+---------------+
| AFRICA | 5 | 50 | 10 | ALGERIA | MOZAMBIQUE | 5 |
| AMERICA | 5 | 47 | 9.4 | ARGENTINA | UNITED STATES | 5 |
| ASIA | 5 | 68 | 13.6 | CHINA | VIETNAM | 5 |
| EUROPE | 5 | 77 | 15.4 | FRANCE | UNITED KINGDOM | 5 |
| MIDDLE EAST | 5 | 58 | 11.6 | EGYPT | SAUDI ARABIA | 5 |
+-------------+-------+-----+------+-----------+----------------+---------------+
Examples:
The following examples show how to use AVG() in an analytic context. They use a table containing integers from 1 to
10. Notice how the AVG() is reported for each input value, as opposed to the GROUP BY clause which condenses the
result set.
select x, property, avg(x) over (partition by property) as avg from int_t where property
in ('odd','even');
+----+----------+-----+
| x | property | avg |
+----+----------+-----+
| 2 | even | 6 |
| 4 | even | 6 |
| 6 | even | 6 |
| 8 | even | 6 |
| 10 | even | 6 |
| 1 | odd | 5 |
| 3 | odd | 5 |
| 5 | odd | 5 |
| 7 | odd | 5 |
| 9 | odd | 5 |
+----+----------+-----+
Adding an ORDER BY clause lets you experiment with results that are cumulative or apply to a moving set of rows (the
“window”). The following examples use AVG() in an analytic context (that is, with an OVER() clause) to produce a
running average of all the even values, then a running average of all the odd values. The basic ORDER BY x clause
implicitly activates a window clause of RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, which is
effectively the same as ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, therefore all of these examples
produce the same results:
select x, property,
avg(x) over (partition by property order by x) as 'cumulative average'
from int_t where property in ('odd','even');
+----+----------+--------------------+
| x | property | cumulative average |
+----+----------+--------------------+
| 2 | even | 2 |
| 4 | even | 3 |
| 6 | even | 4 |
| 8 | even | 5 |
| 10 | even | 6 |
| 1 | odd | 1 |
| 3 | odd | 2 |
| 5 | odd | 3 |
| 7 | odd | 4 |
| 9 | odd | 5 |
+----+----------+--------------------+
select x, property,
avg(x) over
(
partition by property
order by x
range between unbounded preceding and current row
) as 'cumulative average'
from int_t where property in ('odd','even');
+----+----------+--------------------+
| x | property | cumulative average |
+----+----------+--------------------+
| 2 | even | 2 |
| 4 | even | 3 |
| 6 | even | 4 |
| 8 | even | 5 |
| 10 | even | 6 |
| 1 | odd | 1 |
| 3 | odd | 2 |
| 5 | odd | 3 |
| 7 | odd | 4 |
| 9 | odd | 5 |
+----+----------+--------------------+
select x, property,
avg(x) over
(
partition by property
order by x
rows between unbounded preceding and current row
) as 'cumulative average'
from int_t where property in ('odd','even');
+----+----------+--------------------+
| x | property | cumulative average |
+----+----------+--------------------+
| 2 | even | 2 |
| 4 | even | 3 |
| 6 | even | 4 |
| 8 | even | 5 |
| 10 | even | 6 |
| 1 | odd | 1 |
| 3 | odd | 2 |
| 5 | odd | 3 |
| 7 | odd | 4 |
| 9 | odd | 5 |
+----+----------+--------------------+
The following examples show how to construct a moving window, with a running average taking into account 1 row
before and 1 row after the current row, within the same partition (all the even values or all the odd values). Because
of a restriction in the Impala RANGE syntax, this type of moving window is possible with the ROWS BETWEEN clause
but not the RANGE BETWEEN clause:
select x, property,
avg(x) over
(
partition by property
order by x
rows between 1 preceding and 1 following
) as 'moving average'
from int_t where property in ('odd','even');
+----+----------+----------------+
| x | property | moving average |
+----+----------+----------------+
| 2 | even | 3 |
| 4 | even | 4 |
| 6 | even | 6 |
| 8 | even | 8 |
| 10 | even | 9 |
| 1 | odd | 2 |
| 3 | odd | 3 |
| 5 | odd | 5 |
| 7 | odd | 7 |
| 9 | odd | 8 |
+----+----------+----------------+
Restrictions:
Due to the way arithmetic on FLOAT and DOUBLE columns uses high-performance hardware instructions, and distributed
queries can perform these operations in different order for each query, results can vary slightly for aggregate function
calls such as SUM() and AVG() for FLOAT and DOUBLE columns, particularly on large data sets where millions or billions
of values are summed or averaged. For perfect consistency and repeatability, use the DECIMAL data type for such
operations instead of FLOAT or DOUBLE.
Related information:
Impala Analytic Functions on page 429, MAX Function on page 414, MIN Function on page 417
COUNT Function
An aggregate function that returns the number of rows, or the number of non-NULL rows.
Syntax:
Depending on the argument, COUNT() considers rows that meet certain conditions:
• The notation COUNT(*) includes NULL values in the total.
• The notation COUNT(column_name) only considers rows where the column contains a non-NULL value.
• You can also combine COUNT with the DISTINCT operator to eliminate duplicates before counting, and to count
the combinations of values across multiple columns.
When the query contains a GROUP BY clause, returns one value for each combination of grouping values.
Return type: BIGINT
Complex type considerations:
To access a column with a complex type (ARRAY, STRUCT, or MAP) in an aggregation function, you unpack the individual
elements using join notation in the query, and then apply the function to the final scalar item, field, key, or value at
the bottom of any nested type hierarchy in the column. See Complex Types (CDH 5.5 or higher only) on page 156 for
details about using complex types in Impala.
The following example demonstrates calls to several aggregation functions using values from a column containing
nested complex types (an ARRAY of STRUCT items). The array is unpacked inside the query using join notation. The
array elements are referenced using the ITEM pseudocolumn, and the structure fields inside the array elements are
referenced using dot notation. Numeric values such as SUM() and AVG() are computed using the numeric R_NATIONKEY
field, and the general-purpose MAX() and MIN() values are computed from the string N_NAME field.
describe region;
+-------------+-------------------------+---------+
| name | type | comment |
+-------------+-------------------------+---------+
| r_regionkey | smallint | |
| r_name | string | |
| r_comment | string | |
| r_nations | array<struct< | |
| | n_nationkey:smallint, | |
| | n_name:string, | |
| | n_comment:string | |
| | >> | |
+-------------+-------------------------+---------+
| ASIA | 12 |
| ASIA | 18 |
| ASIA | 21 |
| EUROPE | 6 |
| EUROPE | 7 |
| EUROPE | 19 |
| EUROPE | 22 |
| EUROPE | 23 |
| MIDDLE EAST | 4 |
| MIDDLE EAST | 10 |
| MIDDLE EAST | 11 |
| MIDDLE EAST | 13 |
| MIDDLE EAST | 20 |
+-------------+------------------+
select
r_name,
count(r_nations.item.n_nationkey) as count,
sum(r_nations.item.n_nationkey) as sum,
avg(r_nations.item.n_nationkey) as avg,
min(r_nations.item.n_name) as minimum,
max(r_nations.item.n_name) as maximum,
ndv(r_nations.item.n_nationkey) as distinct_vals
from
region, region.r_nations as r_nations
group by r_name
order by r_name;
+-------------+-------+-----+------+-----------+----------------+---------------+
| r_name | count | sum | avg | minimum | maximum | distinct_vals |
+-------------+-------+-----+------+-----------+----------------+---------------+
| AFRICA | 5 | 50 | 10 | ALGERIA | MOZAMBIQUE | 5 |
| AMERICA | 5 | 47 | 9.4 | ARGENTINA | UNITED STATES | 5 |
| ASIA | 5 | 68 | 13.6 | CHINA | VIETNAM | 5 |
| EUROPE | 5 | 77 | 15.4 | FRANCE | UNITED KINGDOM | 5 |
| MIDDLE EAST | 5 | 58 | 11.6 | EGYPT | SAUDI ARABIA | 5 |
+-------------+-------+-----+------+-----------+----------------+---------------+
Examples:
-- How many rows total are in the table, regardless of NULL values?
select count(*) from t1;
-- How many rows are in the table with non-NULL values for a column?
select count(c1) from t1;
-- Count the rows that meet certain conditions.
-- Again, * includes NULLs, so COUNT(*) might be greater than COUNT(col).
select count(*) from t1 where x > 10;
select count(c1) from t1 where x > 10;
-- Can also be used in combination with DISTINCT and/or GROUP BY.
-- Combine COUNT and DISTINCT to find the number of unique values.
-- Must use column names rather than * with COUNT(DISTINCT ...) syntax.
-- Rows with NULL values are not counted.
select count(distinct c1) from t1;
-- Rows with a NULL value in _either_ column are not counted.
select count(distinct c1, c2) from t1;
-- Return more than one result.
select month, year, count(distinct visitor_id) from web_stats group by month, year;
The following examples show how to use COUNT() in an analytic context. They use a table containing integers from 1
to 10. Notice how the COUNT() is reported for each input value, as opposed to the GROUP BY clause which condenses
the result set.
select x, property, count(x) over (partition by property) as count from int_t where
property in ('odd','even');
+----+----------+-------+
| x | property | count |
+----+----------+-------+
| 2 | even | 5 |
| 4 | even | 5 |
| 6 | even | 5 |
| 8 | even | 5 |
| 10 | even | 5 |
| 1 | odd | 5 |
| 3 | odd | 5 |
| 5 | odd | 5 |
| 7 | odd | 5 |
| 9 | odd | 5 |
+----+----------+-------+
Adding an ORDER BY clause lets you experiment with results that are cumulative or apply to a moving set of rows (the
“window”). The following examples use COUNT() in an analytic context (that is, with an OVER() clause) to produce a
running count of all the even values, then a running count of all the odd values. The basic ORDER BY x clause implicitly
activates a window clause of RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, which is effectively
the same as ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, therefore all of these examples produce
the same results:
select x, property,
count(x) over (partition by property order by x) as 'cumulative count'
from int_t where property in ('odd','even');
+----+----------+------------------+
| x | property | cumulative count |
+----+----------+------------------+
| 2 | even | 1 |
| 4 | even | 2 |
| 6 | even | 3 |
| 8 | even | 4 |
| 10 | even | 5 |
| 1 | odd | 1 |
| 3 | odd | 2 |
| 5 | odd | 3 |
| 7 | odd | 4 |
| 9 | odd | 5 |
+----+----------+------------------+
select x, property,
count(x) over
(
partition by property
order by x
range between unbounded preceding and current row
) as 'cumulative total'
from int_t where property in ('odd','even');
+----+----------+------------------+
| x | property | cumulative count |
+----+----------+------------------+
| 2 | even | 1 |
| 4 | even | 2 |
| 6 | even | 3 |
| 8 | even | 4 |
| 10 | even | 5 |
| 1 | odd | 1 |
| 3 | odd | 2 |
| 5 | odd | 3 |
| 7 | odd | 4 |
| 9 | odd | 5 |
+----+----------+------------------+
select x, property,
count(x) over
(
partition by property
order by x
rows between unbounded preceding and current row
) as 'cumulative total'
from int_t where property in ('odd','even');
+----+----------+------------------+
| x | property | cumulative count |
+----+----------+------------------+
| 2 | even | 1 |
| 4 | even | 2 |
| 6 | even | 3 |
| 8 | even | 4 |
| 10 | even | 5 |
| 1 | odd | 1 |
| 3 | odd | 2 |
| 5 | odd | 3 |
| 7 | odd | 4 |
| 9 | odd | 5 |
+----+----------+------------------+
The following examples show how to construct a moving window, with a running count taking into account 1 row
before and 1 row after the current row, within the same partition (all the even values or all the odd values). Therefore,
the count is consistently 3 for rows in the middle of the window, and 2 for rows near the ends of the window, where
there is no preceding or no following row in the partition. Because of a restriction in the Impala RANGE syntax, this
type of moving window is possible with the ROWS BETWEEN clause but not the RANGE BETWEEN clause:
select x, property,
count(x) over
(
partition by property
order by x
rows between 1 preceding and 1 following
) as 'moving total'
from int_t where property in ('odd','even');
+----+----------+--------------+
| x | property | moving total |
+----+----------+--------------+
| 2 | even | 2 |
| 4 | even | 3 |
| 6 | even | 3 |
| 8 | even | 3 |
| 10 | even | 2 |
| 1 | odd | 2 |
| 3 | odd | 3 |
| 5 | odd | 3 |
| 7 | odd | 3 |
| 9 | odd | 2 |
+----+----------+--------------+
Note:
By default, Impala only allows a single COUNT(DISTINCT columns) expression in each query.
If you do not need precise accuracy, you can produce an estimate of the distinct values for a column
by specifying NDV(column); a query can contain multiple instances of NDV(column). To make Impala
automatically rewrite COUNT(DISTINCT) expressions to NDV(), enable the APPX_COUNT_DISTINCT
query option.
To produce the same result as multiple COUNT(DISTINCT) expressions, you can use the following
technique for queries involving a single table:
Because CROSS JOIN is an expensive operation, prefer to use the NDV() technique wherever practical.
Related information:
Impala Analytic Functions on page 429
GROUP_CONCAT Function
An aggregate function that returns a single string representing the argument value concatenated together for each
row of the result set. If the optional separator string is specified, the separator is added between each pair of
concatenated values. The default separator is a comma followed by a space.
Syntax:
Usage notes: concat() and concat_ws() are appropriate for concatenating the values of multiple columns within
the same row, while group_concat() joins together values from different rows.
By default, returns a single string covering the whole result set. To include other columns or values in the result set,
or to produce multiple concatenated strings for subsets of rows, include a GROUP BY clause in the query.
Return type: STRING
Restrictions:
You cannot apply the DISTINCT operator to the argument of this function.
This function cannot be used in an analytic context. That is, the OVER() clause is not allowed at all with this function.
Currently, Impala returns an error if the result value grows larger than 1 GiB.
Examples:
The following examples illustrate various aspects of the GROUP_CONCAT() function.
You can call the function directly on a STRING column. To use it with a numeric column, cast the value to STRING.
The optional separator lets you format the result in flexible ways. The separator can be an arbitrary string expression,
not just a single character.
The default separator is a comma followed by a space. To get a comma-delimited result without extra spaces, specify
a delimiter character that is only a comma.
Including a GROUP BY clause lets you produce a different concatenated result for each group in the result set. In this
example, the only X value that occurs more than once is 1, so that is the only row in the result set where
GROUP_CONCAT() returns a delimited value. For groups containing a single value, GROUP_CONCAT() returns the
original value of its STRING argument.
MAX Function
An aggregate function that returns the maximum value from a set of numbers. Opposite of the MIN function. Its single
argument can be numeric column, or the numeric result of a function or expression applied to the column value. Rows
with a NULL value for the specified column are ignored. If the table is empty, or all the values supplied to MAX are NULL,
MAX returns NULL.
Syntax:
When the query contains a GROUP BY clause, returns one value for each combination of grouping values.
Restrictions: In Impala 2.0 and higher, this function can be used as an analytic function, but with restrictions on any
window clause. For MAX() and MIN(), the window clause is only allowed if the start bound is UNBOUNDED PRECEDING.
Return type: Same as the input value, except for CHAR and VARCHAR arguments which produce a STRING result
Complex type considerations:
To access a column with a complex type (ARRAY, STRUCT, or MAP) in an aggregation function, you unpack the individual
elements using join notation in the query, and then apply the function to the final scalar item, field, key, or value at
the bottom of any nested type hierarchy in the column. See Complex Types (CDH 5.5 or higher only) on page 156 for
details about using complex types in Impala.
The following example demonstrates calls to several aggregation functions using values from a column containing
nested complex types (an ARRAY of STRUCT items). The array is unpacked inside the query using join notation. The
array elements are referenced using the ITEM pseudocolumn, and the structure fields inside the array elements are
referenced using dot notation. Numeric values such as SUM() and AVG() are computed using the numeric R_NATIONKEY
field, and the general-purpose MAX() and MIN() values are computed from the string N_NAME field.
describe region;
+-------------+-------------------------+---------+
| name | type | comment |
+-------------+-------------------------+---------+
| r_regionkey | smallint | |
| r_name | string | |
| r_comment | string | |
| r_nations | array<struct< | |
| | n_nationkey:smallint, | |
| | n_name:string, | |
| | n_comment:string | |
| | >> | |
+-------------+-------------------------+---------+
select
r_name,
count(r_nations.item.n_nationkey) as count,
sum(r_nations.item.n_nationkey) as sum,
avg(r_nations.item.n_nationkey) as avg,
min(r_nations.item.n_name) as minimum,
max(r_nations.item.n_name) as maximum,
ndv(r_nations.item.n_nationkey) as distinct_vals
from
region, region.r_nations as r_nations
group by r_name
order by r_name;
+-------------+-------+-----+------+-----------+----------------+---------------+
| r_name | count | sum | avg | minimum | maximum | distinct_vals |
+-------------+-------+-----+------+-----------+----------------+---------------+
| AFRICA | 5 | 50 | 10 | ALGERIA | MOZAMBIQUE | 5 |
| AMERICA | 5 | 47 | 9.4 | ARGENTINA | UNITED STATES | 5 |
| ASIA | 5 | 68 | 13.6 | CHINA | VIETNAM | 5 |
| EUROPE | 5 | 77 | 15.4 | FRANCE | UNITED KINGDOM | 5 |
| MIDDLE EAST | 5 | 58 | 11.6 | EGYPT | SAUDI ARABIA | 5 |
+-------------+-------+-----+------+-----------+----------------+---------------+
Examples:
The following examples show how to use MAX() in an analytic context. They use a table containing integers from 1 to
10. Notice how the MAX() is reported for each input value, as opposed to the GROUP BY clause which condenses the
result set.
select x, property, max(x) over (partition by property) as max from int_t where property
in ('odd','even');
+----+----------+-----+
| x | property | max |
+----+----------+-----+
| 2 | even | 10 |
| 4 | even | 10 |
| 6 | even | 10 |
| 8 | even | 10 |
| 10 | even | 10 |
| 1 | odd | 9 |
| 3 | odd | 9 |
| 5 | odd | 9 |
| 7 | odd | 9 |
| 9 | odd | 9 |
+----+----------+-----+
Adding an ORDER BY clause lets you experiment with results that are cumulative or apply to a moving set of rows (the
“window”). The following examples use MAX() in an analytic context (that is, with an OVER() clause) to display the
smallest value of X encountered up to each row in the result set. The examples use two columns in the ORDER BY
clause to produce a sequence of values that rises and falls, to illustrate how the MAX() result only increases or stays
the same throughout each partition within the result set. The basic ORDER BY x clause implicitly activates a window
clause of RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, which is effectively the same as ROWS
BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, therefore all of these examples produce the same results:
select x, property,
max(x) over (order by property, x desc) as 'maximum to this point'
from int_t where property in ('prime','square');
+---+----------+-----------------------+
| x | property | maximum to this point |
+---+----------+-----------------------+
| 7 | prime | 7 |
| 5 | prime | 7 |
| 3 | prime | 7 |
| 2 | prime | 7 |
| 9 | square | 9 |
| 4 | square | 9 |
| 1 | square | 9 |
+---+----------+-----------------------+
select x, property,
max(x) over
(
order by property, x desc
rows between unbounded preceding and current row
) as 'maximum to this point'
from int_t where property in ('prime','square');
+---+----------+-----------------------+
| x | property | maximum to this point |
+---+----------+-----------------------+
| 7 | prime | 7 |
| 5 | prime | 7 |
| 3 | prime | 7 |
| 2 | prime | 7 |
| 9 | square | 9 |
| 4 | square | 9 |
| 1 | square | 9 |
+---+----------+-----------------------+
select x, property,
max(x) over
(
order by property, x desc
range between unbounded preceding and current row
) as 'maximum to this point'
from int_t where property in ('prime','square');
+---+----------+-----------------------+
| x | property | maximum to this point |
+---+----------+-----------------------+
| 7 | prime | 7 |
| 5 | prime | 7 |
| 3 | prime | 7 |
| 2 | prime | 7 |
| 9 | square | 9 |
| 4 | square | 9 |
| 1 | square | 9 |
+---+----------+-----------------------+
The following examples show how to construct a moving window, with a running maximum taking into account all
rows before and 1 row after the current row. Because of a restriction in the Impala RANGE syntax, this type of moving
window is possible with the ROWS BETWEEN clause but not the RANGE BETWEEN clause. Because of an extra Impala
restriction on the MAX() and MIN() functions in an analytic context, the lower bound must be UNBOUNDED PRECEDING.
select x, property,
max(x) over
(
order by property, x
rows between unbounded preceding and 1 following
) as 'local maximum'
from int_t where property in ('prime','square');
+---+----------+---------------+
| x | property | local maximum |
+---+----------+---------------+
| 2 | prime | 3 |
| 3 | prime | 5 |
| 5 | prime | 7 |
| 7 | prime | 7 |
| 1 | square | 7 |
| 4 | square | 9 |
| 9 | square | 9 |
+---+----------+---------------+
Related information:
Impala Analytic Functions on page 429, MIN Function on page 417, AVG Function on page 405
MIN Function
An aggregate function that returns the minimum value from a set of numbers. Opposite of the MAX function. Its single
argument can be numeric column, or the numeric result of a function or expression applied to the column value. Rows
with a NULL value for the specified column are ignored. If the table is empty, or all the values supplied to MIN are NULL,
MIN returns NULL.
Syntax:
When the query contains a GROUP BY clause, returns one value for each combination of grouping values.
Restrictions: In Impala 2.0 and higher, this function can be used as an analytic function, but with restrictions on any
window clause. For MAX() and MIN(), the window clause is only allowed if the start bound is UNBOUNDED PRECEDING.
Return type: Same as the input value, except for CHAR and VARCHAR arguments which produce a STRING result
describe region;
+-------------+-------------------------+---------+
| name | type | comment |
+-------------+-------------------------+---------+
| r_regionkey | smallint | |
| r_name | string | |
| r_comment | string | |
| r_nations | array<struct< | |
| | n_nationkey:smallint, | |
| | n_name:string, | |
| | n_comment:string | |
| | >> | |
+-------------+-------------------------+---------+
select
r_name,
count(r_nations.item.n_nationkey) as count,
sum(r_nations.item.n_nationkey) as sum,
avg(r_nations.item.n_nationkey) as avg,
min(r_nations.item.n_name) as minimum,
max(r_nations.item.n_name) as maximum,
ndv(r_nations.item.n_nationkey) as distinct_vals
from
region, region.r_nations as r_nations
group by r_name
order by r_name;
+-------------+-------+-----+------+-----------+----------------+---------------+
| r_name | count | sum | avg | minimum | maximum | distinct_vals |
+-------------+-------+-----+------+-----------+----------------+---------------+
| AFRICA | 5 | 50 | 10 | ALGERIA | MOZAMBIQUE | 5 |
| AMERICA | 5 | 47 | 9.4 | ARGENTINA | UNITED STATES | 5 |
| ASIA | 5 | 68 | 13.6 | CHINA | VIETNAM | 5 |
| EUROPE | 5 | 77 | 15.4 | FRANCE | UNITED KINGDOM | 5 |
| MIDDLE EAST | 5 | 58 | 11.6 | EGYPT | SAUDI ARABIA | 5 |
+-------------+-------+-----+------+-----------+----------------+---------------+
Examples:
The following examples show how to use MIN() in an analytic context. They use a table containing integers from 1 to
10. Notice how the MIN() is reported for each input value, as opposed to the GROUP BY clause which condenses the
result set.
select x, property, min(x) over (partition by property) as min from int_t where property
in ('odd','even');
+----+----------+-----+
| x | property | min |
+----+----------+-----+
| 2 | even | 2 |
| 4 | even | 2 |
| 6 | even | 2 |
| 8 | even | 2 |
| 10 | even | 2 |
| 1 | odd | 1 |
| 3 | odd | 1 |
| 5 | odd | 1 |
| 7 | odd | 1 |
| 9 | odd | 1 |
+----+----------+-----+
Adding an ORDER BY clause lets you experiment with results that are cumulative or apply to a moving set of rows (the
“window”). The following examples use MIN() in an analytic context (that is, with an OVER() clause) to display the
smallest value of X encountered up to each row in the result set. The examples use two columns in the ORDER BY
clause to produce a sequence of values that rises and falls, to illustrate how the MIN() result only decreases or stays
the same throughout each partition within the result set. The basic ORDER BY x clause implicitly activates a window
clause of RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, which is effectively the same as ROWS
BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, therefore all of these examples produce the same results:
select x, property, min(x) over (order by property, x desc) as 'minimum to this point'
from int_t where property in ('prime','square');
+---+----------+-----------------------+
| x | property | minimum to this point |
+---+----------+-----------------------+
| 7 | prime | 7 |
| 5 | prime | 5 |
| 3 | prime | 3 |
| 2 | prime | 2 |
| 9 | square | 2 |
| 4 | square | 2 |
| 1 | square | 1 |
+---+----------+-----------------------+
select x, property,
min(x) over
(
order by property, x desc
range between unbounded preceding and current row
) as 'minimum to this point'
from int_t where property in ('prime','square');
+---+----------+-----------------------+
| x | property | minimum to this point |
+---+----------+-----------------------+
| 7 | prime | 7 |
| 5 | prime | 5 |
| 3 | prime | 3 |
| 2 | prime | 2 |
| 9 | square | 2 |
| 4 | square | 2 |
| 1 | square | 1 |
+---+----------+-----------------------+
select x, property,
min(x) over
(
order by property, x desc
rows between unbounded preceding and current row
) as 'minimum to this point'
from int_t where property in ('prime','square');
+---+----------+-----------------------+
| x | property | minimum to this point |
+---+----------+-----------------------+
| 7 | prime | 7 |
| 5 | prime | 5 |
| 3 | prime | 3 |
| 2 | prime | 2 |
| 9 | square | 2 |
| 4 | square | 2 |
| 1 | square | 1 |
+---+----------+-----------------------+
The following examples show how to construct a moving window, with a running minimum taking into account all rows
before and 1 row after the current row. Because of a restriction in the Impala RANGE syntax, this type of moving window
is possible with the ROWS BETWEEN clause but not the RANGE BETWEEN clause. Because of an extra Impala restriction
on the MAX() and MIN() functions in an analytic context, the lower bound must be UNBOUNDED PRECEDING.
select x, property,
min(x) over
(
order by property, x desc
rows between unbounded preceding and 1 following
) as 'local minimum'
from int_t where property in ('prime','square');
+---+----------+---------------+
| x | property | local minimum |
+---+----------+---------------+
| 7 | prime | 5 |
| 5 | prime | 3 |
| 3 | prime | 2 |
| 2 | prime | 2 |
| 9 | square | 2 |
| 4 | square | 1 |
| 1 | square | 1 |
+---+----------+---------------+
ERROR: AnalysisException: RANGE is only supported with both the lower and upper bounds
UNBOUNDED or one UNBOUNDED and the other CURRENT ROW.
Related information:
Impala Analytic Functions on page 429, MAX Function on page 414, AVG Function on page 405
NDV Function
An aggregate function that returns an approximate value similar to the result of COUNT(DISTINCT col), the “number
of distinct values”. It is much faster than the combination of COUNT and DISTINCT, and uses a constant amount of
memory and thus is less memory-intensive for columns with high cardinality.
Syntax:
Usage notes:
This is the mechanism used internally by the COMPUTE STATS statement for computing the number of distinct values
in a column.
Because this number is an estimate, it might not reflect the precise number of different values in the column, especially
if the cardinality is very low or very high. If the estimated number is higher than the number of rows in the table, Impala
adjusts the value internally during query planning.
Return type: DOUBLE in Impala 2.0 and higher; STRING in earlier releases
Complex type considerations:
To access a column with a complex type (ARRAY, STRUCT, or MAP) in an aggregation function, you unpack the individual
elements using join notation in the query, and then apply the function to the final scalar item, field, key, or value at
the bottom of any nested type hierarchy in the column. See Complex Types (CDH 5.5 or higher only) on page 156 for
details about using complex types in Impala.
The following example demonstrates calls to several aggregation functions using values from a column containing
nested complex types (an ARRAY of STRUCT items). The array is unpacked inside the query using join notation. The
array elements are referenced using the ITEM pseudocolumn, and the structure fields inside the array elements are
referenced using dot notation. Numeric values such as SUM() and AVG() are computed using the numeric R_NATIONKEY
field, and the general-purpose MAX() and MIN() values are computed from the string N_NAME field.
describe region;
+-------------+-------------------------+---------+
| name | type | comment |
+-------------+-------------------------+---------+
| r_regionkey | smallint | |
| r_name | string | |
| r_comment | string | |
| r_nations | array<struct< | |
| | n_nationkey:smallint, | |
| | n_name:string, | |
| | n_comment:string | |
| | >> | |
+-------------+-------------------------+---------+
| AMERICA | 2 |
| AMERICA | 3 |
| AMERICA | 17 |
| AMERICA | 24 |
| ASIA | 8 |
| ASIA | 9 |
| ASIA | 12 |
| ASIA | 18 |
| ASIA | 21 |
| EUROPE | 6 |
| EUROPE | 7 |
| EUROPE | 19 |
| EUROPE | 22 |
| EUROPE | 23 |
| MIDDLE EAST | 4 |
| MIDDLE EAST | 10 |
| MIDDLE EAST | 11 |
| MIDDLE EAST | 13 |
| MIDDLE EAST | 20 |
+-------------+------------------+
select
r_name,
count(r_nations.item.n_nationkey) as count,
sum(r_nations.item.n_nationkey) as sum,
avg(r_nations.item.n_nationkey) as avg,
min(r_nations.item.n_name) as minimum,
max(r_nations.item.n_name) as maximum,
ndv(r_nations.item.n_nationkey) as distinct_vals
from
region, region.r_nations as r_nations
group by r_name
order by r_name;
+-------------+-------+-----+------+-----------+----------------+---------------+
| r_name | count | sum | avg | minimum | maximum | distinct_vals |
+-------------+-------+-----+------+-----------+----------------+---------------+
| AFRICA | 5 | 50 | 10 | ALGERIA | MOZAMBIQUE | 5 |
| AMERICA | 5 | 47 | 9.4 | ARGENTINA | UNITED STATES | 5 |
| ASIA | 5 | 68 | 13.6 | CHINA | VIETNAM | 5 |
| EUROPE | 5 | 77 | 15.4 | FRANCE | UNITED KINGDOM | 5 |
| MIDDLE EAST | 5 | 58 | 11.6 | EGYPT | SAUDI ARABIA | 5 |
+-------------+-------+-----+------+-----------+----------------+---------------+
Restrictions:
This function cannot be used in an analytic context. That is, the OVER() clause is not allowed at all with this function.
Examples:
The following example queries a billion-row table to illustrate the relative performance of COUNT(DISTINCT) and
NDV(). It shows how COUNT(DISTINCT) gives a precise answer, but is inefficient for large-scale data where an
approximate result is sufficient. The NDV() function gives an approximate result but is much faster.
The following example shows how you can code multiple NDV() calls in a single query, to easily learn which columns
have substantially more or fewer distinct values. This technique is faster than running a sequence of queries with
COUNT(DISTINCT) calls.
This example demonstrates how STDDEV() and STDDEV_SAMP() return the same result, while STDDEV_POP() uses
a slightly different calculation to reflect that the input data is considered part of a larger “population”.
This example demonstrates that, because the return value of these aggregate functions is a STRING, you must currently
convert the result with CAST.
Restrictions:
This function cannot be used in an analytic context. That is, the OVER() clause is not allowed at all with this function.
Related information:
The STDDEV(), STDDEV_POP(), and STDDEV_SAMP() functions compute the standard deviation (square root of the
variance) based on the results of VARIANCE(), VARIANCE_POP(), and VARIANCE_SAMP() respectively. See VARIANCE,
VARIANCE_SAMP, VARIANCE_POP, VAR_SAMP, VAR_POP Functions on page 428 for details about the variance property.
SUM Function
An aggregate function that returns the sum of a set of numbers. Its single argument can be numeric column, or the
numeric result of a function or expression applied to the column value. Rows with a NULL value for the specified column
are ignored. If the table is empty, or all the values supplied to MIN are NULL, SUM returns NULL.
Syntax:
When the query contains a GROUP BY clause, returns one value for each combination of grouping values.
Return type: BIGINT for integer arguments, DOUBLE for floating-point arguments
Complex type considerations:
To access a column with a complex type (ARRAY, STRUCT, or MAP) in an aggregation function, you unpack the individual
elements using join notation in the query, and then apply the function to the final scalar item, field, key, or value at
the bottom of any nested type hierarchy in the column. See Complex Types (CDH 5.5 or higher only) on page 156 for
details about using complex types in Impala.
The following example demonstrates calls to several aggregation functions using values from a column containing
nested complex types (an ARRAY of STRUCT items). The array is unpacked inside the query using join notation. The
array elements are referenced using the ITEM pseudocolumn, and the structure fields inside the array elements are
referenced using dot notation. Numeric values such as SUM() and AVG() are computed using the numeric R_NATIONKEY
field, and the general-purpose MAX() and MIN() values are computed from the string N_NAME field.
describe region;
+-------------+-------------------------+---------+
| name | type | comment |
+-------------+-------------------------+---------+
| r_regionkey | smallint | |
| r_name | string | |
| r_comment | string | |
| r_nations | array<struct< | |
| | n_nationkey:smallint, | |
| | n_name:string, | |
| | n_comment:string | |
| | >> | |
+-------------+-------------------------+---------+
select
r_name,
count(r_nations.item.n_nationkey) as count,
sum(r_nations.item.n_nationkey) as sum,
avg(r_nations.item.n_nationkey) as avg,
min(r_nations.item.n_name) as minimum,
max(r_nations.item.n_name) as maximum,
ndv(r_nations.item.n_nationkey) as distinct_vals
from
region, region.r_nations as r_nations
group by r_name
order by r_name;
+-------------+-------+-----+------+-----------+----------------+---------------+
| r_name | count | sum | avg | minimum | maximum | distinct_vals |
+-------------+-------+-----+------+-----------+----------------+---------------+
| AFRICA | 5 | 50 | 10 | ALGERIA | MOZAMBIQUE | 5 |
Examples:
The following example shows how to use SUM() to compute the total for all the values in the table, a subset of values,
or the sum for each combination of values in the GROUP BY clause:
The following examples show how to use SUM() in an analytic context. They use a table containing integers from 1 to
10. Notice how the SUM() is reported for each input value, as opposed to the GROUP BY clause which condenses the
result set.
select x, property, sum(x) over (partition by property) as sum from int_t where property
in ('odd','even');
+----+----------+-----+
| x | property | sum |
+----+----------+-----+
| 2 | even | 30 |
| 4 | even | 30 |
| 6 | even | 30 |
| 8 | even | 30 |
| 10 | even | 30 |
| 1 | odd | 25 |
| 3 | odd | 25 |
| 5 | odd | 25 |
| 7 | odd | 25 |
| 9 | odd | 25 |
+----+----------+-----+
Adding an ORDER BY clause lets you experiment with results that are cumulative or apply to a moving set of rows (the
“window”). The following examples use SUM() in an analytic context (that is, with an OVER() clause) to produce a
running total of all the even values, then a running total of all the odd values. The basic ORDER BY x clause implicitly
activates a window clause of RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, which is effectively
the same as ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, therefore all of these examples produce
the same results:
select x, property,
sum(x) over (partition by property order by x) as 'cumulative total'
from int_t where property in ('odd','even');
+----+----------+------------------+
| x | property | cumulative total |
+----+----------+------------------+
| 2 | even | 2 |
| 4 | even | 6 |
| 6 | even | 12 |
| 8 | even | 20 |
| 10 | even | 30 |
| 1 | odd | 1 |
| 3 | odd | 4 |
| 5 | odd | 9 |
| 7 | odd | 16 |
| 9 | odd | 25 |
+----+----------+------------------+
select x, property,
sum(x) over
(
partition by property
order by x
range between unbounded preceding and current row
) as 'cumulative total'
from int_t where property in ('odd','even');
+----+----------+------------------+
| x | property | cumulative total |
+----+----------+------------------+
| 2 | even | 2 |
| 4 | even | 6 |
| 6 | even | 12 |
| 8 | even | 20 |
| 10 | even | 30 |
| 1 | odd | 1 |
| 3 | odd | 4 |
| 5 | odd | 9 |
| 7 | odd | 16 |
| 9 | odd | 25 |
+----+----------+------------------+
select x, property,
sum(x) over
(
partition by property
order by x
rows between unbounded preceding and current row
) as 'cumulative total'
from int_t where property in ('odd','even');
+----+----------+------------------+
| x | property | cumulative total |
+----+----------+------------------+
| 2 | even | 2 |
| 4 | even | 6 |
| 6 | even | 12 |
| 8 | even | 20 |
| 10 | even | 30 |
| 1 | odd | 1 |
| 3 | odd | 4 |
| 5 | odd | 9 |
| 7 | odd | 16 |
| 9 | odd | 25 |
+----+----------+------------------+
Changing the direction of the ORDER BY clause causes the intermediate results of the cumulative total to be calculated
in a different order:
The following examples show how to construct a moving window, with a running total taking into account 1 row before
and 1 row after the current row, within the same partition (all the even values or all the odd values). Because of a
restriction in the Impala RANGE syntax, this type of moving window is possible with the ROWS BETWEEN clause but not
the RANGE BETWEEN clause:
select x, property,
sum(x) over
(
partition by property
order by x
rows between 1 preceding and 1 following
) as 'moving total'
from int_t where property in ('odd','even');
+----+----------+--------------+
| x | property | moving total |
+----+----------+--------------+
| 2 | even | 6 |
| 4 | even | 12 |
| 6 | even | 18 |
| 8 | even | 24 |
| 10 | even | 18 |
| 1 | odd | 4 |
| 3 | odd | 9 |
| 5 | odd | 15 |
| 7 | odd | 21 |
| 9 | odd | 16 |
+----+----------+--------------+
Restrictions:
Due to the way arithmetic on FLOAT and DOUBLE columns uses high-performance hardware instructions, and distributed
queries can perform these operations in different order for each query, results can vary slightly for aggregate function
calls such as SUM() and AVG() for FLOAT and DOUBLE columns, particularly on large data sets where millions or billions
of values are summed or averaged. For perfect consistency and repeatability, use the DECIMAL data type for such
operations instead of FLOAT or DOUBLE.
Related information:
Impala Analytic Functions on page 429
rows matched by the query. The difference is that STDDEV_SAMP() is scaled by 1/(N-1) while STDDEV_POP() is
scaled by 1/N.
The functions VAR_SAMP() and VAR_POP() are the same as VARIANCE_SAMP() and VARIANCE_POP(), respectively.
These aliases are available in Impala 2.0 and later.
If no input rows match the query, the result of any of these functions is NULL. If a single input row matches the query,
the result of any of these functions is "0.0".
Examples:
This example demonstrates how VARIANCE() and VARIANCE_SAMP() return the same result, while VARIANCE_POP()
uses a slightly different calculation to reflect that the input data is considered part of a larger “population”.
This example demonstrates that, because the return value of these aggregate functions is a STRING, you convert the
result with CAST if you need to do further calculations as a numeric value.
Restrictions:
This function cannot be used in an analytic context. That is, the OVER() clause is not allowed at all with this function.
Related information:
The STDDEV(), STDDEV_POP(), and STDDEV_SAMP() functions compute the standard deviation (square root of the
variance) based on the results of VARIANCE(), VARIANCE_POP(), and VARIANCE_SAMP() respectively. See STDDEV,
STDDEV_SAMP, STDDEV_POP Functions on page 423 for details about the standard deviation property.
OVER Clause
The OVER clause is required for calls to pure analytic functions such as LEAD(), RANK(), and FIRST_VALUE(). When
you include an OVER clause with calls to aggregate functions such as MAX(), COUNT(), or SUM(), they operate as
analytic functions.
Syntax:
PARTITION BY clause:
The PARTITION BY clause acts much like the GROUP BY clause in the outermost block of a query. It divides the rows
into groups containing identical values in one or more columns. These logical groups are known as partitions. Throughout
the discussion of analytic functions, “partitions” refers to the groups produced by the PARTITION BY clause, not to
partitioned tables. However, note the following limitation that applies specifically to analytic function calls involving
partitioned tables.
In queries involving both analytic functions and partitioned tables, partition pruning only occurs for columns named
in the PARTITION BY clause of the analytic function call. For example, if an analytic function query has a clause such
as WHERE year=2016, the way to make the query prune all other YEAR partitions is to include PARTITION BY year
in the analytic function call; for example, OVER (PARTITION BY year,other_columns
other_analytic_clauses).
The sequence of results from an analytic function “resets” for each new partition in the result set. That is, the set of
preceding or following rows considered by the analytic function always come from a single partition. Any MAX(),
SUM(), ROW_NUMBER(), and so on apply to each partition independently. Omit the PARTITION BY clause to apply
the analytic operation to all the rows in the table.
ORDER BY clause:
The ORDER BY clause works much like the ORDER BY clause in the outermost block of a query. It defines the order in
which rows are evaluated for the entire input set, or for each group produced by a PARTITION BY clause. You can
order by one or multiple expressions, and for each expression optionally choose ascending or descending order and
whether nulls come first or last in the sort order. Because this ORDER BY clause only defines the order in which rows
are evaluated, if you want the results to be output in a specific order, also include an ORDER BY clause in the outer
block of the query.
When the ORDER BY clause is omitted, the analytic function applies to all items in the group produced by the PARTITION
BY clause. When the ORDER BY clause is included, the analysis can apply to all or a subset of the items in the group,
depending on the optional window clause.
The order in which the rows are analyzed is only defined for those columns specified in ORDER BY clauses.
One difference between the analytic and outer uses of the ORDER BY clause: inside the OVER clause, ORDER BY 1 or
other integer value is interpreted as a constant sort value (effectively a no-op) rather than referring to column 1.
Window clause:
The window clause is only allowed in combination with an ORDER BY clause. If the ORDER BY clause is specified but
the window clause is not, the default window is RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW. See
Window Clause on page 432 for full details.
HBase considerations:
Because HBase tables are optimized for single-row lookups rather than full scans, analytic functions using the OVER()
clause are not recommended for HBase tables. Although such queries work, their performance is lower than on
comparable tables using HDFS data files.
Parquet considerations:
Analytic functions are very efficient for Parquet tables. The data that is examined during evaluation of the OVER()
clause comes from a specified set of columns, and the values for each column are arranged sequentially within each
data file.
Text table considerations:
Analytic functions are convenient to use with text tables for exploratory business intelligence. When the volume of
data is substantial, prefer to use Parquet tables for performance-critical analytic queries.
Added in: CDH 5.2.0 (Impala 2.0.0)
Examples:
The following example shows how to synthesize a numeric sequence corresponding to all the rows in a table. The new
table has the same columns as the old one, plus an additional column ID containing the integers 1, 2, 3, and so on,
corresponding to the order of a TIMESTAMP column in the original table.
The following example shows how to determine the number of rows containing each value for a column. Unlike a
corresponding GROUP BY query, this one can analyze a single column and still return all values (not just the distinct
ones) from the other columns.
SELECT x, y, z,
count() OVER (PARTITION BY x) AS how_many_x
FROM t1;
Restrictions:
You cannot directly combine the DISTINCT operator with analytic function calls. You can put the analytic function call
in a WITH clause or an inline view, and apply the DISTINCT operator to its result set.
Window Clause
Certain analytic functions accept an optional window clause, which makes the function analyze only certain rows
“around” the current row rather than all rows in the partition. For example, you can get a moving average by specifying
some number of preceding and following rows, or a running count or running total by specifying all rows up to the
current position. This clause can result in different analytic results for rows within the same partition.
The window clause is supported with the AVG(), COUNT(), FIRST_VALUE(), LAST_VALUE(), and SUM() functions.
For MAX() and MIN(), the window clause only allowed if the start bound is UNBOUNDED PRECEDING
Syntax:
ROWS BETWEEN [ { m | UNBOUNDED } PRECEDING | CURRENT ROW] [ AND [CURRENT ROW | { UNBOUNDED
| n } FOLLOWING] ]
RANGE BETWEEN [ {m | UNBOUNDED } PRECEDING | CURRENT ROW] [ AND [CURRENT ROW | { UNBOUNDED
| n } FOLLOWING] ]
ROWS BETWEEN defines the size of the window in terms of the indexes of the rows in the result set. The size of the
window is predictable based on the clauses the position within the result set.
RANGE BETWEEN does not currently support numeric arguments to define a variable-size sliding window.
Currently, Impala supports only some combinations of arguments to the RANGE clause:
• RANGE BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW (the default when ORDER BY is specified and
the window clause is omitted)
• RANGE BETWEEN CURRENT ROW AND UNBOUNDED FOLLOWING
• RANGE BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING
When RANGE is used, CURRENT ROW includes not just the current row but all rows that are tied with the current row
based on the ORDER BY expressions.
Added in: CDH 5.2.0 (Impala 2.0.0)
Examples:
The following examples show financial data for a fictional stock symbol JDR. The closing price moves up and down
each day.
The queries use analytic functions with window clauses to compute moving averages of the closing price. For example,
ROWS BETWEEN 1 PRECEDING AND 1 FOLLOWING produces an average of the value from a 3-day span, producing
a different value for each row. The first row, which has no preceding row, only gets averaged with the row following
it. If the table contained more than one stock symbol, the PARTITION BY clause would limit the window for the
moving average to only consider the prices for a single stock.
The clause ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW produces a cumulative moving average,
from the earliest data up to the value for each day.
CUME_DIST (expr)
OVER ([partition_by_clause] order_by_clause)
The ORDER BY clause is required. The PARTITION BY clause is optional. The window clause is not allowed.
Usage notes:
Within each partition of the result set, the CUME_DIST() value represents an ascending sequence that ends at 1. Each
value represents the proportion of rows in the partition whose values are less than or equal to the value in the current
row.
If the sequence of input values contains ties, the CUME_DIST() results are identical for the tied values.
Impala only supports the CUME_DIST() function in an analytic context, not as a regular aggregate function.
Examples:
This example uses a table with 9 rows. The CUME_DIST() function evaluates the entire table because there is no
PARTITION BY clause, with the rows ordered by the weight of the animal. the sequence of values shows that 1/9 of
the values are less than or equal to the lightest animal (mouse), 2/9 of the values are less than or equal to the
second-lightest animal, and so on up to the heaviest animal (elephant), where 9/9 of the rows are less than or equal
to its weight.
Using a PARTITION BY clause produces a separate sequence for each partition group, in this case one for mammals
and one for birds. Because there are 3 birds and 6 mammals, the sequence illustrates how 1/3 of the “Bird” rows have
a kilos value that is less than or equal to the lightest bird, 1/6 of the “Mammal” rows have a kilos value that is less
than or equal to the lightest mammal, and so on until both the heaviest bird and heaviest mammal have a CUME_DIST()
value of 1.
select name, kind, cume_dist() over (partition by kind order by kilos) from animals
+------------+--------+-----------------------+
| name | kind | cume_dist() OVER(...) |
+------------+--------+-----------------------+
| Ostrich | Bird | 1 |
| Condor | Bird | 0.6666666666666666 |
| Owl | Bird | 0.3333333333333333 |
| Elephant | Mammal | 1 |
| Giraffe | Mammal | 0.8333333333333334 |
| Polar bear | Mammal | 0.6666666666666666 |
| Horse | Mammal | 0.5 |
| Housecat | Mammal | 0.3333333333333333 |
| Mouse | Mammal | 0.1666666666666667 |
+------------+--------+-----------------------+
We can reverse the ordering within each partition group by using an ORDER BY ... DESC clause within the OVER()
clause. Now the lightest (smallest value of kilos) animal of each kind has a CUME_DIST() value of 1.
select name, kind, cume_dist() over (partition by kind order by kilos desc) from animals
+------------+--------+-----------------------+
| name | kind | cume_dist() OVER(...) |
+------------+--------+-----------------------+
| Owl | Bird | 1 |
| Condor | Bird | 0.6666666666666666 |
| Ostrich | Bird | 0.3333333333333333 |
| Mouse | Mammal | 1 |
| Housecat | Mammal | 0.8333333333333334 |
| Horse | Mammal | 0.6666666666666666 |
| Polar bear | Mammal | 0.5 |
| Giraffe | Mammal | 0.3333333333333333 |
The following example manufactures some rows with identical values in the kilos column, to demonstrate how the
results look in case of tie values. For simplicity, it only shows the CUME_DIST() sequence for the “Bird” rows. Now
with 3 rows all with a value of 15, all of those rows have the same CUME_DIST() value. 4/5 of the rows have a value
for kilos that is less than or equal to 15.
insert into animals values ('California Condor', 'Bird', 15), ('Andean Condor', 'Bird',
15)
select name, kind, cume_dist() over (order by kilos) from animals where kind = 'Bird';
+-------------------+------+-----------------------+
| name | kind | cume_dist() OVER(...) |
+-------------------+------+-----------------------+
| Ostrich | Bird | 1 |
| Condor | Bird | 0.8 |
| California Condor | Bird | 0.8 |
| Andean Condor | Bird | 0.8 |
| Owl | Bird | 0.2 |
+-------------------+------+-----------------------+
The following example shows how to use an ORDER BY clause in the outer block to order the result set in case of ties.
Here, all the “Bird” rows are together, then in descending order by the result of the CUME_DIST() function, and all
tied CUME_DIST() values are ordered by the animal name.
select name, kind, cume_dist() over (partition by kind order by kilos) as ordering
from animals
where
kind = 'Bird'
order by kind, ordering desc, name;
+-------------------+------+----------+
| name | kind | ordering |
+-------------------+------+----------+
| Ostrich | Bird | 1 |
| Andean Condor | Bird | 0.8 |
| California Condor | Bird | 0.8 |
| Condor | Bird | 0.8 |
| Owl | Bird | 0.2 |
+-------------------+------+----------+
DENSE_RANK Function
Returns an ascending sequence of integers, starting with 1. The output sequence produces duplicate integers for
duplicate values of the ORDER BY expressions. After generating duplicate output values for the “tied” input values,
the function continues the sequence with the next higher integer. Therefore, the sequence contains duplicates but no
gaps when the input contains duplicates. Starts the sequence over for each group produced by the PARTITIONED BY
clause.
Syntax:
The PARTITION BY clause is optional. The ORDER BY clause is required. The window clause is not allowed.
Usage notes:
Often used for top-N and bottom-N queries. For example, it could produce a “top 10” report including all the items
with the 10 highest values, even if several items tied for 1st place.
Similar to ROW_NUMBER and RANK. These functions differ in how they treat duplicate combinations of values.
Added in: CDH 5.2.0 (Impala 2.0.0)
Examples:
The following example demonstrates how the DENSE_RANK() function identifies where each value “places” in the
result set, producing the same result for duplicate values, but with a strict sequence from 1 to the number of groups.
For example, when results are ordered by the X column, both 1 values are tied for first; both 2 values are tied for
second; and so on.
The following examples show how the DENSE_RANK() function is affected by the PARTITION property within the
ORDER BY clause.
Partitioning by the PROPERTY column groups all the even, odd, and so on values together, and DENSE_RANK() returns
the place of each value within the group, producing several ascending sequences.
Partitioning by the X column groups all the duplicate numbers together and returns the place each value within the
group; because each value occurs only 1 or 2 times, DENSE_RANK() designates each X value as either first or second
within its group.
The following example shows how DENSE_RANK() produces a continuous sequence while still allowing for ties. In this
case, Croesus and Midas both have the second largest fortune, while Crassus has the third largest. (In RANK Function
on page 443, you see a similar query with the RANK() function that shows that while Crassus has the third largest
fortune, he is the fourth richest person.)
select dense_rank() over (order by net_worth desc) as placement, name, net_worth from
wealth order by placement, name;
+-----------+---------+---------------+
| placement | name | net_worth |
+-----------+---------+---------------+
| 1 | Solomon | 2000000000.00 |
| 2 | Croesus | 1000000000.00 |
| 2 | Midas | 1000000000.00 |
| 3 | Crassus | 500000000.00 |
| 4 | Scrooge | 80000000.00 |
+-----------+---------+---------------+
Related information:
RANK Function on page 443, ROW_NUMBER Function on page 445
FIRST_VALUE Function
Returns the expression value from the first row in the window. The return value is NULL if the input expression is NULL.
Syntax:
The PARTITION BY clause is optional. The ORDER BY clause is required. The window clause is optional.
Usage notes:
If any duplicate values occur in the tuples evaluated by the ORDER BY clause, the result of this function is not
deterministic. Consider adding additional ORDER BY columns to ensure consistent ordering.
Changing the order in which the names are evaluated changes which greeting is applied to each group.
Related information:
LAST_VALUE Function on page 439
LAG Function
This function returns the value of an expression using column values from a preceding row. You specify an integer
offset, which designates a row position some number of rows previous to the current row. Any column references in
the expression argument refer to column values from that prior row. Typically, the table contains a time sequence or
numeric sequence column that clearly distinguishes the ordering of the rows.
Syntax:
The ORDER BY clause is required. The PARTITION BY clause is optional. The window clause is not allowed.
Usage notes:
Sometimes used an an alternative to doing a self-join.
Added in: CDH 5.2.0 (Impala 2.0.0)
Examples:
The following example uses the same stock data created in Window Clause on page 432. For each day, the query prints
the closing price alongside the previous day's closing price. The first row for each stock symbol has no previous row,
so that LAG() value is NULL.
The following example does an arithmetic operation between the current row and a value from the previous row, to
produce a delta value for each day. This example also demonstrates how ORDER BY works independently in the
different parts of the query. The ORDER BY closing_date in the OVER clause makes the query analyze the rows in
chronological order. Then the outer query block uses ORDER BY closing_date DESC to present the results with
the most recent date first.
Related information:
This function is the converse of LEAD Function on page 440.
LAST_VALUE Function
Returns the expression value from the last row in the window. This same value is repeated for all result rows for the
group. The return value is NULL if the input expression is NULL.
Syntax:
The PARTITION BY clause is optional. The ORDER BY clause is required. The window clause is optional.
Usage notes:
If any duplicate values occur in the tuples evaluated by the ORDER BY clause, the result of this function is not
deterministic. Consider adding additional ORDER BY columns to ensure consistent ordering.
Added in: CDH 5.2.0 (Impala 2.0.0)
Examples:
The following example uses the same MAIL_MERGE table as in the example for FIRST_VALUE Function on page 437.
Because the default window when ORDER BY is used is BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW, the
query requires the UNBOUNDED FOLLOWING to look ahead to subsequent rows and find the last value for each country.
Related information:
FIRST_VALUE Function on page 437
LEAD Function
This function returns the value of an expression using column values from a following row. You specify an integer
offset, which designates a row position some number of rows after to the current row. Any column references in the
expression argument refer to column values from that later row. Typically, the table contains a time sequence or
numeric sequence column that clearly distinguishes the ordering of the rows.
Syntax:
The ORDER BY clause is required. The PARTITION BY clause is optional. The window clause is not allowed.
Usage notes:
Sometimes used an an alternative to doing a self-join.
Added in: CDH 5.2.0 (Impala 2.0.0)
Examples:
The following example uses the same stock data created in Window Clause on page 432. The query analyzes the closing
price for a stock symbol, and for each day evaluates if the closing price for the following day is higher or lower.
end as "trending"
from stock_ticker
order by closing_date;
+--------------+---------------------+---------------+---------------+
| stock_symbol | closing_date | closing_price | trending |
+--------------+---------------------+---------------+---------------+
| JDR | 2014-09-13 00:00:00 | 12.86 | higher |
| JDR | 2014-09-14 00:00:00 | 12.89 | higher |
| JDR | 2014-09-15 00:00:00 | 12.94 | flat or lower |
| JDR | 2014-09-16 00:00:00 | 12.55 | higher |
| JDR | 2014-09-17 00:00:00 | 14.03 | higher |
| JDR | 2014-09-18 00:00:00 | 14.75 | flat or lower |
| JDR | 2014-09-19 00:00:00 | 13.98 | NULL |
+--------------+---------------------+---------------+---------------+
Related information:
This function is the converse of LAG Function on page 438.
The ORDER BY clause is required. The PARTITION BY clause is optional. The window clause is not allowed.
Usage notes:
The “ntile” name is derived from the practice of dividing result sets into fourths (quartile), tenths (decile), and so on.
The NTILE() function divides the result set based on an arbitrary percentile value.
The number of buckets must be a positive integer.
The number of items in each bucket is identical or almost so, varying by at most 1. If the number of items does not
divide evenly between the buckets, the remaining N items are divided evenly among the first N buckets.
If the number of buckets N is greater than the number of input rows in the partition, then the first N buckets each
contain one item, and the remaining buckets are empty.
Examples:
The following example shows divides groups of animals into 4 buckets based on their weight. The ORDER BY ...
DESC clause in the OVER() clause means that the heaviest 25% are in the first group, and the lightest 25% are in the
fourth group. (The ORDER BY in the outermost part of the query shows how you can order the final result set
independently from the order in which the rows are evaluated by the OVER() clause.) Because there are 9 rows in the
group, divided into 4 buckets, the first bucket receives the extra item.
The following examples show how the PARTITION clause works for the NTILE() function. Here, we divide each kind
of animal (mammal or bird) into 2 buckets, the heavier half and the lighter half.
select name, kind, ntile(2) over (partition by kind order by kilos desc) as half
from animals
order by kind;
+------------+--------+------+
| name | kind | half |
+------------+--------+------+
| Ostrich | Bird | 1 |
| Condor | Bird | 1 |
| Owl | Bird | 2 |
| Elephant | Mammal | 1 |
| Giraffe | Mammal | 1 |
| Polar bear | Mammal | 1 |
| Horse | Mammal | 2 |
| Housecat | Mammal | 2 |
| Mouse | Mammal | 2 |
+------------+--------+------+
Again, the result set can be ordered independently from the analytic evaluation. This next example lists all the animals
heaviest to lightest, showing that elephant and giraffe are in the “top half” of mammals by weight, while housecat and
mouse are in the “bottom half”.
select name, kind, ntile(2) over (partition by kind order by kilos desc) as half
from animals
order by kilos desc;
+------------+--------+------+
| name | kind | half |
+------------+--------+------+
| Elephant | Mammal | 1 |
| Giraffe | Mammal | 1 |
| Polar bear | Mammal | 1 |
| Horse | Mammal | 2 |
| Ostrich | Bird | 1 |
| Condor | Bird | 1 |
| Housecat | Mammal | 2 |
| Owl | Bird | 2 |
| Mouse | Mammal | 2 |
+------------+--------+------+
PERCENT_RANK (expr)
OVER ([partition_by_clause] order_by_clause)
Calculates the rank, expressed as a percentage, of each row within a group of rows. If rank is the value for that same
row from the RANK() function (from 1 to the total number of rows in the partition group), then the PERCENT_RANK()
value is calculated as (rank - 1) / (rows_in_group - 1) . If there is only a single item in the partition group,
its PERCENT_RANK() value is 0.
The ORDER BY clause is required. The PARTITION BY clause is optional. The window clause is not allowed.
Usage notes:
This function is similar to the RANK and CUME_DIST() functions: it returns an ascending sequence representing the
position of each row within the rows of the same partition group. The actual numeric sequence is calculated differently,
and the handling of duplicate (tied) values is different.
The return values range from 0 to 1 inclusive. The first row in each partition group always has the value 0. A NULL value
is considered the lowest possible value. In the case of duplicate input values, all the corresponding rows in the result
set have an identical value: the lowest PERCENT_RANK() value of those tied rows. (In contrast to CUME_DIST(),
where all tied rows have the highest CUME_DIST() value.)
Examples:
The following example uses the same ANIMALS table as the examples for CUME_DIST() and NTILE(), with a few
additional rows to illustrate the results where some values are NULL or there is only a single row in a partition group.
As with CUME_DIST(), there is an ascending sequence for each kind of animal. For example, the “Birds” and “Mammals”
rows each have a PERCENT_RANK() sequence that ranges from 0 to 1. The “Reptile” row has a PERCENT_RANK() of
0 because that partition group contains only a single item. Both “Mythical” animals have a PERCENT_RANK() of 0
because a NULL is considered the lowest value within its partition group.
select name, kind, percent_rank() over (partition by kind order by kilos) from animals;
+-----------------------+----------+--------------------------+
| name | kind | percent_rank() OVER(...) |
+-----------------------+----------+--------------------------+
| Mouse | Mammal | 0 |
| Housecat | Mammal | 0.2 |
| Horse | Mammal | 0.4 |
| Polar bear | Mammal | 0.6 |
| Giraffe | Mammal | 0.8 |
| Elephant | Mammal | 1 |
| Komodo dragon | Reptile | 0 |
| Owl | Bird | 0 |
| California Condor | Bird | 0.25 |
| Andean Condor | Bird | 0.25 |
| Condor | Bird | 0.25 |
| Ostrich | Bird | 1 |
| Fire-breathing dragon | Mythical | 0 |
| Unicorn | Mythical | 0 |
+-----------------------+----------+--------------------------+
RANK Function
Returns an ascending sequence of integers, starting with 1. The output sequence produces duplicate integers for
duplicate values of the ORDER BY expressions. After generating duplicate output values for the “tied” input values,
the function increments the sequence by the number of tied values. Therefore, the sequence contains both duplicates
and gaps when the input contains duplicates. Starts the sequence over for each group produced by the PARTITIONED
BY clause.
Syntax:
The PARTITION BY clause is optional. The ORDER BY clause is required. The window clause is not allowed.
Usage notes:
Often used for top-N and bottom-N queries. For example, it could produce a “top 10” report including several items
that were tied for 10th place.
Similar to ROW_NUMBER and DENSE_RANK. These functions differ in how they treat duplicate combinations of values.
Added in: CDH 5.2.0 (Impala 2.0.0)
Examples:
The following example demonstrates how the RANK() function identifies where each value “places” in the result set,
producing the same result for duplicate values, and skipping values in the sequence to account for the number of
duplicates. For example, when results are ordered by the X column, both 1 values are tied for first; both 2 values are
tied for third; and so on.
The following examples show how the RANK() function is affected by the PARTITION property within the ORDER BY
clause.
Partitioning by the PROPERTY column groups all the even, odd, and so on values together, and RANK() returns the
place of each value within the group, producing several ascending sequences.
| 5 | 3 | prime |
| 7 | 4 | prime |
| 10 | 1 | round |
| 1 | 1 | square |
| 4 | 2 | square |
| 9 | 3 | square |
+----+------+----------+
Partitioning by the X column groups all the duplicate numbers together and returns the place each value within the
group; because each value occurs only 1 or 2 times, RANK() designates each X value as either first or second within
its group.
The following example shows how a magazine might prepare a list of history's wealthiest people. Croesus and Midas
are tied for second, then Crassus is fourth.
select rank() over (order by net_worth desc) as rank, name, net_worth from wealth order
by rank, name;
+------+---------+---------------+
| rank | name | net_worth |
+------+---------+---------------+
| 1 | Solomon | 2000000000.00 |
| 2 | Croesus | 1000000000.00 |
| 2 | Midas | 1000000000.00 |
| 4 | Crassus | 500000000.00 |
| 5 | Scrooge | 80000000.00 |
+------+---------+---------------+
Related information:
DENSE_RANK Function on page 435, ROW_NUMBER Function on page 445
ROW_NUMBER Function
Returns an ascending sequence of integers, starting with 1. Starts the sequence over for each group produced by the
PARTITIONED BY clause. The output sequence includes different values for duplicate input values. Therefore, the
sequence never contains any duplicates or gaps, regardless of duplicate input values.
Syntax:
The ORDER BY clause is required. The PARTITION BY clause is optional. The window clause is not allowed.
Usage notes:
Often used for top-N and bottom-N queries where the input values are known to be unique, or precisely N rows are
needed regardless of duplicate values.
Because its result value is different for each row in the result set (when used without a PARTITION BY clause),
ROW_NUMBER() can be used to synthesize unique numeric ID values, for example for result sets involving unique values
or tuples.
Similar to RANK and DENSE_RANK. These functions differ in how they treat duplicate combinations of values.
Added in: CDH 5.2.0 (Impala 2.0.0)
Examples:
The following example demonstrates how ROW_NUMBER() produces a continuous numeric sequence, even though
some values of X are repeated.
The following example shows how a financial institution might assign customer IDs to some of history's wealthiest
figures. Although two of the people have identical net worth figures, unique IDs are required for this purpose.
ROW_NUMBER() produces a sequence of five different values for the five input rows.
Related information:
RANK Function on page 443, DENSE_RANK Function on page 435
UDF Concepts
Depending on your use case, you might write all-new functions, reuse Java UDFs that you have already written for
Hive, or port Hive Java UDF code to higher-performance native Impala UDFs in C++. You can code either scalar functions
for producing results one row at a time, or more complex aggregate functions for doing analysis across. The following
sections discuss these different aspects of working with UDFs.
UDFs and UDAFs
Depending on your use case, the user-defined functions (UDFs) you write might accept or produce different numbers
of input and output values:
• The most general kind of user-defined function (the one typically referred to by the abbreviation UDF) takes a
single input value and produces a single output value. When used in a query, it is called once for each row in the
result set. For example:
• A user-defined aggregate function (UDAF) accepts a group of values and returns a single value. You use UDAFs to
summarize and condense sets of rows, in the same style as the built-in COUNT, MAX(), SUM(), and AVG() functions.
When called in a query that uses the GROUP BY clause, the function is called once for each combination of GROUP
BY values. For example:
-- Evaluates batches of rows and returns a separate value for each batch.
select most_profitable_location(store_id, sales, expenses, tax_rate, depreciation) from
franchise_data group by year;
• Currently, Impala does not support other categories of user-defined functions, such as user-defined table functions
(UDTFs) or window functions.
'/user/hive/udfs/hive.jar' symbol='org.apache.hadoop.hive.ql.udf.UDFLower';
ERROR: AnalysisException: Function cannot have the same name as a builtin: lower
[localhost:21000] > create function my_lower(string) returns string location
'/user/hive/udfs/hive.jar' symbol='org.apache.hadoop.hive.ql.udf.UDFLower';
[localhost:21000] > select my_lower('Some String NOT ALREADY LOWERCASE');
+----------------------------------------------------+
| udfs.my_lower('some string not already lowercase') |
+----------------------------------------------------+
| some string not already lowercase |
+----------------------------------------------------+
Returned 1 row(s) in 0.11s
[localhost:21000] > create table t2 (s string);
[localhost:21000] > insert into t2 values ('lower'),('UPPER'),('Init cap'),('CamelCase');
Inserted 4 rows in 2.28s
[localhost:21000] > select * from t2;
+-----------+
| s |
+-----------+
| lower |
| UPPER |
| Init cap |
| CamelCase |
+-----------+
Returned 4 row(s) in 0.47s
[localhost:21000] > select my_lower(s) from t2;
+------------------+
| udfs.my_lower(s) |
+------------------+
| lower |
| upper |
| init cap |
| camelcase |
+------------------+
Returned 4 row(s) in 0.54s
[localhost:21000] > select my_lower(concat('ABC ',s,' XYZ')) from t2;
+------------------------------------------+
| udfs.my_lower(concat('abc ', s, ' xyz')) |
+------------------------------------------+
| abc lower xyz |
| abc upper xyz |
| abc init cap xyz |
| abc camelcase xyz |
+------------------------------------------+
Returned 4 row(s) in 0.22s
+----------------+
Returned 3 row(s) in 0.60s
[localhost:21000] > select my_neg(4.0);
ERROR: AnalysisException: No matching function with signature: udfs.my_neg(FLOAT).
[localhost:21000] > select my_neg(cast(4.0 as int));
+-------------------------------+
| udfs.my_neg(cast(4.0 as int)) |
+-------------------------------+
| -4 |
+-------------------------------+
Returned 1 row(s) in 0.11s
[localhost:21000] > create function my_neg(double) returns double location
'/user/hive/udfs/hive.jar' symbol='org.apache.hadoop.hive.ql.udf.UDFOPNegative';
[localhost:21000] > select my_neg(4.0);
+------------------+
| udfs.my_neg(4.0) |
+------------------+
| -4 |
+------------------+
Returned 1 row(s) in 0.11s
You can find the sample files mentioned here in the Impala github repo.
Note: The UDF development code does not rely on Impala being installed on the same machine. You
can write and compile UDFs on a minimal development system, then deploy them on a different one
for use with Impala. If you develop UDFs on a server managed by Cloudera Manager through the
parcel mechanism, you still install the UDF development kit through the package mechanism; this
small standalone package does not interfere with the parcels containing the main Impala code.
When you are ready to start writing your own UDFs, download the sample code and build scripts from the Cloudera
sample UDF github. Then see Writing User-Defined Functions (UDFs) on page 450 for how to code UDFs, and Examples
of Creating and Using UDFs on page 456 for how to build and run UDFs.
// This is the only Impala header required to develop UDFs and UDAs. This header
// contains the types that need to be used and the FunctionContext object. The context
// object serves as the interface object between the UDF/UDA and the impala process.
For the basic declarations needed to write a scalar UDF, see the header file udf-sample.h within the sample build
environment, which defines a simple function named AddUdf():
#ifndef IMPALA_UDF_SAMPLE_UDF_H
#define IMPALA_UDF_SAMPLE_UDF_H
#include <impala_udf/udf.h>
#endif
For sample C++ code for a simple function named AddUdf(), see the source file udf-sample.cc within the sample
build environment:
#include "udf-sample.h"
// In this sample we are declaring a UDF that adds two ints and returns an int.
IntVal AddUdf(FunctionContext* context, const IntVal& arg1, const IntVal& arg2) {
if (arg1.is_null || arg2.is_null) return IntVal::null();
return IntVal(arg1.val + arg2.val);
}
Each data type has a corresponding structure defined in the C++ and Java header files, with two member fields and
some predefined comparison operators and constructors:
• is_null indicates whether the value is NULL or not. val holds the actual argument or return value when it is
non-NULL.
• Each struct also defines a null() member function that constructs an instance of the struct with the is_null
flag set.
• The built-in SQL comparison operators and clauses such as <, >=, BETWEEN, and ORDER BY all work automatically
based on the SQL return type of each UDF. For example, Impala knows how to evaluate BETWEEN 1 AND
udf_returning_int(col1) or ORDER BY udf_returning_string(col2) without you declaring any
comparison operators within the UDF itself.
For convenience within your UDF code, each struct defines == and != operators for comparisons with other structs
of the same type. These are for typical C++ comparisons within your own code, not necessarily reproducing SQL
semantics. For example, if the is_null flag is set in both structs, they compare as equal. That behavior of null
comparisons is different from SQL (where NULL == NULL is NULL rather than true), but more in line with typical
C++ behavior.
• Each kind of struct has one or more constructors that define a filled-in instance of the struct, optionally with
default values.
• Impala cannot process UDFs that accept the composite or nested types as arguments or return them as result
values. This limitation applies both to Impala UDFs written in C++ and Java-based Hive UDFs.
• You can overload functions by creating multiple functions with the same SQL name but different argument types.
For overloaded functions, you must use different C++ or Java entry point names in the underlying functions.
The data types defined on the C++ side (in /usr/include/impala_udf/udf.h) are:
• IntVal represents an INT column.
• BigIntVal represents a BIGINT column. Even if you do not need the full range of a BIGINT value, it can be
useful to code your function arguments as BigIntVal to make it convenient to call the function with different
kinds of integer columns and expressions as arguments. Impala automatically casts smaller integer types to larger
ones when appropriate, but does not implicitly cast large integer types to smaller ones.
• SmallIntVal represents a SMALLINT column.
• TinyIntVal represents a TINYINT column.
• StringVal represents a STRING column. It has a len field representing the length of the string, and a ptr field
pointing to the string data. It has constructors that create a new StringVal struct based on a null-terminated
C-style string, or a pointer plus a length; these new structs still refer to the original string data rather than allocating
a new buffer for the data. It also has a constructor that takes a pointer to a FunctionContext struct and a length,
that does allocate space for a new copy of the string data, for use in UDFs that return string values.
• BooleanVal represents a BOOLEAN column.
• FloatVal represents a FLOAT column.
• DoubleVal represents a DOUBLE column.
• TimestampVal represents a TIMESTAMP column. It has a date field, a 32-bit integer representing the Gregorian
date, that is, the days past the epoch date. It also has a time_of_day field, a 64-bit integer representing the
current time of day in nanoseconds.
In the CREATE FUNCTION statement, after the type of the first optional argument, include ... to indicate it could be
followed by more arguments of the same type. For example, the following function accepts a STRING argument,
followed by one or more additional STRING arguments:
[localhost:21000] > create function my_concat(string, string ...) returns string location
'/user/test_user/udfs/sample.so' symbol='Concat';
The call from the SQL query must pass at least one argument to the variable-length portion of the argument list.
When Impala calls the function, it fills in the initial set of required arguments, then passes the number of extra arguments
and a pointer to the first of those optional arguments.
void function_name(impala_udf::FunctionContext*,
impala_udf::FunctionContext::FunctionScope)
Currently, only THREAD_SCOPE is implemented, not FRAGMENT_SCOPE. See udf.h for details about the scope values.
Error Handling for UDFs
To handle errors in UDFs, you call functions that are members of the initial FunctionContext* argument passed to
your function.
A UDF can record one or more warnings, for conditions that indicate minor, recoverable problems that do not cause
the query to stop. The signature for this function is:
For a serious problem that requires cancelling the query, a UDF can set an error flag that prevents the query from
returning any results. The signature for this function is:
For convenience, you can use a naming convention for the underlying functions and Impala automatically recognizes
those entry points. Specify the UPDATE_FN clause, using an entry point name containing the string update or Update.
When you omit the other _FN clauses from the SQL statement, Impala looks for entry points with names formed by
substituting the update or Update portion of the specified name.
uda-sample.h:
and generates a Makefile customized for your particular directory paths. Then the make command runs the actual
build steps based on the rules in the Makefile.
Impala loads the shared library from an HDFS location. After building a shared library containing one or more UDFs,
use hdfs dfs or hadoop fs commands to copy the binary file to an HDFS location readable by Impala.
The final step in deployment is to issue a CREATE FUNCTION statement in the impala-shell interpreter to make
Impala aware of the new function. See CREATE FUNCTION Statement on page 232 for syntax details. Because each
function is associated with a particular database, always issue a USE statement to the appropriate database before
creating a function, or specify a fully qualified name, that is, CREATE FUNCTION db_name.function_name.
As you update the UDF code and redeploy updated versions of a shared library, use DROP FUNCTION and CREATE
FUNCTION to let Impala pick up the latest version of the code.
Note:
Currently, Impala UDFs and UDAs are not persisted in the metastore database. Information about
these functions is held in the memory of the catalogd daemon. You must reload them by running
the CREATE FUNCTION statements again each time you restart the catalogd daemon.
# Use the appropriate package installation command for your Linux distribution.
sudo yum install gcc-c++ cmake boost-devel
sudo yum install impala-udf-devel
# The package name on Ubuntu and Debian is impala-udf-dev.
Then, unpack the sample code in udf_samples.tar.gz and use that as a template to set up your build environment.
To build the original samples:
The sample code to examine, experiment with, and adapt is in these files:
• udf-sample.h: Header file that declares the signature for a scalar UDF (AddUDF).
• udf-sample.cc: Sample source for a simple UDF that adds two integers. Because Impala can reference multiple
function entry points from the same shared library, you could add other UDF functions in this file and add their
signatures to the corresponding header file.
• udf-sample-test.cc: Basic unit tests for the sample UDF.
• uda-sample.h: Header file that declares the signature for sample aggregate functions. The SQL functions will
be called COUNT, AVG, and STRINGCONCAT. Because aggregate functions require more elaborate coding to handle
the processing for multiple phases, there are several underlying C++ functions such as CountInit, AvgUpdate,
and StringConcatFinalize.
• uda-sample.cc: Sample source for simple UDAFs that demonstrate how to manage the state transitions as the
underlying functions are called during the different phases of query processing.
– The UDAF that imitates the COUNT function keeps track of a single incrementing number; the merge functions
combine the intermediate count values from each Impala node, and the combined number is returned
verbatim by the finalize function.
– The UDAF that imitates the AVG function keeps track of two numbers, a count of rows processed and the
sum of values for a column. These numbers are updated and merged as with COUNT, then the finalize function
divides them to produce and return the final average value.
– The UDAF that concatenates string values into a comma-separated list demonstrates how to manage storage
for a string that increases in length as the function is called for multiple rows.
int index;
uint8_t *ptr;
int count;
int index;
uint8_t *ptr;
for (ptr = arg1.ptr, count = 0, index = 0; index <= arg1.len; index++, ptr++)
{
uint8_t c = tolower(*ptr);
if (c == 'a' || c == 'e' || c == 'i' || c == 'o' || c == 'u')
{
count++;
}
}
return IntVal(count);
}
int index;
std::string original((const char *)arg1.ptr,arg1.len);
std::string shorter("");
We build a shared library, libudfsample.so, and put the library file into HDFS where Impala can read it:
$ make
[ 0%] Generating udf_samples/uda-sample.ll
[ 16%] Built target uda-sample-ir
[ 33%] Built target udasample
[ 50%] Built target uda-sample-test
[ 50%] Generating udf_samples/udf-sample.ll
[ 66%] Built target udf-sample-ir
Scanning dependencies of target udfsample
[ 83%] Building CXX object CMakeFiles/udfsample.dir/udf-sample.o
Linking CXX shared library udf_samples/libudfsample.so
[ 83%] Built target udfsample
Linking CXX executable udf_samples/udf-sample-test
[100%] Built target udf-sample-test
$ hdfs dfs -put ./udf_samples/libudfsample.so /user/hive/udfs/libudfsample.so
Finally, we go into the impala-shell interpreter where we set up some sample data, issue CREATE FUNCTION
statements to set up the SQL function names, and call the functions in some queries:
+-----------------------+
| NULL |
+-----------------------+
Returned 1 row(s) in 0.11s
[localhost:21000] > select s, has_vowels(s) from t2;
+-----------+--------------------+
| s | udfs.has_vowels(s) |
+-----------+--------------------+
| lower | true |
| UPPER | true |
| Init cap | true |
| CamelCase | true |
+-----------+--------------------+
Returned 4 row(s) in 0.24s
+-----------+----------------------+
Returned 4 row(s) in 0.24s
We add the function bodies to a C++ source file (in this case, uda-sample.cc):
As with the sample UDF, we build a shared library and put it into HDFS:
$ make
[ 0%] Generating udf_samples/uda-sample.ll
[ 16%] Built target uda-sample-ir
Scanning dependencies of target udasample
[ 33%] Building CXX object CMakeFiles/udasample.dir/uda-sample.o
Linking CXX shared library udf_samples/libudasample.so
[ 33%] Built target udasample
Scanning dependencies of target uda-sample-test
[ 50%] Building CXX object CMakeFiles/uda-sample-test.dir/uda-sample-test.o
Linking CXX executable udf_samples/uda-sample-test
[ 50%] Built target uda-sample-test
[ 50%] Generating udf_samples/udf-sample.ll
[ 66%] Built target udf-sample-ir
[ 83%] Built target udfsample
[100%] Built target udf-sample-test
$ hdfs dfs -put ./udf_samples/libudasample.so /user/hive/udfs/libudasample.so
To create the SQL function, we issue a CREATE AGGREGATE FUNCTION statement and specify the underlying C++
function names for the different phases:
[localhost:21000] > -- Compute the same value using literals or the UDA;
[localhost:21000] > select 1*1 + 2*2 + 3*3 + 4*4;
+-------------------------------+
| 1 * 1 + 2 * 2 + 3 * 3 + 4 * 4 |
+-------------------------------+
| 30 |
+-------------------------------+
Returned 1 row(s) in 0.12s
[localhost:21000] > select sum_of_squares(x) from sos;
+------------------------+
| udfs.sum_of_squares(x) |
+------------------------+
| 30 |
+------------------------+
Returned 1 row(s) in 0.35s
Until we create the overloaded version of the UDA, it can only handle a single data type. To allow it to handle DOUBLE
as well as BIGINT, we issue another CREATE AGGREGATE FUNCTION statement:
[localhost:21000] > -- Compute the same value using literals or the UDA;
[localhost:21000] > select 1.1*1.1 + 2.2*2.2 + 3.3*3.3 + 4.4*4.4;
+-----------------------------------------------+
| 1.1 * 1.1 + 2.2 * 2.2 + 3.3 * 3.3 + 4.4 * 4.4 |
+-----------------------------------------------+
| 36.3 |
+-----------------------------------------------+
Returned 1 row(s) in 0.12s
[localhost:21000] > select sum_of_squares(y) from sos;
+------------------------+
| udfs.sum_of_squares(y) |
+------------------------+
| 36.3 |
+------------------------+
Returned 1 row(s) in 0.35s
Typically, you use a UDA in queries with GROUP BY clauses, to produce a result set with a separate aggregate value
for each combination of values from the GROUP BY clause. Let's change our sample table to use 0 to indicate rows
containing even values, and 1 to flag rows containing odd values. Then the GROUP BY query can return two values,
the sum of the squares for the even values, and the sum of the squares for the odd values:
[localhost:21000] > insert overwrite sos values (1, 1), (2, 0), (3, 1), (4, 0);
Inserted 4 rows in 1.24s
Because Impala and Hive share the same metastore database and their tables are often used interchangeably, the
following section covers differences between Impala and Hive in detail.
Note:
By default, Impala only allows a single COUNT(DISTINCT columns) expression in each query.
If you do not need precise accuracy, you can produce an estimate of the distinct values for a
column by specifying NDV(column); a query can contain multiple instances of NDV(column).
To make Impala automatically rewrite COUNT(DISTINCT) expressions to NDV(), enable the
APPX_COUNT_DISTINCT query option.
To produce the same result as multiple COUNT(DISTINCT) expressions, you can use the following
technique for queries involving a single table:
Because CROSS JOIN is an expensive operation, prefer to use the NDV() technique wherever
practical.
User-defined functions (UDFs) are supported starting in Impala 1.2. See Impala User-Defined Functions (UDFs) on page
447 for full details on Impala UDFs.
• Impala supports high-performance UDFs written in C++, as well as reusing some Java-based Hive UDFs.
• Impala supports scalar UDFs and user-defined aggregate functions (UDAFs). Impala does not currently support
user-defined table generating functions (UDTFs).
• Only Impala-supported column types are supported in Java-based UDFs.
Impala does not currently support these HiveQL statements:
• ANALYZE TABLE (the Impala equivalent is COMPUTE STATS)
• DESCRIBE COLUMN
• DESCRIBE DATABASE
• EXPORT TABLE
• IMPORT TABLE
• SHOW TABLE EXTENDED
• SHOW TBLPROPERTIES
• SHOW INDEXES
• SHOW COLUMNS
• INSERT OVERWRITE DIRECTORY; use INSERT OVERWRITE table_name or CREATE TABLE AS SELECT to
materialize query results into the HDFS directory associated with an Impala table.
Impala respects the serialization.null.format table property only for TEXT tables and ignores the property for
Parquet and other formats. Hive respects the serialization.null.format property for Parquet and other formats
and converts matching values to NULL during the scan. See for using the table property in Impala.
• The Impala TIMESTAMP type can represent dates ranging from 1400-01-01 to 9999-12-31. This is different from
the Hive date range, which is 0000-01-01 to 9999-12-31.
• Impala does not return column overflows as NULL, so that customers can distinguish between NULL data and
overflow conditions similar to how they do so with traditional database systems. Impala returns the largest or
smallest value in the range for the type. For example, valid values for a tinyint range from -128 to 127. In Impala,
a tinyint with a value of -200 returns -128 rather than NULL. A tinyint with a value of 200 returns 127.
Miscellaneous features:
• Impala does not provide virtual columns.
• Impala does not expose locking.
• Impala does not expose some configuration properties.
• Although Impala supports INTERVAL expressions for datetime arithmetic, as shown in TIMESTAMP Data Type on
page 148, INTERVAL is not available as a column data type in Impala. For any INTERVAL values stored in tables,
convert them to numeric values that you can add or subtract using the functions in Impala Date and Time Functions
on page 363. For example, if you had a table DEADLINES with an INT column TIME_PERIOD, you could construct
dates N days in the future like so:
• For YEAR columns, change to the smallest Impala integer type that has sufficient range. See Data Types on page
116 for details about ranges, casting, and so on for the various numeric data types.
• Change any DECIMAL and NUMBER types. If fixed-point precision is not required, you can use FLOAT or DOUBLE
on the Impala side depending on the range of values. For applications that require precise decimal values, such
as financial data, you might need to make more extensive changes to table structure and application logic, such
as using separate integer columns for dollars and cents, or encoding numbers as string values and writing UDFs
to manipulate them. See Data Types on page 116 for details about ranges, casting, and so on for the various numeric
data types.
• FLOAT, DOUBLE, and REAL types are supported in Impala. Remove any precision and scale specifications. (In
Impala, REAL is just an alias for DOUBLE; columns declared as REAL are turned into DOUBLE behind the scenes.)
See Data Types on page 116 for details about ranges, casting, and so on for the various numeric data types.
• Most integer types from other systems have equivalents in Impala, perhaps under different names such as BIGINT
instead of INT8. For any that are unavailable, for example MEDIUMINT, switch to the smallest Impala integer type
that has sufficient range. Remove any precision specifications. See Data Types on page 116 for details about ranges,
casting, and so on for the various numeric data types.
• Remove any UNSIGNED constraints. All Impala numeric types are signed. See Data Types on page 116 for details
about ranges, casting, and so on for the various numeric data types.
• For any types holding bitwise values, use an integer type with enough range to hold all the relevant bits within a
positive integer. See Data Types on page 116 for details about ranges, casting, and so on for the various numeric
data types.
For example, TINYINT has a maximum positive value of 127, not 256, so to manipulate 8-bit bitfields as positive
numbers switch to the next largest type SMALLINT.
Impala does not support notation such as b'0101' for bit literals.
• For BLOB values, use STRING to represent CLOB or TEXT types (character based large objects) up to 32 KB in size.
Binary large objects such as BLOB, RAW BINARY, and VARBINARY do not currently have an equivalent in Impala.
• For Boolean-like types such as BOOL, use the Impala BOOLEAN type.
• Because Impala currently does not support composite or nested types, any spatial data types in other database
systems do not have direct equivalents in Impala. You could represent spatial values in string format and write
UDFs to process them. See Impala User-Defined Functions (UDFs) on page 447 for details. Where practical, separate
spatial types into separate tables so that Impala can still work with the non-spatial data.
• Take out any DEFAULT clauses. Impala can use data files produced from many different sources, such as Pig, Hive,
or MapReduce jobs. The fast import mechanisms of LOAD DATA and external tables mean that Impala is flexible
about the format of data files, and Impala does not necessarily validate or cleanse data before querying it. When
copying data through Impala INSERT statements, you can use conditional functions such as CASE or NVL to
substitute some other value for NULL fields; see Impala Conditional Functions on page 391 for details.
• Take out any constraints from your CREATE TABLE and ALTER TABLE statements, for example PRIMARY KEY,
FOREIGN KEY, UNIQUE, NOT NULL, UNSIGNED, or CHECK constraints. Impala can use data files produced from
many different sources, such as Pig, Hive, or MapReduce jobs. Therefore, Impala expects initial data validation to
happen earlier during the ETL or ELT cycle. After data is loaded into Impala tables, you can perform queries to test
for NULL values. When copying data through Impala INSERT statements, you can use conditional functions such
as CASE or NVL to substitute some other value for NULL fields; see Impala Conditional Functions on page 391 for
details.
Do as much verification as practical before loading data into Impala. After data is loaded into Impala, you can do
further verification using SQL queries to check if values have expected ranges, if values are NULL or not, and so
on. If there is a problem with the data, you will need to re-run earlier stages of the ETL process, or do an INSERT
... SELECT statement in Impala to copy the faulty data to a new table and transform or filter out the bad values.
• Take out any CREATE INDEX, DROP INDEX, and ALTER INDEX statements, and equivalent ALTER TABLE
statements. Remove any INDEX, KEY, or PRIMARY KEY clauses from CREATE TABLE and ALTER TABLE statements.
Impala is optimized for bulk read operations for data warehouse-style queries, and therefore does not support
indexes for its tables.
• Calls to built-in functions with out-of-range or otherwise incorrect arguments, return NULL in Impala as opposed
to raising exceptions. (This rule applies even when the ABORT_ON_ERROR=true query option is in effect.) Run
small-scale queries using representative data to doublecheck that calls to built-in functions are returning expected
values rather than NULL. For example, unsupported CAST operations do not raise an error in Impala:
• For any other type not supported in Impala, you could represent their values in string format and write UDFs to
process them. See Impala User-Defined Functions (UDFs) on page 447 for details.
• To detect the presence of unsupported or unconvertable data types in data files, do initial testing with the
ABORT_ON_ERROR=true query option in effect. This option causes queries to fail immediately if they encounter
disallowed type conversions. See ABORT_ON_ERROR Query Option on page 306 for details. For example:
set abort_on_error=true;
select count(*) from (select * from t1);
-- The above query will fail if the data files for T1 contain any
-- values that can't be converted to the expected Impala data types.
-- For example, if T1.C1 is defined as INT but the column contains
-- floating-point values like 1.1, the query will return an error.
• If your database, table, column, or other names conflict with Impala reserved words, use different names or quote
the names with backticks. See Impala Reserved Words on page 585 for the current list of Impala reserved words.
Conversely, if you use a keyword that Impala does not recognize, it might be interpreted as a table or column
alias. For example, in SELECT * FROM t1 NATURAL JOIN t2, Impala does not recognize the NATURAL keyword
and interprets it as an alias for the table t1. If you experience any unexpected behavior with queries, check the
list of reserved words to make sure all keywords in join and WHERE clauses are recognized.
• Impala supports subqueries only in the FROM clause of a query, not within the WHERE clauses. Therefore, you
cannot use clauses such as WHERE column IN (subquery). Also, Impala does not allow EXISTS or NOT EXISTS
clauses (although EXISTS is a reserved keyword).
• Impala supports UNION and UNION ALL set operators, but not INTERSECT. Prefer UNION ALL over UNION when
you know the data sets are disjoint or duplicate values are not a problem; UNION ALL is more efficient because
it avoids materializing and sorting the entire result set to eliminate duplicate values.
• Within queries, Impala requires query aliases for any subqueries:
-- Without the alias 'contents_of_t1' at the end, query gives syntax error.
select count(*) from (select * from t1) contents_of_t1;
• When an alias is declared for an expression in a query, that alias cannot be referenced again within the same
query block:
-- Can't reference AVERAGE twice in the SELECT list where it's defined.
select avg(x) as average, average+1 from t1 group by x;
ERROR: AnalysisException: couldn't resolve column reference: 'average'
For Impala, either repeat the expression again, or abstract the expression into a WITH clause, creating named
columns that can be referenced multiple times anywhere in the base query:
• Impala does not support certain rarely used join types that are less appropriate for high-volume tables used for
data warehousing. In some cases, Impala supports join types but requires explicit syntax to ensure you do not do
inefficient joins of huge tables by accident. For example, Impala does not support natural joins or anti-joins, and
requires the CROSS JOIN operator for Cartesian products. See Joins in Impala SELECT Statements on page 282
for details on the syntax for Impala join clauses.
• Impala has a limited choice of partitioning types. Partitions are defined based on each distinct combination of
values for one or more partition key columns. Impala does not redistribute or check data to create evenly distributed
partitions; you must choose partition key columns based on your knowledge of the data volume and distribution.
Adapt any tables that use range, list, hash, or key partitioning to use the Impala partition syntax for CREATE
TABLE and ALTER TABLE statements. Impala partitioning is similar to range partitioning where every range has
exactly one value, or key partitioning where the hash function produces a separate bucket for every combination
of key values. See Partitioning for Impala Tables on page 522 for usage details, and CREATE TABLE Statement on
page 235 and ALTER TABLE Statement on page 215 for syntax.
Note: Because the number of separate partitions is potentially higher than in other database
systems, keep a close eye on the number of partitions and the volume of data in each one; scale
back the number of partition key columns if you end up with too many partitions with a small
volume of data in each one. Remember, to distribute work for a query across a cluster, you need
at least one HDFS block per node. HDFS blocks are typically multiple megabytes, especially for
Parquet files. Therefore, if each partition holds only a few megabytes of data, you are unlikely to
see much parallelism in the query because such a small amount of data is typically processed by
a single node.
• For “top-N” queries, Impala uses the LIMIT clause rather than comparing against a pseudocolumn named ROWNUM
or ROW_NUM. See LIMIT Clause on page 293 for details.
INSERT ... SELECT statement to copy the data into a new table and reorganize into a more efficient layout in
the same operation. See INSERT Statement on page 263 for details about the INSERT statement.
You can do INSERT ... SELECT into a table with a more efficient file format (see How Impala Works with
Hadoop File Formats on page 527) or from an unpartitioned table into a partitioned one (see Partitioning for Impala
Tables on page 522).
• The number of expressions allowed in an Impala query might be smaller than for some other database systems,
causing failures for very complicated queries (typically produced by automated SQL generators). Where practical,
keep the number of expressions in the WHERE clauses to approximately 2000 or fewer. As a workaround, set the
query option DISABLE_CODEGEN=true if queries fail for this reason. See DISABLE_CODEGEN Query Option on
page 308 for details.
• If practical, rewrite UNION queries to use the UNION ALL operator instead. Prefer UNION ALL over UNION when
you know the data sets are disjoint or duplicate values are not a problem; UNION ALL is more efficient because
it avoids materializing and sorting the entire result set to eliminate duplicate values.
For a list of the impala-shell command-line options, see impala-shell Configuration Options on page 470. For reference
information about the impala-shell interactive commands, see impala-shell Command Reference on page 476.
Note:
These options are different than the configuration options for the impalad daemon itself. For the
impalad options, see Modifying Impala Startup Options on page 43.
--print_header print_header=true
-o filename or --output_file output_file=filename Stores all query results in the specified file. Typically used
filename to store the results of a single query issued from the
command line with the -q option. Also works for
interactive sessions; you see the messages such as number
of rows fetched, but not the actual result set. To suppress
these incidental messages when combining the -q and -o
options, redirect stderr to /dev/null. Added in Impala
1.0.1.
-p or --show_profiles show_profiles=true Displays the query execution plan (same output as the
EXPLAIN statement) and a more detailed low-level
breakdown of execution steps, for every query executed
by the shell.
-f query_file or query_file=path_to_query_file Passes a SQL query from a file. Multiple statements must
--query_file=query_file be semicolon (;) delimited. In CDH 5.5 / Impala 2.3 and
higher, you can specify a filename of - to represent
standard input. This feature makes it convenient to use
impala-shell as part of a Unix pipeline where SQL
statements are generated dynamically by other tools.
[impala]
verbose=true
default_db=tpc_benchmarking
write_delimited=true
output_delimiter=,
output_file=/home/tester1/benchmark_results.csv
show_profiles=true
The following example shows a configuration file that connects to a specific remote Impala node, runs a single query
within a particular database, then exits. You would typically use this kind of single-purpose configuration setting with
the impala-shell command-line option --config_file=path_to_config_file, to easily select between many
predefined queries that could be run against different databases, hosts, or even different clusters. To run a sequence
of statements instead of a single query, specify the configuration option query_file=path_to_query_file instead.
[impala]
impalad=impala-test-node1.example.com
default_db=site_stats
# Issue a predefined query and immediately exit.
query=select count(*) from web_traffic where event_date = trunc(now(),'dd')
# When you are logged into the same machine running impalad.
# The prompt will reflect the current hostname.
$ impala-shell
# When you are logged into the same machine running impalad.
# The host will reflect the hostname 'localhost'.
$ impala-shell -i localhost
# When you are logged onto a different host, perhaps a client machine
# outside the Hadoop cluster.
$ impala-shell -i some.other.hostname
# When you are logged onto a different host, and impalad is listening
# on a non-default port. Perhaps a load balancer is forwarding requests
# to a different host/port combination behind the scenes.
$ impala-shell -i some.other.hostname:port_number
$ impala-shell
Welcome to the Impala shell. Press TAB twice to see a list of available commands.
(Shell
build version: Impala Shell v2.4.x (hash) built on
date)
[Not connected] >
2. Locate the hostname of a DataNode within the cluster that is running an instance of the impalad daemon. If that
DataNode uses a non-default port (something other than port 21000) for impala-shell connections, find out
the port number also.
3. Use the connect command to connect to an Impala instance. Enter a command of the form:
Note: Replace impalad-host with the hostname you have configured for any DataNode running
Impala in your environment. The changed prompt indicates a successful connection.
# A utility command that you might run while developing shell scripts
# to manipulate HDFS files.
$ impala-shell -i localhost -d database_of_interest -q 'show tables'
[impalad-host:21000] >
connect describe explain help history insert quit refresh select
set shell show use version
[impalad-host:21000] >
Note: Commands must be terminated by a semi-colon. A command can span multiple lines.
For example:
A comment is considered part of the statement it precedes, so when you enter a -- or /* */ comment, you get a
continuation prompt until you finish entering a statement ending with a semicolon:
Use the up-arrow and down-arrow keys to cycle through and edit previous commands. impala-shell uses the
readline library and so supports a standard set of keyboard shortcuts for editing and cursor movement, such as
Ctrl-A for beginning of line and Ctrl-E for end of line.
Command Explanation
alter Changes the underlying structure or settings of an Impala table, or a table shared between
Impala and Hive. See ALTER TABLE Statement on page 215 and ALTER VIEW Statement on page
224 for details.
compute stats Gathers important performance-related information for a table, used by Impala to optimize
queries. See COMPUTE STATS Statement on page 226 for details.
connect Connects to the specified instance of impalad. The default port of 21000 is assumed unless
you provide another value. You can connect to any host in your cluster that is running impalad.
If you connect to an instance of impalad that was started with an alternate port specified
by the --fe_port flag, you must provide that alternate port. See Connecting to impalad
through impala-shell on page 474 for examples.
The SET statement has no effect until the impala-shell interpreter is connected to an
Impala server. Once you are connected, any query options you set remain in effect as you
issue a subsequent CONNECT command to connect to a different Impala host.
Command Explanation
describe Shows the columns, column data types, and any column comments for a specified table.
DESCRIBE FORMATTED shows additional information such as the HDFS data directory,
partitions, and internal properties for the table. See DESCRIBE Statement on page 245 for
details about the basic DESCRIBE output and the DESCRIBE FORMATTED variant. You can
use DESC as shorthand for the DESCRIBE command.
drop Removes a schema object, and in some cases its associated data files. See DROP TABLE
Statement on page 258, DROP VIEW Statement on page 259, DROP DATABASE Statement on
page 252, and DROP FUNCTION Statement on page 254 for details.
explain Provides the execution plan for a query. EXPLAIN represents a query as a series of steps. For
example, these steps might be map/reduce stages, metastore operations, or file system
operations such as move or rename. See EXPLAIN Statement on page 260 and Using the EXPLAIN
Plan for Performance Tuning on page 507 for details.
history Maintains an enumerated cross-session command history. This history is stored in the
~/.impalahistory file.
insert Writes the results of a query to a specified table. This either overwrites table data or appends
data to the existing table content. See INSERT Statement on page 263 for details.
invalidate Updates impalad metadata for table existence and structure. Use this command after creating,
metadata dropping, or altering databases, tables, or partitions in Hive. See INVALIDATE METADATA
Statement on page 271 for details.
profile Displays low-level information about the most recent query. Used for performance diagnosis
and tuning. The report starts with the same information as produced by the EXPLAIN
statement and the SUMMARY command. See Using the Query Profile for Performance Tuning
on page 509 for details.
quit Exits the shell. Remember to include the final semicolon so that the shell recognizes the end
of the command.
refresh Refreshes impalad metadata for the locations of HDFS blocks corresponding to Impala data
files. Use this command after loading new data files into an Impala table through Hive or
through HDFS commands. See REFRESH Statement on page 277 for details.
select Specifies the data set on which to complete some action. All information returned from
select can be sent to some output such as the console or a file or can be used to complete
some other element of query. See SELECT Statement on page 280 for details.
set Manages query options for an impala-shell session. The available options are the ones
listed in Query Options for the SET Statement on page 305. These options are used for query
tuning and troubleshooting. Issue SET with no arguments to see the current query options,
either based on the impalad defaults, as specified by you at impalad startup, or based on
earlier SET statements in the same session. To modify option values, issue commands with
the syntax set option=value. To restore an option to its default, use the unset command.
Some options take Boolean values of true and false. Others take numeric arguments, or
quoted string values.
The SET statement has no effect until the impala-shell interpreter is connected to an
Impala server. Once you are connected, any query options you set remain in effect as you
issue a subsequent CONNECT command to connect to a different Impala host.
Command Explanation
In Impala 2.0 and later, SET is available as a SQL statement for any kind of application, not
only through impala-shell. See SET Statement on page 304 for details.
shell Executes the specified command in the operating system shell without exiting impala-shell.
You can use the ! character as shorthand for the shell command.
show Displays metastore data for schema objects created and accessed through Impala, Hive, or
both. show can be used to gather information about objects such as databases, tables, and
functions. See SHOW Statement on page 322 for details.
summary Summarizes the work performed in various stages of a query. It provides a higher-level view
of the information displayed by the EXPLAIN command. Added in Impala 1.4.0. See Using
the SUMMARY Report for Performance Tuning on page 508 for details about the report format
and how to interpret it.
The time, memory usage, and so on reported by SUMMARY only include the portions of the
statement that read data, not when data is written. Therefore, the PROFILE command is
better for checking the performance and scalability of INSERT statements.
In CDH 5.5 / Impala 2.3 and higher, you can see a continuously updated report of the summary
information while a query is in progress. See LIVE_SUMMARY Query Option (CDH 5.5 or higher
only) on page 315 for details.
unset Removes any user-specified value for a query option and returns the option to its default
value. See Query Options for the SET Statement on page 305 for the available query options.
use Indicates the database against which to execute subsequent commands. Lets you avoid using
fully qualified names when referring to tables in databases other than default. See USE
Statement on page 337 for details. Not effective with the -q option, because that option only
allows a single statement in the argument.
Note:
Before starting any performance tuning or benchmarking, make sure your system is configured with
all the recommended minimum hardware requirements from Hardware Requirements on page 24
and software settings from Post-Installation Configuration for Impala on page 30.
• Partitioning for Impala Tables on page 522. This technique physically divides the data based on the different values
in frequently queried columns, allowing queries to skip reading a large percentage of the data in a table.
• Performance Considerations for Join Queries on page 482. Joins are the main class of queries that you can tune at
the SQL level, as opposed to changing physical factors such as the file format or the hardware configuration. The
related topics Overview of Column Statistics on page 490 and Overview of Table Statistics on page 489 are also
important primarily for join performance.
• Overview of Table Statistics on page 489 and Overview of Column Statistics on page 490. Gathering table and column
statistics, using the COMPUTE STATS statement, helps Impala automatically optimize the performance for join
queries, without requiring changes to SQL query statements. (This process is greatly simplified in Impala 1.2.2 and
higher, because the COMPUTE STATS statement gathers both kinds of statistics in one operation, and does not
require any setup and configuration as was previously necessary for the ANALYZE TABLE statement in Hive.)
• Testing Impala Performance on page 506. Do some post-setup testing to ensure Impala is using optimal settings
for performance, before conducting any benchmark tests.
• Benchmarking Impala Queries on page 500. The configuration and sample data that you use for initial experiments
with Impala is often not appropriate for doing performance tests.
• Controlling Impala Resource Usage on page 500. The more memory Impala can utilize, the better query performance
you can expect. In a cluster running other kinds of workloads as well, you must make tradeoffs to make sure all
Hadoop components have enough memory to perform well, so you might cap the memory that Impala can use.
• Using Impala to Query the Amazon S3 Filesystem on page 566. Queries against data stored in the Amazon Simple
Storage Service (S3) have different performance characteristics than when the data is stored in HDFS.
A good source of tips related to scalability and performance tuning is the Impala Cookbook presentation. These slides
are updated periodically as new features come out and new benchmarks are performed.
Note: For smaller volumes of data, a few gigabytes or less for each table or partition, you might not
see significant performance differences between file formats. At small data volumes, reduced I/O
from an efficient compressed file format can be counterbalanced by reduced opportunity for parallel
execution. When planning for a production deployment or conducting benchmarks, always use realistic
data volumes to get a true picture of performance and scalability.
Gather statistics for all tables used in performance-critical or high-volume join queries
Gather the statistics with the COMPUTE STATS statement. See Performance Considerations for Join Queries on page
482 for details.
Hotspot analysis
In the context of Impala, a hotspot is defined as “an Impala daemon that for a single query or a workload is spending
a far greater amount of time processing data relative to its neighbours”.
Before discussing the options to tackle this issue some background is first required to understand how this problem
can occur.
By default, the scheduling of scan based plan fragments is deterministic. This means that for multiple queries needing
to read the same block of data, the same node will be picked to host the scan. The default scheduling logic does not
take into account node workload from prior queries. The complexity of materializing a tuple depends on a few factors,
namely: decoding and decompression. If the tuples are densely packed into data pages due to good
encoding/compression ratios, there will be more work required when reconstructing the data. Each compression codec
offers different performance tradeoffs and should be considered before writing the data. Due to the deterministic
nature of the scheduler, single nodes can become bottlenecks for highly concurrent queries that use the same tables.
If, for example, a Parquet based dataset is tiny, e.g. a small dimension table, such that it fits into a single HDFS block
(Impala by default will create 256 MB blocks when Parquet is used, each containing a single row group) then there are
a number of options that can be considered to resolve the potential scheduling hotspots when querying this data:
• In and higher, the scheduler’s deterministic behaviour can be changed using the following query options:
REPLICA_PREFERENCE and RANDOM_REPLICA. For a detailed description of each of these modes see IMPALA-2696.
• HDFS caching can be used to cache block replicas. This will cause the Impala scheduler to randomly pick (from
and higher) a node that is hosting a cached block replica for the scan. Note, although HDFS caching has benefits,
it serves only to help with the reading of raw block data and not cached tuple data, but with the right number of
cached replicas (by default, HDFS only caches one replica), even load distribution can be achieved for smaller
datasets.
• Do not compress the table data. The uncompressed table data spans more nodes and eliminates skew caused by
compression.
• Reduce the Parquet file size via the PARQUET_FILE_SIZE query option when writing the table data. Using this
approach the data will span more nodes. However it’s not recommended to drop the size below 32 MB.
If statistics are not available for all the tables in the join query, or if Impala chooses a join order that is not the most
efficient, you can override the automatic join order optimization by specifying the STRAIGHT_JOIN keyword immediately
after the SELECT and any DISTINCT or ALL keywords. In this case, Impala uses the order the tables appear in the
query to guide how the joins are processed.
When you use the STRAIGHT_JOIN technique, you must order the tables in the join query manually instead of relying
on the Impala optimizer. The optimizer uses sophisticated techniques to estimate the size of the result set at each
stage of the join. For manual ordering, use this heuristic approach to start with, and then experiment to fine-tune the
order:
• Specify the largest table first. This table is read from disk by each Impala node and so its size is not significant in
terms of memory usage during the query.
• Next, specify the smallest table. The contents of the second, third, and so on tables are all transmitted across the
network. You want to minimize the size of the result set from each subsequent stage of the join query. The most
likely approach involves joining a small table first, so that the result set remains small even as subsequent larger
tables are processed.
• Join the next smallest table, then the next smallest, and so on.
For example, if you had tables BIG, MEDIUM, SMALL, and TINY, the logical join order to try would be BIG, TINY, SMALL,
MEDIUM.
The terms “largest” and “smallest” refers to the size of the intermediate result set based on the number of rows and
columns from each table that are part of the result set. For example, if you join one table sales with another table
customers, a query might find results from 100 different customers who made a total of 5000 purchases. In that case,
you would specify SELECT ... FROM sales JOIN customers ..., putting customers on the right side because
it is smaller in the context of this query.
The Impala query planner chooses between different techniques for performing join queries, depending on the absolute
and relative sizes of the tables. Broadcast joins are the default, where the right-hand table is considered to be smaller
than the left-hand table, and its contents are sent to all the other nodes involved in the query. The alternative technique
is known as a partitioned join (not related to a partitioned table), which is more suitable for large tables of roughly
equal size. With this technique, portions of each table are sent to appropriate other nodes where those subsets of
rows can be processed in parallel. The choice of broadcast or partitioned join also depends on statistics being available
for all tables in the join, gathered by the COMPUTE STATS statement.
To see which join strategy is used for a particular query, issue an EXPLAIN statement for the query. If you find that a
query uses a broadcast join when you know through benchmarking that a partitioned join would be more efficient, or
vice versa, add a hint to the query to specify the precise join mechanism to use. See Query Hints in Impala SELECT
Statements on page 301 for details.
Note:
The STRAIGHT_JOIN hint affects the join order of table references in the query block containing the
hint. It does not affect the join order of nested queries, such as views, inline views, or WHERE-clause
subqueries. To use this hint for performance tuning of complex queries, apply the hint to all query
blocks that need a fixed join order.
In this example, the subselect from the BIG table produces a very small result set, but the table might still be treated
as if it were the biggest and placed first in the join order. Using STRAIGHT_JOIN for the last join clause prevents the
final table from being reordered, keeping it as the rightmost table in the join order.
select straight_join x from medium join small join (select * from big where c1 < 10) as
big
where medium.id = small.id and small.id = big.id;
-- If the query contains [DISTINCT | ALL], the hint goes after those keywords.
select distinct straight_join x from medium join small join (select * from big where c1
< 10) as big
where medium.id = small.id and small.id = big.id;
[localhost:21000] > create table big stored as parquet as select * from raw_data;
+----------------------------+
| summary |
+----------------------------+
| Inserted 1000000000 row(s) |
+----------------------------+
Returned 1 row(s) in 671.56s
[localhost:21000] > desc big;
+-----------+---------+---------+
| name | type | comment |
+-----------+---------+---------+
| id | int | |
| val | int | |
| zfill | string | |
| name | string | |
| assertion | boolean | |
+-----------+---------+---------+
Returned 5 row(s) in 0.01s
[localhost:21000] > create table medium stored as parquet as select * from big limit
200 * floor(1e6);
+---------------------------+
| summary |
+---------------------------+
| Inserted 200000000 row(s) |
+---------------------------+
Returned 1 row(s) in 138.31s
[localhost:21000] > create table small stored as parquet as select id,val,name from big
where assertion = true limit 1 * floor(1e6);
+-------------------------+
| summary |
+-------------------------+
| Inserted 1000000 row(s) |
+-------------------------+
Returned 1 row(s) in 6.32s
For any kind of performance experimentation, use the EXPLAIN statement to see how any expensive query will be
performed without actually running it, and enable verbose EXPLAIN plans containing more performance-oriented
detail: The most interesting plan lines are highlighted in bold, showing that without statistics for the joined tables,
Impala cannot make a good estimate of the number of rows involved at each stage of processing, and is likely to stick
with the BROADCAST join mechanism that sends a complete copy of one of the tables to each node.
+----------------------------------------------------------+
Returned 64 row(s) in 0.04s
Gathering statistics for all the tables is straightforward, one COMPUTE STATS statement per table:
With statistics in place, Impala can choose a more effective join order rather than following the left-to-right sequence
of tables in the query, and can choose BROADCAST or PARTITIONED join strategies based on the overall sizes and
number of rows in the table:
[localhost:21000] > explain select count(*) from medium join big where big.id = medium.id;
Query: explain select count(*) from medium join big where big.id = medium.id
+-----------------------------------------------------------+
| Explain String |
+-----------------------------------------------------------+
| Estimated Per-Host Requirements: Memory=937.23MB VCores=2 |
| |
| PLAN FRAGMENT 0 |
| PARTITION: UNPARTITIONED |
| |
| 6:AGGREGATE (merge finalize) |
| | output: SUM(COUNT(*)) |
| | cardinality: 1 |
| | per-host memory: unavailable |
| | tuple ids: 2 |
| | |
| 5:EXCHANGE |
| cardinality: 1 |
| per-host memory: unavailable |
| tuple ids: 2 |
| |
| PLAN FRAGMENT 1 |
| PARTITION: RANDOM |
| |
| STREAM DATA SINK |
| EXCHANGE ID: 5 |
| UNPARTITIONED |
| |
| 3:AGGREGATE |
| | output: COUNT(*) |
| | cardinality: 1 |
| | per-host memory: 10.00MB |
| | tuple ids: 2 |
| | |
| 2:HASH JOIN |
| | join op: INNER JOIN (BROADCAST) |
| | hash predicates: |
| | big.id = medium.id |
| | cardinality: 1443004441 |
| | per-host memory: 839.23MB |
| | tuple ids: 1 0 |
| | |
| |----4:EXCHANGE |
| | cardinality: 200000000 |
| | per-host memory: 0B |
| | tuple ids: 0 |
| | |
| 1:SCAN HDFS |
| table=join_order.big #partitions=1/1 size=23.12GB |
| table stats: 1000000000 rows total |
| column stats: all |
| cardinality: 1000000000 |
| per-host memory: 88.00MB |
| tuple ids: 1 |
| |
| PLAN FRAGMENT 2 |
| PARTITION: RANDOM |
| |
| STREAM DATA SINK |
| EXCHANGE ID: 4 |
| UNPARTITIONED |
| |
| 0:SCAN HDFS |
| table=join_order.medium #partitions=1/1 size=4.62GB |
| table stats: 200000000 rows total |
| column stats: all |
| cardinality: 200000000 |
| per-host memory: 88.00MB |
| tuple ids: 0 |
+-----------------------------------------------------------+
Returned 64 row(s) in 0.04s
[localhost:21000] > explain select count(*) from small join big where big.id = small.id;
Query: explain select count(*) from small join big where big.id = small.id
+-----------------------------------------------------------+
| Explain String |
+-----------------------------------------------------------+
| Estimated Per-Host Requirements: Memory=101.15MB VCores=2 |
| |
| PLAN FRAGMENT 0 |
| PARTITION: UNPARTITIONED |
| |
| 6:AGGREGATE (merge finalize) |
| | output: SUM(COUNT(*)) |
| | cardinality: 1 |
| | per-host memory: unavailable |
| | tuple ids: 2 |
| | |
| 5:EXCHANGE |
| cardinality: 1 |
| per-host memory: unavailable |
| tuple ids: 2 |
| |
| PLAN FRAGMENT 1 |
| PARTITION: RANDOM |
| |
| STREAM DATA SINK |
| EXCHANGE ID: 5 |
| UNPARTITIONED |
| |
| 3:AGGREGATE |
| | output: COUNT(*) |
| | cardinality: 1 |
| | per-host memory: 10.00MB |
| | tuple ids: 2 |
| | |
| 2:HASH JOIN |
| | join op: INNER JOIN (BROADCAST) |
| | hash predicates: |
| | big.id = small.id |
| | cardinality: 1000000000 |
| | per-host memory: 3.15MB |
| | tuple ids: 1 0 |
| | |
| |----4:EXCHANGE |
| | cardinality: 1000000 |
| | per-host memory: 0B |
| | tuple ids: 0 |
| | |
| 1:SCAN HDFS |
| table=join_order.big #partitions=1/1 size=23.12GB |
| table stats: 1000000000 rows total |
| column stats: all |
| cardinality: 1000000000 |
| per-host memory: 88.00MB |
| tuple ids: 1 |
| |
| PLAN FRAGMENT 2 |
| PARTITION: RANDOM |
| |
| STREAM DATA SINK |
| EXCHANGE ID: 4 |
| UNPARTITIONED |
| |
| 0:SCAN HDFS |
| table=join_order.small #partitions=1/1 size=17.93MB |
| table stats: 1000000 rows total |
| column stats: all |
| cardinality: 1000000 |
| per-host memory: 32.00MB |
| tuple ids: 0 |
+-----------------------------------------------------------+
Returned 64 row(s) in 0.03s
When queries like these are actually run, the execution times are relatively consistent regardless of the table order in
the query text. Here are examples using both the unique ID column and the VAL column containing duplicate values:
[localhost:21000] > select count(*) from big join small on (big.id = small.id);
Query: select count(*) from big join small on (big.id = small.id)
+----------+
| count(*) |
+----------+
| 1000000 |
+----------+
Returned 1 row(s) in 21.68s
[localhost:21000] > select count(*) from small join big on (big.id = small.id);
Query: select count(*) from small join big on (big.id = small.id)
+----------+
| count(*) |
+----------+
| 1000000 |
+----------+
Returned 1 row(s) in 20.45s
[localhost:21000] > select count(*) from big join small on (big.val = small.val);
+------------+
| count(*) |
+------------+
| 2000948962 |
+------------+
Returned 1 row(s) in 108.85s
[localhost:21000] > select count(*) from small join big on (big.val = small.val);
+------------+
| count(*) |
+------------+
| 2000948962 |
+------------+
Returned 1 row(s) in 100.76s
Note: When examining the performance of join queries and the effectiveness of the join order
optimization, make sure the query involves enough data and cluster resources to see a difference
depending on the query plan. For example, a single data file of just a few megabytes will reside in a
single HDFS block and be processed on a single node. Likewise, if you use a single-node or two-node
cluster, there might not be much difference in efficiency for the broadcast or partitioned join strategies.
Note:
Originally, Impala relied on the Hive mechanism for collecting statistics, through the Hive ANALYZE
TABLE statement which initiates a MapReduce job. For better user-friendliness and reliability, Impala
implements its own COMPUTE STATS statement in Impala 1.2.2 and higher, along with the DROP
STATS, SHOW TABLE STATS, and SHOW COLUMN STATS statements.
|...
+------------+--------+---------+--------------+-------------------+---------+-------------------+...
Impala performs some optimizations using this metadata on its own, and other optimizations by using a combination
of table and column statistics.
To check that table statistics are available for a table, and see the details of those statistics, use the statement SHOW
TABLE STATS table_name. See SHOW Statement on page 322 for details.
If you use the Hive-based methods of gathering statistics, see the Hive wiki for information about the required
configuration on the Hive side. Cloudera recommends using the Impala COMPUTE STATS statement to avoid potential
configuration and scalability issues with the statistics-gathering process.
If you run the Hive statement ANALYZE TABLE COMPUTE STATISTICS FOR COLUMNS, Impala can only use the
resulting column statistics if the table is unpartitioned. Impala cannot use Hive-generated column statistics for a
partitioned table.
Note:
For column statistics to be effective in Impala, you also need to have table statistics for the applicable
tables, as described in Overview of Table Statistics on page 489. When you use the Impala COMPUTE
STATS statement, both table and column statistics are automatically gathered at the same time, for
all columns in the table.
Currently, the COMPUTE STATS statement under CDH 4 does not store any statistics for DECIMAL
columns. When Impala runs under CDH 5, which has better support for DECIMAL in the metastore
database, COMPUTE STATS does collect statistics for DECIMAL columns and Impala uses the statistics
to optimize query performance.
Note: Prior to Impala 1.4.0, COMPUTE STATS counted the number of NULL values in each column
and recorded that figure in the metastore database. Because Impala does not currently make use of
the NULL count during query planning, Impala 1.4.0 and higher speeds up the COMPUTE STATS
statement by skipping this NULL counting.
To check whether column statistics are available for a particular set of columns, use the SHOW COLUMN STATS
table_name statement, or check the extended EXPLAIN output for a query against that table that refers to those
columns. See SHOW Statement on page 322 and EXPLAIN Statement on page 260 for details.
If you run the Hive statement ANALYZE TABLE COMPUTE STATISTICS FOR COLUMNS, Impala can only use the
resulting column statistics if the table is unpartitioned. Impala cannot use Hive-generated column statistics for a
partitioned table.
+-------+-------+-----+-------+--------+---------+--------------+-------------------+---------+...
| year | month | day | #Rows | #Files | Size | Bytes Cached | Cache Replication |
Format |...
+-------+-------+-----+-------+--------+---------+--------------+-------------------+---------+...
| 2013 | 12 | 1 | -1 | 1 | 2.51MB | NOT CACHED | NOT CACHED |
PARQUET |...
| 2013 | 12 | 2 | -1 | 1 | 2.53MB | NOT CACHED | NOT CACHED |
PARQUET |...
| 2013 | 12 | 3 | -1 | 1 | 2.52MB | NOT CACHED | NOT CACHED |
PARQUET |...
| 2013 | 12 | 4 | -1 | 1 | 2.51MB | NOT CACHED | NOT CACHED |
PARQUET |...
| 2013 | 12 | 5 | -1 | 1 | 2.52MB | NOT CACHED | NOT CACHED |
PARQUET |...
| Total | | | -1 | 5 | 12.58MB | 0B | |
|...
+-------+-------+-----+-------+--------+---------+--------------+-------------------+---------+...
Note: Partitioned tables can grow so large that scanning the entire table, as the COMPUTE STATS
statement does, is impractical just to update the statistics for a new partition. The standard COMPUTE
STATS statement might take hours, or even days. That situation is where you switch to using
incremental statistics, a feature available in CDH 5.3 / Impala 2.1 and higher. See Overview of
Incremental Statistics on page 493 for details about this feature and the COMPUTE INCREMENTAL
STATS syntax.
If you run the Hive statement ANALYZE TABLE COMPUTE STATISTICS FOR COLUMNS, Impala can only use the
resulting column statistics if the table is unpartitioned. Impala cannot use Hive-generated column statistics for a
partitioned table.
The metadata for incremental statistics is handled differently from the original style of statistics:
• If you have an existing partitioned table for which you have already computed statistics, issuing COMPUTE
INCREMENTAL STATS without a partition clause causes Impala to rescan the entire table. Once the incremental
statistics are computed, any future COMPUTE INCREMENTAL STATS statements only scan any new partitions
and any partitions where you performed DROP INCREMENTAL STATS.
• The SHOW TABLE STATS and SHOW PARTITIONS statements now include an additional column showing whether
incremental statistics are available for each column. A partition could already be covered by the original type of
statistics based on a prior COMPUTE STATS statement, as indicated by a value other than -1 under the #Rows
column. Impala query planning uses either kind of statistics when available.
• COMPUTE INCREMENTAL STATS takes more time than COMPUTE STATS for the same volume of data. Therefore
it is most suitable for tables with large data volume where new partitions are added frequently, making it impractical
to run a full COMPUTE STATS operation for each new partition. For unpartitioned tables, or partitioned tables
that are loaded once and not updated with new partitions, use the original COMPUTE STATS syntax.
• COMPUTE INCREMENTAL STATS uses some memory in the catalogd process, proportional to the number of
partitions and number of columns in the applicable table. The memory overhead is approximately 400 bytes for
each column in each partition. This memory is reserved in the catalogd daemon, the statestored daemon,
and in each instance of the impalad daemon.
• In cases where new files are added to an existing partition, issue a REFRESH statement for the table, followed by
a DROP INCREMENTAL STATS and COMPUTE INCREMENTAL STATS sequence for the changed partition.
• The DROP INCREMENTAL STATS statement operates only on a single partition at a time. To remove statistics
(whether incremental or not) from all partitions of a table, issue a DROP STATS statement with no INCREMENTAL
or PARTITION clauses.
The following considerations apply to incremental statistics when the structure of an existing table is changed (known
as schema evolution):
• If you use an ALTER TABLE statement to drop a column, the existing statistics remain valid and COMPUTE
INCREMENTAL STATS does not rescan any partitions.
• If you use an ALTER TABLE statement to add a column, Impala rescans all partitions and fills in the appropriate
column-level values the next time you run COMPUTE INCREMENTAL STATS.
• If you use an ALTER TABLE statement to change the data type of a column, Impala rescans all partitions and fills
in the appropriate column-level values the next time you run COMPUTE INCREMENTAL STATS.
• If you use an ALTER TABLE statement to change the file format of a table, the existing statistics remain valid and
a subsequent COMPUTE INCREMENTAL STATS does not rescan any partitions.
See COMPUTE STATS Statement on page 226 and DROP STATS Statement on page 255 for syntax details.
+-------+--------+------+--------------+--------+-------------------+
| 0 | 0 | 0B | NOT CACHED | TEXT | false |
+-------+--------+------+--------------+--------+-------------------+
The following example shows a similar progression with a partitioned table. Initially, #Rows is -1. After a COMPUTE
STATS operation, #Rows changes to an accurate value. Any newly added partition starts with no statistics, meaning
that you must collect statistics after adding a new partition.
Note: Because the default COMPUTE STATS statement creates and updates statistics for all partitions
in a table, if you expect to frequently add new partitions, use the COMPUTE INCREMENTAL STATS
syntax instead, which lets you compute stats for a single specified partition, or only for those partitions
that do not already have incremental stats.
If checking each individual table is impractical, due to a large number of tables or views that hide the underlying base
tables, you can also check for missing statistics for a particular query. Use the EXPLAIN statement to preview query
efficiency before actually running the query. Use the query profile output available through the PROFILE command
in impala-shell or the web UI to verify query execution and timing after running the query. Both the EXPLAIN plan
and the PROFILE output display a warning if any tables or partitions involved in the query do not have statistics.
| | |
| 00:SCAN HDFS [incremental_stats.no_stats] |
| partitions=1/1 files=0 size=0B |
+------------------------------------------------------------------------------------+
Because Impala uses the partition pruning technique when possible to only evaluate certain partitions, if you have a
partitioned table with statistics for some partitions and not others, whether or not the EXPLAIN statement shows the
warning depends on the actual partitions used by the query. For example, you might see warnings or not for different
queries against the same table:
-- No warning because all the partitions for the year 2012 have stats.
EXPLAIN SELECT ... FROM t1 WHERE year = 2012;
To confirm if any partitions at all in the table are missing statistics, you might explain a query that scans the entire
table, such as SELECT COUNT(*) FROM table_name.
-- Set total number of rows. Applies to both unpartitioned and partitioned tables.
alter table table_name set tblproperties('numRows'='new_value',
'STATS_GENERATED_VIA_STATS_TASK'='true');
-- Set total number of rows for a specific partition. Applies to partitioned tables
only.
-- You must specify all the partition key columns in the PARTITION clause.
alter table table_name partition (keycol1=val1,keycol2=val2...) set
tblproperties('numRows'='new_value', 'STATS_GENERATED_VIA_STATS_TASK'='true');
This statement avoids re-scanning any data files. (The requirement to include the STATS_GENERATED_VIA_STATS_TASK
property is relatively new, as a result of the issue HIVE-8648 for the Hive metastore.)
For a partitioned table, update both the per-partition number of rows and the number of rows for the whole table:
-- If the table originally contained 1 million rows, and we add another partition with
30 thousand rows,
-- change the numRows property for the partition and the overall table.
alter table partitioned_data partition(year=2009, month=4) set tblproperties
('numRows'='30000', 'STATS_GENERATED_VIA_STATS_TASK'='true');
alter table partitioned_data set tblproperties ('numRows'='1030000',
'STATS_GENERATED_VIA_STATS_TASK'='true');
In practice, the COMPUTE STATS statement, or COMPUTE INCREMENTAL STATS for a partitioned table, should be
fast and convenient enough that this technique is only useful for the very largest partitioned tables. Because the column
statistics might be left in a stale state, do not use this technique as a replacement for COMPUTE STATS. Only use this
technique if all other means of collecting statistics are impractical, or as a low-overhead operation that you run in
between periodic COMPUTE STATS or COMPUTE INCREMENTAL STATS operations.
This example shows table and column statistics for the STORE column used in the TPC-DS benchmarks for decision
support systems. It is a tiny table holding data for 12 stores. Initially, before any statistics are gathered by a COMPUTE
STATS statement, most of the numeric fields show placeholder values of -1, indicating that the figures are unknown.
The figures that are filled in are values that are easily countable or deducible at the physical level, such as the number
of files, total data size of the files, and the maximum and average sizes for data types that have a constant size such
as INT, FLOAT, and TIMESTAMP.
| s_company_id | INT | -1 | -1 | 4 | 4 |
| s_company_name | STRING | -1 | -1 | -1 | -1 |
| s_street_number | STRING | -1 | -1 | -1 | -1 |
| s_street_name | STRING | -1 | -1 | -1 | -1 |
| s_street_type | STRING | -1 | -1 | -1 | -1 |
| s_suite_number | STRING | -1 | -1 | -1 | -1 |
| s_city | STRING | -1 | -1 | -1 | -1 |
| s_county | STRING | -1 | -1 | -1 | -1 |
| s_state | STRING | -1 | -1 | -1 | -1 |
| s_zip | STRING | -1 | -1 | -1 | -1 |
| s_country | STRING | -1 | -1 | -1 | -1 |
| s_gmt_offset | FLOAT | -1 | -1 | 4 | 4 |
| s_tax_percentage | FLOAT | -1 | -1 | 4 | 4 |
+--------------------+-----------+------------------+--------+----------+----------+
Returned 29 row(s) in 0.04s
With the Hive ANALYZE TABLE statement for column statistics, you had to specify each column for which to gather
statistics. The Impala COMPUTE STATS statement automatically gathers statistics for all columns, because it reads
through the entire table relatively quickly and can efficiently compute the values for all the columns. This example
shows how after running the COMPUTE STATS statement, statistics are filled in for both the table and all its columns:
|
| s_company_id | INT | 1 | -1 | 4 | 4
|
| s_company_name | STRING | 1 | -1 | 7 | 7
|
| s_street_number | STRING | 9 | -1 | 3 |
2.833300113677979 |
| s_street_name | STRING | 12 | -1 | 11 |
6.583300113677979 |
| s_street_type | STRING | 8 | -1 | 9 |
4.833300113677979 |
| s_suite_number | STRING | 11 | -1 | 9 | 8.25
|
| s_city | STRING | 2 | -1 | 8 | 6.5
|
| s_county | STRING | 1 | -1 | 17 | 17
|
| s_state | STRING | 1 | -1 | 2 | 2
|
| s_zip | STRING | 2 | -1 | 5 | 5
|
| s_country | STRING | 1 | -1 | 13 | 13
|
| s_gmt_offset | FLOAT | 1 | -1 | 4 | 4
|
| s_tax_percentage | FLOAT | 5 | -1 | 4 | 4
|
+--------------------+-----------+------------------+--------+----------+-------------------+
Returned 29 row(s) in 0.04s
The following example shows how statistics are represented for a partitioned table. In this case, we have set up a table
to hold the world's most trivial census data, a single STRING field, partitioned by a YEAR column. The table statistics
include a separate entry for each partition, plus final totals for the numeric fields. The column statistics include some
easily deducible facts for the partitioning column, such as the number of distinct values (the number of partition
subdirectories).
The following example shows how the statistics are filled in by a COMPUTE STATS statement in Impala.
For examples showing how some queries work differently when statistics are available, see Examples of Join Order
Optimization on page 484. You can see how Impala executes a query differently in each case by observing the EXPLAIN
output before and after collecting statistics. Measure the before and after query times, and examine the throughput
numbers in before and after SUMMARY or PROFILE output, to verify how much the improved plan speeds up performance.
• You can restrict the amount of memory Impala reserves during query execution by specifying the -mem_limit
option for the impalad daemon. See Modifying Impala Startup Options on page 43 for details. This limit applies
only to the memory that is directly consumed by queries; Impala reserves additional memory at startup, for
example to hold cached metadata.
• For production deployment, Cloudera recommends that you implement resource isolation using mechanisms such
as cgroups, which you can configure using Cloudera Manager. For details, see the Static Resource Pools in the
Cloudera Manager documentation.
For details about the hdfs cacheadmin command, see the CDH documentation.
Once HDFS caching is enabled and one or more pools are available, see Enabling HDFS Caching for Impala Tables and
Partitions on page 502 for how to choose which Impala data to load into the HDFS cache. On the Impala side, you specify
the cache pool name defined by the hdfs cacheadmin command in the Impala DDL statements that enable HDFS
caching for a table or partition, such as CREATE TABLE ... CACHED IN pool or ALTER TABLE ... SET CACHED
IN pool.
In Impala 2.2 / CDH 5.4 and higher, the optional WITH REPLICATION clause for CREATE TABLE and ALTER TABLE
lets you specify a replication factor, the number of hosts on which to cache the same data blocks. When Impala
processes a cached data block, where the cache replication factor is greater than 1, Impala randomly selects a host
that has a cached copy of that data block. This optimization avoids excessive CPU usage on a single host when the
same cached data block is processed multiple times. Cloudera recommends specifying a value greater than or equal
to the HDFS block replication factor.
-- Cache the data from one partition on up to 4 hosts, to minimize CPU load on any
-- single host when the same data block is processed multiple times.
alter table census partition (year=1970)
set cached in 'pool_name' with replication = 4;
See CREATE TABLE Statement on page 235 for the full syntax.
Other memory considerations:
Certain DDL operations, such as ALTER TABLE ... SET LOCATION, are blocked while the underlying HDFS directories
contain cached files. You must uncache the files first, before changing the location, dropping the table, and so on.
When data is requested to be pinned in memory, that process happens in the background without blocking access to
the data while the caching is in progress. Loading the data from disk could take some time. Impala reads each HDFS
data block from memory if it has been pinned already, or from disk if it has not been pinned yet.
The amount of data that you can pin on each node through the HDFS caching mechanism is subject to a quota that is
enforced by the underlying HDFS service. Before requesting to pin an Impala table or partition in memory, check that
its size does not exceed this quota.
Note: Because the HDFS cache consists of combined memory from all the DataNodes in the cluster,
cached tables or partitions can be bigger than the amount of HDFS cache memory on any single host.
• If you have designated a table or partition as cached through the CREATE TABLE or ALTER TABLE statements,
subsequent attempts to relocate the table or partition through an ALTER TABLE ... SET LOCATION statement
will fail. You must issue an ALTER TABLE ... SET UNCACHED statement for the table or partition first. Otherwise,
Impala would lose track of some cached data files and have no way to uncache them later.
For queries involving smaller amounts of data, or in single-user workloads, you might not notice a significant difference
in query response time with or without HDFS caching. Even with HDFS caching turned off, the data for the query might
still be in the Linux OS buffer cache. The benefits become clearer as data volume increases, and especially as the system
processes more concurrent queries. HDFS caching improves the scalability of the overall system. That is, it prevents
query performance from declining when the workload outstrips the capacity of the Linux OS cache.
Due to a limitation of HDFS, zero-copy reads are not supported with encryption. Cloudera recommends not using HDFS
caching for Impala data files in encryption zones. The queries fall back to the normal read path during query execution,
which might cause some performance overhead.
SELECT considerations:
The Impala HDFS caching feature interacts with the SELECT statement and query performance as follows:
• Impala automatically reads from memory any data that has been designated as cached and actually loaded into
the HDFS cache. (It could take some time after the initial request to fully populate the cache for a table with large
size or many partitions.) The speedup comes from two aspects: reading from RAM instead of disk, and accessing
the data straight from the cache area instead of copying from one RAM area to another. This second aspect yields
further performance improvement over the standard OS caching mechanism, which still results in
memory-to-memory copying of cached data.
• For small amounts of data, the query speedup might not be noticeable in terms of wall clock time. The performance
might be roughly the same with HDFS caching turned on or off, due to recently used data being held in the Linux
OS cache. The difference is more pronounced with:
– Data volumes (for all queries running concurrently) that exceed the size of the Linux OS cache.
– A busy cluster running many concurrent queries, where the reduction in memory-to-memory copying and
overall memory usage during queries results in greater scalability and throughput.
– Thus, to really exercise and benchmark this feature in a development environment, you might need to simulate
realistic workloads and concurrent queries that match your production environment.
– One way to simulate a heavy workload on a lightly loaded system is to flush the OS buffer cache (on each
DataNode) between iterations of queries against the same tables or partitions:
$ sync
$ echo 1 > /proc/sys/vm/drop_caches
• Impala queries take advantage of HDFS cached data regardless of whether the cache directive was issued by
Impala or externally through the hdfs cacheadmin command, for example for an external table where the
cached data files might be accessed by several different Hadoop components.
• If your query returns a large result set, the time reported for the query could be dominated by the time needed
to print the results on the screen. To measure the time for the underlying query processing, query the COUNT()
of the big result set, which does all the same processing but only prints a single line to the screen.
Note: In the preceding example, replace hostname and port with the name and port of your
Impala server. The default port is 25000.
2. After the query completes, review the contents of the Impala logs. You should find a recent message similar to
the following:
The presence of remote scans may indicate impalad is not running on the correct nodes. This can be because some
DataNodes do not have impalad running or it can be because the impalad instance that is starting the query is unable
to contact one or more of the impalad instances.
To understand the causes of this issue:
1. Connect to the debugging web server. By default, this server runs on port 25000. This page lists all impalad
instances running in your cluster. If there are fewer instances than you expect, this often indicates some DataNodes
are not running impalad. Ensure impalad is started on all DataNodes.
2. If you are using multi-homed hosts, ensure that the Impala daemon's hostname resolves to the interface on which
impalad is running. The hostname Impala is using is displayed when impalad starts. To explicitly set the hostname,
use the --hostname flag.
3. Check that statestored is running as expected. Review the contents of the state store log to ensure all instances
of impalad are listed as having connected to the state store.
Starting Impala on page 42 and Using the Impala Shell (impala-shell Command) on page 470. Information on logging
can be found in Using Impala Logging on page 574. Log messages and their interpretations are as follows:
• See EXPLAIN_LEVEL Query Option on page 309 for details about the EXPLAIN_LEVEL query option, which lets you
customize how much detail to show in the EXPLAIN plan depending on whether you are doing high-level or
low-level tuning, dealing with logical or physical aspects of the query.
The EXPLAIN plan is also printed at the beginning of the query profile report described in Using the Query Profile for
Performance Tuning on page 509, for convenience in examining both the logical and physical aspects of the query
side-by-side.
The amount of detail displayed in the EXPLAIN output is controlled by the EXPLAIN_LEVEL query option. You typically
increase this setting from standard to extended (or from 1 to 2) when doublechecking the presence of table and
column statistics during performance tuning, or when estimating query resource usage in conjunction with the resource
management features in CDH 5.
Notice how the longest initial phase of the query is measured in seconds (s), while later phases working on smaller
intermediate results are measured in milliseconds (ms) or even nanoseconds (ns).
Here is an example from a more complicated query, as it would appear in the PROFILE output:
Operator #Hosts Avg Time Max Time #Rows Est. #Rows Peak Mem Est.
Peak Mem Detail
------------------------------------------------------------------------------------------------------------------------
09:MERGING-EXCHANGE 1 79.738us 79.738us 5 5 0
-1.00 B UNPARTITIONED
05:TOP-N 3 84.693us 88.810us 5 5 12.00 KB
120.00 B
PLAN FRAGMENT 0
PARTITION: UNPARTITIONED
4:EXCHANGE
cardinality: unavailable
PLAN FRAGMENT 1
PARTITION: RANDOM
2:HASH JOIN
| join op: INNER JOIN (BROADCAST)
| hash predicates:
| t1.id = t2.parent
| cardinality: unavailable
| per-host memory: 2.00GB
| tuple ids: 0 1
|
|----3:EXCHANGE
| cardinality: unavailable
| per-host memory: 0B
| tuple ids: 1
|
0:SCAN HDFS
table=stats_testing.t1 #partitions=1/1 size=33B
table stats: unavailable
column stats: unavailable
cardinality: unavailable
per-host memory: 32.00MB
tuple ids: 0
PLAN FRAGMENT 2
PARTITION: RANDOM
1:SCAN HDFS
table=stats_testing.t2 #partitions=1/1 size=960.00KB
table stats: unavailable
column stats: unavailable
cardinality: unavailable
per-host memory: 96.00MB
tuple ids: 1
----------------
Query Timeline: 20s670ms
- Start execution: 2.559ms (2.559ms)
- Planning finished: 23.587ms (21.27ms)
- Rows available: 666.199ms (642.612ms)
- First row fetched: 668.919ms (2.719ms)
- Unregister query: 20s668ms (20s000ms)
ImpalaServer:
- ClientFetchWaitTimer: 19s637ms
- RowMaterializationTimer: 167.121ms
Execution Profile 6540a03d4bee0691:4963d6269b210ebd:(Active: 837.815ms, % non-child:
0.00%)
Per Node Peak Memory Usage: impala-1.example.com:22000(7.42 MB)
- FinalizationTimer: 0ns
Coordinator Fragment:(Active: 195.198ms, % non-child: 0.00%)
MemoryUsage(500.0ms): 16.00 KB, 7.42 MB, 7.33 MB, 7.10 MB, 6.94 MB, 6.71 MB, 6.56
MB, 6.40 MB, 6.17 MB, 6.02 MB, 5.79 MB, 5.63 MB, 5.48 MB, 5.25 MB, 5.09 MB, 4.86 MB,
4.71 MB, 4.47 MB, 4.32 MB, 4.09 MB, 3.93 MB, 3.78 MB, 3.55 MB, 3.39 MB, 3.16 MB, 3.01
MB, 2.78 MB, 2.62 MB, 2.39 MB, 2.24 MB, 2.08 MB, 1.85 MB, 1.70 MB, 1.54 MB, 1.31 MB,
1.16 MB, 948.00 KB, 790.00 KB, 553.00 KB, 395.00 KB, 237.00 KB
ThreadUsage(500.0ms): 1
- AverageThreadTokens: 1.00
- PeakMemoryUsage: 7.42 MB
- PrepareTime: 36.144us
- RowsProduced: 98.30K (98304)
- TotalCpuTime: 20s449ms
- TotalNetworkWaitTime: 191.630ms
- TotalStorageWaitTime: 0ns
-
RowsRead: 3
-
RowsReturned: 3
-
RowsReturnedRate: 220.33 K/sec
-
ScanRangesComplete: 1
-
ScannerThreadsInvoluntaryContextSwitches: 26
-
ScannerThreadsTotalWallClockTime: 55.199ms
- DelimiterParseTime: 2.463us
- MaterializeTupleTime(*): 1.226us
- ScannerThreadsSysTime: 0ns
- ScannerThreadsUserTime: 42.993ms
- ScannerThreadsVoluntaryContextSwitches: 1
- TotalRawHdfsReadTime(*): 112.86us
- TotalReadThroughput: 0.00 /sec
Averaged Fragment 2:(Active: 190.120ms, % non-child: 0.00%)
split sizes: min: 960.00 KB, max: 960.00 KB, avg: 960.00 KB, stddev: 0.00
completion times: min:191.736ms max:191.736ms mean: 191.736ms stddev:0ns
execution rates: min:4.89 MB/sec max:4.89 MB/sec mean:4.89 MB/sec stddev:0.00
/sec
num instances: 1
- AverageThreadTokens: 0.00
- PeakMemoryUsage: 906.33 KB
- PrepareTime: 3.67ms
- RowsProduced: 98.30K (98304)
- TotalCpuTime: 403.351ms
- TotalNetworkWaitTime: 34.999ms
- TotalStorageWaitTime: 108.675ms
CodeGen:(Active: 162.57ms, % non-child: 85.24%)
- CodegenTime: 3.133ms
- CompileTime: 148.316ms
- LoadTime: 12.317ms
- ModuleFileSize: 95.27 KB
DataStreamSender (dst_id=3):(Active: 70.620ms, % non-child: 37.14%)
- BytesSent: 1.15 MB
- NetworkThroughput(*): 23.30 MB/sec
- OverallThroughput: 16.23 MB/sec
- PeakMemoryUsage: 5.33 KB
- SerializeBatchTime: 22.69ms
- ThriftTransmitTime(*): 49.178ms
- UncompressedRowBatchSize: 3.28 MB
HDFS_SCAN_NODE (id=1):(Active: 118.839ms, % non-child: 62.51%)
- AverageHdfsReadThreadConcurrency: 0.00
- AverageScannerThreadConcurrency: 0.00
- BytesRead: 960.00 KB
- BytesReadLocal: 960.00 KB
- BytesReadShortCircuit: 960.00 KB
- NumDisksAccessed: 1
- NumScannerThreadsStarted: 1
- PeakMemoryUsage: 869.00 KB
- PerReadThreadRawHdfsThroughput: 130.21 MB/sec
- RowsRead: 98.30K (98304)
- RowsReturned: 98.30K (98304)
- RowsReturnedRate: 827.20 K/sec
- ScanRangesComplete: 15
- ScannerThreadsInvoluntaryContextSwitches: 34
- ScannerThreadsTotalWallClockTime: 189.774ms
- DelimiterParseTime: 15.703ms
- MaterializeTupleTime(*): 3.419ms
- ScannerThreadsSysTime: 1.999ms
- ScannerThreadsUserTime: 44.993ms
- ScannerThreadsVoluntaryContextSwitches: 118
- TotalRawHdfsReadTime(*): 7.199ms
- TotalReadThroughput: 0.00 /sec
Fragment 1:
Instance 6540a03d4bee0691:4963d6269b210ebf
(host=impala-1.example.com:22000):(Active: 442.360ms, % non-child: 0.00%)
Hdfs split stats (<volume id>:<# splits>/<split lengths>): 0:1/33.00 B
MemoryUsage(500.0ms): 69.33 KB
ThreadUsage(500.0ms): 1
- AverageThreadTokens: 1.00
- PeakMemoryUsage: 6.06 MB
- PrepareTime: 7.291ms
- RowsProduced: 98.30K (98304)
- TotalCpuTime: 784.259ms
- TotalNetworkWaitTime: 388.818ms
- TotalStorageWaitTime: 3.934ms
CodeGen:(Active: 312.862ms, % non-child: 70.73%)
- CodegenTime: 2.669ms
- CompileTime: 302.467ms
- LoadTime: 9.231ms
- ModuleFileSize: 95.27 KB
DataStreamSender (dst_id=4):(Active: 80.63ms, % non-child: 18.10%)
- BytesSent: 2.33 MB
- NetworkThroughput(*): 35.89 MB/sec
- OverallThroughput: 29.06 MB/sec
- PeakMemoryUsage: 5.33 KB
- SerializeBatchTime: 26.487ms
- ThriftTransmitTime(*): 64.814ms
- UncompressedRowBatchSize: 6.66 MB
HASH_JOIN_NODE (id=2):(Active: 362.25ms, % non-child: 3.92%)
ExecOption: Build Side Codegen Enabled, Probe Side Codegen Enabled, Hash Table
Built Asynchronously
- BuildBuckets: 1.02K (1024)
- BuildRows: 98.30K (98304)
- BuildTime: 12.622ms
- LoadFactor: 0.00
- PeakMemoryUsage: 6.02 MB
- ProbeRows: 3
- ProbeTime: 3.579ms
- RowsReturned: 98.30K (98304)
- RowsReturnedRate: 271.54 K/sec
EXCHANGE_NODE (id=3):(Active: 344.680ms, % non-child: 77.92%)
- BytesReceived: 1.15 MB
- ConvertRowBatchTime: 2.792ms
- DataArrivalWaitTime: 339.936ms
- DeserializeRowBatchTimer: 9.910ms
- FirstBatchArrivalWaitTime: 199.474ms
- PeakMemoryUsage: 156.00 KB
- RowsReturned: 98.30K (98304)
- RowsReturnedRate: 285.20 K/sec
- SendersBlockedTimer: 0ns
- SendersBlockedTotalTimer(*): 0ns
HDFS_SCAN_NODE (id=0):(Active: 13.616us, % non-child: 0.00%)
Hdfs split stats (<volume id>:<# splits>/<split lengths>): 0:1/33.00 B
Hdfs Read Thread Concurrency Bucket: 0:0% 1:0%
File Formats: TEXT/NONE:1
ExecOption: Codegen enabled: 1 out of 1
- AverageHdfsReadThreadConcurrency: 0.00
- AverageScannerThreadConcurrency: 0.00
- BytesRead: 33.00 B
- BytesReadLocal: 33.00 B
- BytesReadShortCircuit: 33.00 B
- NumDisksAccessed: 1
- NumScannerThreadsStarted: 1
- PeakMemoryUsage: 46.00 KB
- PerReadThreadRawHdfsThroughput: 287.52 KB/sec
- RowsRead: 3
- RowsReturned: 3
- RowsReturnedRate: 220.33 K/sec
- ScanRangesComplete: 1
- ScannerThreadsInvoluntaryContextSwitches: 26
- ScannerThreadsTotalWallClockTime: 55.199ms
- DelimiterParseTime: 2.463us
- MaterializeTupleTime(*): 1.226us
- ScannerThreadsSysTime: 0ns
- ScannerThreadsUserTime: 42.993ms
- ScannerThreadsVoluntaryContextSwitches: 1
- TotalRawHdfsReadTime(*): 112.86us
- TotalReadThroughput: 0.00 /sec
Fragment 2:
Instance 6540a03d4bee0691:4963d6269b210ec0
(host=impala-1.example.com:22000):(Active: 190.120ms, % non-child: 0.00%)
Hdfs split stats (<volume id>:<# splits>/<split lengths>): 0:15/960.00 KB
- AverageThreadTokens: 0.00
- PeakMemoryUsage: 906.33 KB
- PrepareTime: 3.67ms
- RowsProduced: 98.30K (98304)
- TotalCpuTime: 403.351ms
- TotalNetworkWaitTime: 34.999ms
- TotalStorageWaitTime: 108.675ms
CodeGen:(Active: 162.57ms, % non-child: 85.24%)
- CodegenTime: 3.133ms
- CompileTime: 148.316ms
- LoadTime: 12.317ms
- ModuleFileSize: 95.27 KB
DataStreamSender (dst_id=3):(Active: 70.620ms, % non-child: 37.14%)
- BytesSent: 1.15 MB
- NetworkThroughput(*): 23.30 MB/sec
- OverallThroughput: 16.23 MB/sec
- PeakMemoryUsage: 5.33 KB
- SerializeBatchTime: 22.69ms
- ThriftTransmitTime(*): 49.178ms
- UncompressedRowBatchSize: 3.28 MB
HDFS_SCAN_NODE (id=1):(Active: 118.839ms, % non-child: 62.51%)
Hdfs split stats (<volume id>:<# splits>/<split lengths>): 0:15/960.00 KB
Hdfs Read Thread Concurrency Bucket: 0:0% 1:0%
File Formats: TEXT/NONE:15
ExecOption: Codegen enabled: 15 out of 15
- AverageHdfsReadThreadConcurrency: 0.00
- AverageScannerThreadConcurrency: 0.00
- BytesRead: 960.00 KB
- BytesReadLocal: 960.00 KB
- BytesReadShortCircuit: 960.00 KB
- NumDisksAccessed: 1
- NumScannerThreadsStarted: 1
- PeakMemoryUsage: 869.00 KB
- PerReadThreadRawHdfsThroughput: 130.21 MB/sec
- RowsRead: 98.30K (98304)
- RowsReturned: 98.30K (98304)
- RowsReturnedRate: 827.20 K/sec
- ScanRangesComplete: 15
- ScannerThreadsInvoluntaryContextSwitches: 34
- ScannerThreadsTotalWallClockTime: 189.774ms
- DelimiterParseTime: 15.703ms
- MaterializeTupleTime(*): 3.419ms
- ScannerThreadsSysTime: 1.999ms
- ScannerThreadsUserTime: 44.993ms
- ScannerThreadsVoluntaryContextSwitches: 118
- TotalRawHdfsReadTime(*): 7.199ms
- TotalReadThroughput: 0.00 /sec
For each phase of the query, you see an Avg Time and a Max Time value, along with #Hosts indicating how many hosts
are involved in that query phase. For all the phases with #Hosts greater than one, look for cases where the maximum
time is substantially greater than the average time. Focus on the phases that took the longest, for example, those
taking multiple seconds rather than milliseconds or microseconds.
If you detect that some hosts take longer than others, first rule out non-Impala causes. One reason that some hosts
could be slower than others is if those hosts have less capacity than the others, or if they are substantially busier due
to unevenly distributed non-Impala workloads:
• For clusters running Impala, keep the relative capacities of all hosts roughly equal. Any cost savings from including
some underpowered hosts in the cluster will likely be outweighed by poor or uneven performance, and the time
spent diagnosing performance issues.
• If non-Impala workloads cause slowdowns on some hosts but not others, use the appropriate load-balancing
techniques for the non-Impala components to smooth out the load across the cluster.
If the hosts on your cluster are evenly powered and evenly loaded, examine the detailed profile output to determine
which host is taking longer than others for the query phase in question. Examine how many bytes are processed during
that phase on that host, how much memory is used, and how many bytes are transmitted across the network.
The most common symptom is a higher number of bytes read on one host than others, due to one host being requested
to process a higher number of HDFS data blocks. This condition is more likely to occur when the number of blocks
accessed by the query is relatively small. For example, if you have a 10-node cluster and the query processes 10 HDFS
blocks, each node might not process exactly one block. If one node sits idle while another node processes two blocks,
the query could take twice as long as if the data was perfectly distributed.
Possible solutions in this case include:
• If the query is artificially small, perhaps for benchmarking purposes, scale it up to process a larger data set. For
example, if some nodes read 10 HDFS data blocks while others read 11, the overall effect of the uneven distribution
is much lower than when some nodes did twice as much work as others. As a guideline, aim for a “sweet spot”
where each node reads 2 GB or more from HDFS per query. Queries that process lower volumes than that could
experience inconsistent performance that smooths out as queries become more data-intensive.
• If the query processes only a few large blocks, so that many nodes sit idle and cannot help to parallelize the query,
consider reducing the overall block size. For example, you might adjust the PARQUET_FILE_SIZE query option
before copying or converting data into a Parquet table. Or you might adjust the granularity of data files produced
earlier in the ETL pipeline by non-Impala components. In Impala 2.0 and later, the default Parquet block size is
256 MB, reduced from 1 GB, to improve parallelism for common cluster sizes and data volumes.
• Reduce the amount of compression applied to the data. For text data files, the highest degree of compression
(gzip) produces unsplittable files that are more difficult for Impala to process in parallel, and require extra memory
during processing to hold the compressed and uncompressed data simultaneously. For binary formats such as
Parquet and Avro, compression can result in fewer data blocks overall, but remember that when queries process
relatively few blocks, there is less opportunity for parallel execution and many nodes in the cluster might sit idle.
Note that when Impala writes Parquet data with the query option COMPRESSION_CODEC=NONE enabled, the data
is still typically compact due to the encoding schemes used by Parquet, independent of the final compression
step.
If it takes a very long time for a cluster to start up, and impala-shell consistently displays This Impala daemon
is not ready to accept user requests, the statestore might be taking too long to send the entire catalog
topic to the cluster. In this case, consider adding --load_catalog_in_background=false to your catalog service
configuration. This setting stops the statestore from loading the entire catalog into memory at cluster startup. Instead,
metadata for each table is loaded when the table is accessed for the first time.
In Impala 2.2, these values are halved: the threshold value is 8 MB and the memory buffer is 256 MB. In Impala 2.3 /
CDH 5.5 and higher, the memory for the buffer is allocated in pieces, only as needed, to avoid sudden large jumps in
memory usage. A query that uses multiple such operators might allocate multiple such memory buffers, as the size of
the data structure for each operator crosses the threshold on a particular host.
Therefore, a query that processes a relatively small amount of data on each host would likely never reach the threshold
for any operator, and would never allocate any extra memory buffers. A query that did process millions of groups,
distinct values, join keys, and so on might cross the threshold, causing its memory requirement to rise suddenly and
then flatten out. The larger the cluster, less data is processed on any particular host, thus reducing the chance of
requiring the extra memory allocation.
Added in: This feature was added to the ORDER BY clause in Impala 1.4 for CDH 4, and in CDH 5.1. This feature was
extended to cover join queries, aggregation functions, and analytic functions in Impala 2.0 for CDH 4, and in CDH 5.2.
The size of the memory work area required by each operator that spills was reduced from 512 megabytes to 256
megabytes in Impala 2.2 (CDH 5.4).
Avoiding queries that spill to disk:
Because the extra I/O can impose significant performance overhead on these types of queries, try to avoid this situation
by using the following steps:
1. Detect how often queries spill to disk, and how much temporary data is written. Refer to the following sources:
• The output of the PROFILE command in the impala-shell interpreter. This data shows the memory usage
for each host and in total across the cluster. The BlockMgr.BytesWritten counter reports how much data
was written to disk during the query.
• The Impala Queries dialog in Cloudera Manager. You can see the peak memory usage for a query, combined
across all nodes in the cluster.
• The Queries tab in the Impala debug web user interface. Select the query to examine and click the
corresponding Profile link. This data breaks down the memory usage for a single host within the cluster, the
host whose web interface you are connected to.
2. Use one or more techniques to reduce the possibility of the queries spilling to disk:
• Increase the Impala memory limit if practical, for example, if you can increase the available memory by more
than the amount of temporary data written to disk on a particular node. Remember that in Impala 2.0 and
later, you can issue SET MEM_LIMIT as a SQL statement, which lets you fine-tune the memory usage for
queries from JDBC and ODBC applications.
• Increase the number of nodes in the cluster, to increase the aggregate memory available to Impala and reduce
the amount of memory required on each node.
• Increase the overall memory capacity of each DataNode at the hardware level.
• On a cluster with resources shared between Impala and other Hadoop components, use resource management
features to allocate more memory for Impala. See Integrated Resource Management with YARN on page 83
for details.
• If the memory pressure is due to running many concurrent queries rather than a few memory-intensive ones,
consider using the Impala admission control feature to lower the limit on the number of concurrent queries.
By spacing out the most resource-intensive queries, you can avoid spikes in memory usage and improve
overall response times. See Admission Control and Query Queuing on page 75 for details.
• Tune the queries with the highest memory requirements, using one or more of the following techniques:
– Run the COMPUTE STATS statement for all tables involved in large-scale joins and aggregation queries.
– Minimize your use of STRING columns in join columns. Prefer numeric values instead.
– Examine the EXPLAIN plan to understand the execution strategy being used for the most
resource-intensive queries. See Using the EXPLAIN Plan for Performance Tuning on page 507 for details.
– If Impala still chooses a suboptimal execution strategy even with statistics available, or if it is impractical
to keep the statistics up to date for huge or rapidly changing tables, add hints to the most
resource-intensive queries to select the right execution strategy. See Query Hints in Impala SELECT
Statements on page 301 for details.
• If your queries experience substantial performance overhead due to spilling, enable the
DISABLE_UNSAFE_SPILLS query option. This option prevents queries whose memory usage is likely to be
exorbitant from spilling to disk. See DISABLE_UNSAFE_SPILLS Query Option (CDH 5.2 or higher only) on page
308 for details. As you tune problematic queries using the preceding steps, fewer and fewer will be cancelled
by this option setting.
Issue the PROFILE command to get a detailed breakdown of the memory usage on each node during the query. The
crucial part of the profile output concerning memory is the BlockMgr portion. For example, this profile shows that
the query did not quite exceed the memory limit.
BlockMgr:
- BlockWritesIssued: 1
- BlockWritesOutstanding: 0
- BlocksCreated: 24
- BlocksRecycled: 1
- BufferedPins: 0
- MaxBlockSize: 8.00 MB (8388608)
- MemoryLimit: 200.00 MB (209715200)
- PeakMemoryUsage: 192.22 MB (201555968)
- TotalBufferWaitTime: 0ns
- TotalEncryptionTime: 0ns
- TotalIntegrityCheckTime: 0ns
- TotalReadBlockTime: 0ns
In this case, because the memory limit was already below any recommended value, I increased the volume of data for
the query rather than reducing the memory limit any further.
Set the MEM_LIMIT query option to a value that is smaller than the peak memory usage reported in the profile output.
Do not specify a memory limit lower than about 300 MB, because with such a low limit, queries could fail to start for
other reasons. Now try the memory-intensive query again.
Check if the query fails with a message like the following:
WARNINGS: Spilling has been disabled for plans that do not have stats and are not hinted
to prevent potentially bad plans from using too many cluster resources. Compute stats
on
these tables, hint the plan or disable this behavior via query options to enable spilling.
If so, the query could have consumed substantial temporary disk space, slowing down so much that it would not
complete in any reasonable time. Rather than rely on the spill-to-disk feature in this case, issue the COMPUTE STATS
statement for the table or tables in your sample query. Then run the query again, check the peak memory usage again
in the PROFILE output, and adjust the memory limit again if necessary to be lower than the peak memory usage.
At this point, you have a query that is memory-intensive, but Impala can optimize it efficiently so that the memory
usage is not exorbitant. You have set an artificial constraint through the MEM_LIMIT option so that the query would
normally fail with an out-of-memory error. But the automatic spill-to-disk feature means that the query should actually
succeed, at the expense of some extra disk I/O to read and write temporary work data.
Try the query again, and confirm that it succeeds. Examine the PROFILE output again. This time, look for lines of this
form:
- SpilledPartitions: N
If you see any such lines with N greater than 0, that indicates the query would have failed in Impala releases prior to
2.0, but now it succeeded because of the spill-to-disk feature. Examine the total time taken by the AGGREGATION_NODE
or other query fragments containing non-zero SpilledPartitions values. Compare the times to similar fragments
that did not spill, for example in the PROFILE output when the same query is run with a higher memory limit. This
gives you an idea of the performance penalty of the spill operation for a particular query with a particular memory
limit. If you make the memory limit just a little lower than the peak memory usage, the query only needs to write a
small amount of temporary data to disk. The lower you set the memory limit, the more temporary data is written and
the slower the query becomes.
Now repeat this procedure for actual queries used in your environment. Use the DISABLE_UNSAFE_SPILLS setting
to identify cases where queries used more memory than necessary due to lack of statistics on the relevant tables and
columns, and issue COMPUTE STATS where necessary.
When to use DISABLE_UNSAFE_SPILLS:
You might wonder, why not leave DISABLE_UNSAFE_SPILLS turned on all the time. Whether and how frequently to
use this option depends on your system environment and workload.
DISABLE_UNSAFE_SPILLS is suitable for an environment with ad hoc queries whose performance characteristics
and memory usage are not known in advance. It prevents “worst-case scenario” queries that use large amounts of
memory unnecessarily. Thus, you might turn this option on within a session while developing new SQL code, even
though it is turned off for existing applications.
Organizations where table and column statistics are generally up-to-date might leave this option turned on all the
time, again to avoid worst-case scenarios for untested queries or if a problem in the ETL pipeline results in a table with
no statistics. Turning on DISABLE_UNSAFE_SPILLS lets you “fail fast” in this case and immediately gather statistics
or tune the problematic queries.
Some organizations might leave this option turned off. For example, you might have tables large enough that the
COMPUTE STATS takes substantial time to run, making it impractical to re-run after loading new data. If you have
examined the EXPLAIN plans of your queries and know that they are operating efficiently, you might leave
DISABLE_UNSAFE_SPILLS turned off. In that case, you know that any queries that spill will not go overboard with
their memory consumption.
Turning off the spill-to-disk feature:
You might turn off the spill-to-disk feature if you are in an environment with constraints on disk space, or if you prefer
for queries that exceed the memory capacity in your cluster to “fail fast” so that you can tune and retry them.
To turn off this feature, set the following configuration options for each impalad daemon, either through the impalad
advanced configuration snippet in Cloudera Manager, or during impalad startup on each DataNode on systems not
managed by Cloudera Manager:
-enable_partitioned_aggregation=false
-enable_partitioned_hash_join=false
If you have the ability to customize such queries or the query generation logic that produces them, replace sequences
of repetitive expressions with single operators such as IN or BETWEEN that can represent multiple values or ranges.
For example, instead of a large number of OR clauses:
Note: If you are creating a partition for the first time and specifying its location, for maximum
efficiency, use a single ALTER TABLE statement including both the ADD PARTITION and
LOCATION clauses, rather than separate statements with ADD PARTITION and SET LOCATION
clauses.
• INSERT: When you insert data into a partitioned table, you identify the partitioning columns. One or more values
from each inserted row are not stored in data files, but instead determine the directory where that row value is
stored. You can also specify which partition to load a set of data into, with INSERT OVERWRITE statements; you
can replace the contents of a specific partition but you cannot append data to a specific partition.
By default, if an INSERT statement creates any new subdirectories underneath a partitioned table, those
subdirectories are assigned default HDFS permissions for the impala user. To make each subdirectory have the
same permissions as its parent directory in HDFS, specify the --insert_inherit_permissions startup option
for the impalad daemon.
• Although the syntax of the SELECT statement is the same whether or not the table is partitioned, the way queries
interact with partitioned tables can have a dramatic impact on performance and scalability. The mechanism that
lets queries skip certain partitions during a query is known as partition pruning; see Partition Pruning for Queries
on page 524 for details.
• In Impala 1.4 and later, there is a SHOW PARTITIONS statement that displays information about each partition
in a table. See SHOW Statement on page 322 for details.
When you specify some partition key columns in an INSERT statement, but leave out the values, Impala determines
which partition to insert. This technique is called dynamic partitioning:
The more key columns you specify in the PARTITION clause, the fewer columns you need in the SELECT list. The
trailing columns in the SELECT list are substituted in order for the partition key columns with no specified value.
For a report of the volume of data that was actually read and processed at each stage of the query, check the output
of the SUMMARY command immediately after running the query. For a more detailed analysis, look at the output of
the PROFILE command; it includes this same summary report near the start of the profile output.
If a view applies to a partitioned table, any partition pruning considers the clauses on both the original query and any
additional WHERE predicates in the query that refers to the view. Prior to Impala 1.4, only the WHERE clauses on the
original query from the CREATE VIEW statement were used for partition pruning.
In queries involving both analytic functions and partitioned tables, partition pruning only occurs for columns named
in the PARTITION BY clause of the analytic function call. For example, if an analytic function query has a clause such
as WHERE year=2016, the way to make the query prune all other YEAR partitions is to include PARTITION BY year
in the analytic function call; for example, OVER (PARTITION BY year,other_columns
other_analytic_clauses).
day, hour, and minute. If you have data with a geographic component, you might partition based on postal code
if you have many megabytes of data for each postal code, but if not, you might partition by some larger region
such as city, state, or country. state
[localhost:21000] > create table census (name string) partitioned by (year smallint);
[localhost:21000] > alter table census add partition (year=2012); -- Text format;
[localhost:21000] > alter table census add partition (year=2013); -- Text format switches
to Parquet before data loaded;
[localhost:21000] > alter table census partition (year=2013) set fileformat parquet;
At this point, the HDFS directory for year=2012 contains a text-format data file, while the HDFS directory for year=2013
contains a Parquet data file. As always, when loading non-trivial data, you would use INSERT ... SELECT or LOAD
DATA to import data in large batches, rather than INSERT ... VALUES which produces small files that are inefficient
for real-world queries.
For other file types that Impala cannot create natively, you can switch into Hive and issue the ALTER TABLE ...
SET FILEFORMAT statements and INSERT or LOAD DATA statements there. After switching back to Impala, issue a
REFRESH table_name statement so that Impala recognizes any partitions or new data added through Hive.
Managing Partitions
You can add, drop, set the expected file format, or set the HDFS location of the data files for individual partitions within
an Impala table. See ALTER TABLE Statement on page 215 for syntax details, and Setting Different File Formats for
Partitions on page 526 for tips on managing tables containing partitions with different file formats.
Note: If you are creating a partition for the first time and specifying its location, for maximum efficiency,
use a single ALTER TABLE statement including both the ADD PARTITION and LOCATION clauses,
rather than separate statements with ADD PARTITION and SET LOCATION clauses.
What happens to the data files when a partition is dropped depends on whether the partitioned table is designated
as internal or external. For an internal (managed) table, the data files are deleted. For example, if data in the partitioned
table is a copy of raw data files stored elsewhere, you might save disk space by dropping older partitions that are no
longer required for reporting, knowing that the original data is still available if needed later. For an external table, the
data files are left alone. For example, dropping a partition without deleting the associated files lets Impala consider a
smaller set of partitions, improving query efficiency and reducing overhead for DDL operations on the table; if the data
is needed again later, you can add the partition again. See Overview of Impala Tables on page 208 for details and
examples.
File Type Format Compression Codecs Impala Can CREATE? Impala Can INSERT?
Parquet Structured Snappy, gzip; currently Yes. Yes: CREATE TABLE, INSERT,
Snappy by default LOAD DATA, and query.
Text Unstructured LZO, gzip, bzip2, Yes. For CREATE TABLE with no Yes: CREATE TABLE, INSERT,
Snappy STORED AS clause, the default file LOAD DATA, and query. If LZO
format is uncompressed text, with compression is used, you must
values separated by ASCII 0x01 create the table and load data in
characters (typically represented Hive. If other kinds of compression
as Ctrl-A). are used, you must load data
through LOAD DATA, Hive, or
manually in HDFS.
Avro Structured Snappy, gzip, deflate, Yes, in Impala 1.4.0 and higher. No. Import data by using LOAD
bzip2 Before that, create the table using DATA on data files already in the
Hive. right format, or use INSERT in
Hive followed by REFRESH
table_name in Impala.
RCFile Structured Snappy, gzip, deflate, Yes. No. Import data by using LOAD
bzip2 DATA on data files already in the
right format, or use INSERT in
Hive followed by REFRESH
table_name in Impala.
SequenceFile Structured Snappy, gzip, deflate, Yes. No. Import data by using LOAD
bzip2 DATA on data files already in the
right format, or use INSERT in
Hive followed by REFRESH
table_name in Impala.
Impala can only query the file formats listed in the preceding table. In particular, Impala does not support the ORC file
format.
File Type Format Compression Codecs Impala Can CREATE? Impala Can INSERT?
Text Unstructured LZO, gzip, bzip2, Yes. For CREATE TABLE with no Yes: CREATE TABLE, INSERT,
Snappy STORED AS clause, the default file LOAD DATA, and query. If LZO
format is uncompressed text, with compression is used, you must
values separated by ASCII 0x01 create the table and load data in
characters (typically represented Hive. If other kinds of compression
as Ctrl-A). are used, you must load data
through LOAD DATA, Hive, or
manually in HDFS.
Note:
Impala supports bzip files created by the bzip2 command, but not bzip files with multiple streams
created by the pbzip2 command. Impala decodes only the data from the first part of such files,
leading to incomplete results.
The maximum size that Impala can accommodate for an individual bzip file is 1 GB (after
uncompression).
The data files created by any INSERT statements will use the Ctrl-A character (hex 01) as a separator between each
column value.
A common use case is to import existing text files into an Impala table. The syntax is more verbose; the significant part
is the FIELDS TERMINATED BY clause, which must be preceded by the ROW FORMAT DELIMITED clause. The statement
can end with a STORED AS TEXTFILE clause, but that clause is optional because text format tables are the default.
For example:
You can create tables with specific separator characters to import text files in familiar formats such as CSV, TSV, or
pipe-separated. You can also use these tables to produce output data files, by copying data into them through the
INSERT ... SELECT syntax and then extracting the data files from the Impala data directory.
In Impala 1.3.1 and higher, you can specify a delimiter character '\0' to use the ASCII 0 (nul) character for text tables:
Note:
Do not surround string values with quotation marks in text data files that you construct. If you need
to include the separator character inside a field value, for example to put a string value with a comma
inside a CSV-format data file, specify an escape character on the CREATE TABLE statement with the
ESCAPED BY clause, and insert that character immediately before any separator characters that need
escaping.
Issue a DESCRIBE FORMATTED table_name statement to see the details of how each table is represented internally
in Impala.
Complex type considerations: Although you can create tables in this file format using the complex types (ARRAY,
STRUCT, and MAP) available in CDH 5.5 / Impala 2.3 and higher, currently, Impala can query these types only in Parquet
tables.
• Otherwise, the file names are not significant. When you put files into an HDFS directory through ETL jobs, or point
Impala to an existing HDFS directory with the CREATE EXTERNAL TABLE statement, or move data files under
external control with the LOAD DATA statement, Impala preserves the original filenames.
Filenames for data produced through Impala INSERT statements are given unique names to avoid filename conflicts.
An INSERT ... SELECT statement produces one data file from each node that processes the SELECT part of the
statement. An INSERT ... VALUES statement produces a separate data file for each statement; because Impala is
more efficient querying a small number of huge files than a large number of tiny files, the INSERT ... VALUES syntax
is not recommended for loading a substantial volume of data. If you find yourself with a table that is inefficient due to
too many small data files, reorganize the data into a few large files by doing INSERT ... SELECT to transfer the data
to a new table.
Special values within text data files:
• Impala recognizes the literal strings inf for infinity and nan for “Not a Number”, for FLOAT and DOUBLE columns.
• Impala recognizes the literal string \N to represent NULL. When using Sqoop, specify the options
--null-non-string and --null-string to ensure all NULL values are represented correctly in the Sqoop
output files. By default, Sqoop writes NULL values using the string null, which causes a conversion error when
such rows are evaluated by Impala. (A workaround for existing tables and data files is to change the table properties
through ALTER TABLE name SET TBLPROPERTIES("serialization.null.format"="null").)
This can be a useful technique to see how Impala represents special values within a text-format data file. Use the
DESCRIBE FORMATTED statement to see the HDFS directory where the data files are stored, then use Linux commands
such as hdfs dfs -ls hdfs_directory and hdfs dfs -cat hdfs_file to display the contents of an
Impala-created text file.
To create a few rows in a text table for test purposes, you can use the INSERT ... VALUES syntax:
Note: Because Impala and the HDFS infrastructure are optimized for multi-megabyte files, avoid the
INSERT ... VALUES notation when you are inserting many rows. Each INSERT ... VALUES
statement produces a new tiny file, leading to fragmentation and reduced performance. When creating
any substantial volume of new data, use one of the bulk loading techniques such as LOAD DATA or
INSERT ... SELECT. Or, use an HBase table for single-row INSERT operations, because HBase
tables are not subject to the same fragmentation issues as tables stored on HDFS.
When you create a text file for use with an Impala text table, specify \N to represent a NULL value. For the differences
between NULL and empty strings, see NULL on page 187.
If a text file has fewer fields than the columns in the corresponding Impala table, all the corresponding columns are
set to NULL when the data in that file is read by an Impala query.
If a text file has more fields than the columns in the corresponding Impala table, the extra fields are ignored when the
data in that file is read by an Impala query.
You can also use manual HDFS operations such as hdfs dfs -put or hdfs dfs -cp to put data files in the data
directory for an Impala table. When you copy or move new data files into the HDFS directory for the Impala table, issue
a REFRESH table_name statement in impala-shell before issuing the next query against that table, to make Impala
recognize the newly added files.
Impala can work with LZO-compressed text files. LZO-compressed files are preferable to text files compressed by other
codecs, because LZO-compressed files are “splittable”, meaning that different portions of a file can be uncompressed
and processed independently by different nodes.
Impala does not currently support writing LZO-compressed text files.
Because Impala can query LZO-compressed files but currently cannot write them, you use Hive to do the initial CREATE
TABLE and load the data, then switch back to Impala to run queries. For instructions on setting up LZO compression
for Hive CREATE TABLE and INSERT statements, see the LZO page on the Hive wiki. Once you have created an LZO
text table, you can also manually add LZO-compressed text files to it, produced by the lzop command or similar
method.
Note:
The level of the impala-lzo package is closely tied to the version of Impala you use. Any time
you upgrade Impala, re-do the installation command for impala-lzo on each applicable machine
to make sure you have the appropriate version of that package.
3. For core-site.xml on the client and server (that is, in the configuration directories for both Impala and Hadoop),
append com.hadoop.compression.lzo.LzopCodec to the comma-separated list of codecs. For example:
<property>
<name>io.compression.codecs</name>
<value>org.apache.hadoop.io.compress.DefaultCodec,org.apache.hadoop.io.compress.GzipCodec,
org.apache.hadoop.io.compress.BZip2Codec,org.apache.hadoop.io.compress.DeflateCodec,
org.apache.hadoop.io.compress.SnappyCodec,com.hadoop.compression.lzo.LzopCodec</value>
</property>
Note:
If this is the first time you have edited the Hadoop core-site.xml file, note that the
/etc/hadoop/conf directory is typically a symbolic link, so the canonical core-site.xml
might reside in a different directory:
$ ls -l /etc/hadoop
total 8
lrwxrwxrwx. 1 root root 29 Feb 26 2013 conf ->
/etc/alternatives/hadoop-conf
lrwxrwxrwx. 1 root root 10 Feb 26 2013 conf.dist -> conf.empty
drwxr-xr-x. 2 root root 4096 Feb 26 2013 conf.empty
drwxr-xr-x. 2 root root 4096 Oct 28 15:46 conf.pseudo
STORED AS
INPUTFORMAT 'com.hadoop.mapred.DeprecatedLzoTextInputFormat'
OUTPUTFORMAT 'org.apache.hadoop.hive.ql.io.HiveIgnoreKeyTextOutputFormat'
Once you have created LZO-compressed text tables, you can convert data stored in other tables (regardless of file
format) by using the INSERT ... SELECT statement in Hive.
Files in an LZO-compressed table must use the .lzo extension. Examine the files in the HDFS data directory after doing
the INSERT in Hive, to make sure the files have the right extension. If the required settings are not in place, you end
up with regular uncompressed files, and Impala cannot access the table because it finds data files with the wrong
(uncompressed) format.
After loading data into an LZO-compressed text table, index the files so that they can be split. You index the files by
running a Java class, com.hadoop.compression.lzo.DistributedLzoIndexer, through the Linux command
line. This Java class is included in the hadoop-lzo package.
Run the indexer using a command like the following:
Note: If the path of the JAR file in the preceding example is not recognized, do a find command to
locate hadoop-lzo-*-gplextras.jar and use that path.
Indexed files have the same name as the file they index, with the .index extension. If the data files are not indexed,
Impala queries still work, but the queries read the data from remote DataNodes, which is very inefficient.
Once the LZO-compressed tables are created, and data is loaded and indexed, you can query them through Impala.
As always, the first time you start impala-shell after creating a table in Hive, issue an INVALIDATE METADATA
statement so that Impala recognizes the new table. (In Impala 1.2 and higher, you only have to run INVALIDATE
METADATA on one node, rather than on all the Impala nodes.)
File Type Format Compression Codecs Impala Can CREATE? Impala Can INSERT?
Parquet Structured Snappy, gzip; currently Yes. Yes: CREATE TABLE, INSERT,
Snappy by default LOAD DATA, and query.
Or, to clone the column names and data types of an existing table:
In Impala 1.4.0 and higher, you can derive column definitions from a raw Parquet data file, even without an existing
Impala table. For example, you can create an external table pointing to an HDFS directory, and base the column
definitions on one of the files in that directory:
STORED AS PARQUET
LOCATION '/user/etl/destination';
Or, you can refer to an existing data file and create a new empty table with suitable column definitions. Then you can
use INSERT to create new data files or LOAD DATA to transfer existing data files into the new table.
The default properties of the newly created table are the same as for any other CREATE TABLE statement. For example,
the default file format is text; if you want the new table to use the Parquet file format, include the STORED AS PARQUET
file also.
In this example, the new table is partitioned by year, month, and day. These partition key columns are not part of the
data file, so you specify them in the CREATE TABLE statement:
See CREATE TABLE Statement on page 235 for more details about the CREATE TABLE LIKE PARQUET syntax.
Once you have created a table, to insert data into that table, use a command similar to the following, again with your
own table names:
If the Parquet table has a different number of columns or different column names than the other table, specify the
names of columns from the other table rather than * in the SELECT statement.
• If column statistics are available for all partition key columns in the source table mentioned in the INSERT ...
SELECT query, Impala chooses whether to use the [SHUFFLE] or [NOSHUFFLE] technique based on the estimated
number of distinct values in those columns and the number of nodes involved in the INSERT operation. In this
case, you might need the [SHUFFLE] or the [NOSHUFFLE] hint to override the execution plan selected by Impala.
Any INSERT statement for a Parquet table requires enough free space in the HDFS filesystem to write one block.
Because Parquet data files use a block size of 1 GB by default, an INSERT might fail (even for a very small amount of
data) if your HDFS is running low on space.
Avoid the INSERT...VALUES syntax for Parquet tables, because INSERT...VALUES produces a separate tiny data
file for each INSERT...VALUES statement, and the strength of Parquet is in its handling of data (compressing,
parallelizing, and so on) in large chunks.
If you have one or more Parquet data files produced outside of Impala, you can quickly make the data queryable
through Impala by one of the following methods:
• The LOAD DATA statement moves a single data file or a directory full of data files into the data directory for an
Impala table. It does no validation or conversion of the data. The original data files must be somewhere in HDFS,
not the local filesystem.
• The CREATE TABLE statement with the LOCATION clause creates a table where the data continues to reside
outside the Impala data directory. The original data files must be somewhere in HDFS, not the local filesystem.
For extra safety, if the data is intended to be long-lived and reused by other applications, you can use the CREATE
EXTERNAL TABLE syntax so that the data files are not deleted by an Impala DROP TABLE statement.
• If the Parquet table already exists, you can copy Parquet data files directly into it, then use the REFRESH statement
to make Impala recognize the newly added data. Remember to preserve the block size of the Parquet data files
by using the hadoop distcp -pb command rather than a -put or -cp operation on the Parquet files. See
Example of Copying Parquet Data Files on page 540 for an example of this kind of operation.
Note:
Currently, Impala always decodes the column data in Parquet files based on the ordinal position of
the columns, not by looking up the position of each column based on its name. Parquet files produced
outside of Impala must write column data in the same order as the columns are declared in the Impala
table. Any optional columns that are omitted from the data files must be the rightmost columns in
the Impala table definition.
If you created compressed Parquet files through some tool other than Impala, make sure that any
compression codecs are supported in Parquet by Impala. For example, Impala does not currently
support LZO compression in Parquet files. Also doublecheck that you used any recommended
compatibility settings in the other tool, such as spark.sql.parquet.binaryAsString when
writing Parquet files through Spark.
Recent versions of Sqoop can produce Parquet output files using the --as-parquetfile option.
If you use Sqoop to convert RDBMS data to Parquet, be careful with interpreting any resulting values from DATE,
DATETIME, or TIMESTAMP columns. The underlying values are represented as the Parquet INT64 type, which is
represented as BIGINT in the Impala table. The Parquet values represent the time in milliseconds, while Impala
interprets BIGINT as the time in seconds. Therefore, if you have a BIGINT column in a Parquet table that was imported
this way from Sqoop, divide the values by 1000 when interpreting as the TIMESTAMP type.
If the data exists outside Impala and is in some other format, combine both of the preceding techniques. First, use a
LOAD DATA or CREATE EXTERNAL TABLE ... LOCATION statement to bring the data into an Impala table that
uses the appropriate file format. Then, use an INSERT...SELECT statement to copy the data to the Parquet table,
converting to Parquet format as part of the process.
Loading data into Parquet tables is a memory-intensive operation, because the incoming data is buffered until it reaches
one data block in size, then that chunk of data is organized and compressed in memory before being written out. The
memory consumption can be larger when inserting data into partitioned Parquet tables, because a separate data file
is written for each combination of partition key column values, potentially requiring several large chunks to be
manipulated in memory at once.
When inserting into a partitioned Parquet table, Impala redistributes the data among the nodes to reduce memory
consumption. You might still need to temporarily increase the memory dedicated to Impala during the insert operation,
or break up the load operation into several INSERT statements, or both.
Note: All the preceding techniques assume that the data you are loading matches the structure of
the destination table, including column order, column names, and partition layout. To transform or
reorganize the data, start by loading the data into a Parquet table that matches the underlying structure
of the data, then use one of the table-copying techniques such as CREATE TABLE AS SELECT or
INSERT ... SELECT to reorder or rename columns, divide the data among multiple partitions, and
so on. For example to take a single comprehensive Parquet data file and load it into a partitioned
table, you would use an INSERT ... SELECT statement with dynamic partitioning to let Impala
create separate data files with the appropriate partition values; for an example, see INSERT Statement
on page 263.
The query processes only 2 columns out of a large number of total columns. If the table is partitioned by the STATE
column, it is even more efficient because the query only has to read and decode 1 column from each data file, and it
can read only the data files in the partition directory for the state 'CA', skipping the data files for all the other states,
which will be physically located in other directories.
The following is a relatively inefficient query for a Parquet table:
Impala would have to read the entire contents of each large data file, and decompress the contents of each column
for each row group, negating the I/O optimizations of the column-oriented format. This query might still be faster for
a Parquet table than a table with some other file format, but it does not take advantage of the unique strengths of
Parquet data files.
Impala can optimize queries on Parquet tables, especially join queries, better when statistics are available for all the
tables. Issue the COMPUTE STATS statement for each table after substantial amounts of data are loaded into or
appended to it. See COMPUTE STATS Statement on page 226 for details.
to partition the data, try to find a granularity where each partition contains 256 MB or more of data, rather than
creating a large number of smaller files split among many partitions.
Inserting into a partitioned Parquet table can be a resource-intensive operation, because each Impala node could
potentially be writing a separate data file to HDFS for each combination of different values for the partition key columns.
The large number of simultaneous open files could exceed the HDFS “transceivers” limit. To avoid exceeding this limit,
consider the following techniques:
• Load different subsets of data using separate INSERT statements with specific values for the PARTITION clause,
such as PARTITION (year=2010).
• Increase the “transceivers” value for HDFS, sometimes spelled “xcievers” (sic). The property value in the
hdfs-site.xml configuration file is dfs.datanode.max.transfer.threads. For example, if you were loading
12 years of data partitioned by year, month, and day, even a value of 4096 might not be high enough. This blog
post explores the considerations for setting this value higher or lower, using HBase examples for illustration.
• Use the COMPUTE STATS statement to collect column statistics on the source table from which data is being
copied, so that the Impala query can estimate the number of different values in the partition key columns and
distribute the work accordingly.
Because Parquet data files are typically large, each directory will have a different number of data files and the row
groups will be arranged differently.
At the same time, the less aggressive the compression, the faster the data can be decompressed. In this case using a
table with a billion rows, a query that evaluates all the values for a particular column runs faster with no compression
than with Snappy compression, and faster with Snappy compression than with Gzip compression. Query performance
depends on several other factors, so as always, run your own benchmarks with your own data to determine the ideal
tradeoff between data size, CPU efficiency, and speed of insert and query operations.
queries demonstrate that the new table now contains 3 billion rows featuring a variety of compression codecs for the
data files.
First, we create the table in Impala so that there is a destination directory in HDFS to put the data files:
Then in the shell, we copy the relevant data files into the data directory for this new table. Rather than using hdfs
dfs -cp as with typical files, we use hadoop distcp -pb to ensure that the special block size of the Parquet data
files is preserved.
Back in the impala-shell interpreter, we use the REFRESH statement to alert the Impala server to the new data
files for this table, then we can run queries demonstrating that the data files represent 3 billion rows, and the values
for one of the numeric columns match what was in the original smaller tables:
If you are running a level of Impala that is older than 1.1.1, do the metadata update through Hive:
Impala 1.1.1 and higher can reuse Parquet data files created by Hive, without any action required.
Impala supports the scalar data types that you can encode in a Parquet data file, but not composite or nested types
such as maps or arrays. In Impala 2.2.0 / CDH 5.4.0 and higher, Impala can query Parquet data files that include
composite or nested types, as long as the query only refers to columns with scalar types.
If you copy Parquet data files between nodes, or even between different directories on the same node, make sure to
preserve the block size by using the command hadoop distcp -pb. To verify that the block size was preserved, issue
the command hdfs fsck -blocks HDFS_path_of_impala_table_dir and check that the average block size is
at or near 256 MB (or whatever other size is defined by the PARQUET_FILE_SIZE query option).. (The hadoop distcp
operation typically leaves some directories behind, with names matching _distcp_logs_*, that you can delete from
the destination directory afterward.) Issue the command hadoop distcp for details about distcp command syntax.
Impala can query Parquet files that use the PLAIN, PLAIN_DICTIONARY, BIT_PACKED, and RLE encodings. Currently,
Impala does not support RLE_DICTIONARY encoding. When creating files outside of Impala for use by Impala, make
sure to use one of the supported encodings. In particular, for MapReduce jobs, parquet.writer.version must not
be defined (especially as PARQUET_2_0) for writing the configurations of Parquet MR jobs. Use the default version (or
format). The default format, 1.0, includes some enhancements that are compatible with older versions. Data using the
2.0 format might not be consumable by Impala, due to use of the RLE_DICTIONARY encoding.
To examine the internal structure and data of Parquet files, you can use the parquet-tools command that comes
with CDH. Make sure this command is in your $PATH. (Typically, it is symlinked from /usr/bin; sometimes, depending
on your installation setup, you might need to locate it under a CDH-specific bin directory.) The arguments to this
command let you perform operations such as:
• cat: Print a file's contents to standard out. Use the -j option to output JSON.
• head: Print the first few records of a file to standard output.
• schema: Print the Parquet schema for the file.
• meta: Print the file footer metadata, including key-value properties (like Avro schema), compression ratios,
encodings, compression used, and row group information.
• dump: Print all data and metadata.
Use parquet-tools -h to see usage information for all the arguments. (The -j option for displaying the data in
JSON format is only available in CDH 5.5 and higher.) Here are some examples showing parquet-tools usage:
$ # Be careful doing this for a big file! Use parquet-tools head to be safe.
$ parquet-tools cat sample.parq
year = 1992
month = 1
day = 2
dayofweek = 4
dep_time = 748
crs_dep_time = 750
arr_time = 851
crs_arr_time = 846
carrier = US
flight_num = 53
actual_elapsed_time = 63
crs_elapsed_time = 56
arrdelay = 5
depdelay = -2
origin = CMH
dest = IND
distance = 182
cancelled = 0
diverted = 0
year = 1992
month = 1
day = 3
...
year = 1992
month = 1
day = 3
...
-------------------------------------------------------------------
year: INT32 SNAPPY DO:4 FPO:35 SZ:10103/49723/4.92 VC:20636601
ENC:PLAIN_DICTIONARY,RLE,PLAIN
month: INT32 SNAPPY DO:10147 FPO:10210 SZ:11380/35732/3.14 VC:20636601
ENC:PLAIN_DICTIONARY,RLE,PLAIN
day: INT32 SNAPPY DO:21572 FPO:21714 SZ:3071658/9868452/3.21 VC:20636601
ENC:PLAIN_DICTIONARY,RLE,PLAIN
dayofweek: INT32 SNAPPY DO:3093276 FPO:3093319 SZ:2274375/5941876/2.61
VC:20636601 ENC:PLAIN_DICTIONARY,RLE,PLAIN
dep_time: INT32 SNAPPY DO:5367705 FPO:5373967 SZ:28281281/28573175/1.01
VC:20636601 ENC:PLAIN_DICTIONARY,RLE,PLAIN
crs_dep_time: INT32 SNAPPY DO:33649039 FPO:33654262 SZ:10220839/11574964/1.13
VC:20636601 ENC:PLAIN_DICTIONARY,RLE,PLAIN
arr_time: INT32 SNAPPY DO:43869935 FPO:43876489 SZ:28562410/28797767/1.01
VC:20636601 ENC:PLAIN_DICTIONARY,RLE,PLAIN
crs_arr_time: INT32 SNAPPY DO:72432398 FPO:72438151 SZ:10908972/12164626/1.12
VC:20636601 ENC:PLAIN_DICTIONARY,RLE,PLAIN
carrier: BINARY SNAPPY DO:83341427 FPO:83341558 SZ:114916/128611/1.12
VC:20636601 ENC:PLAIN_DICTIONARY,RLE,PLAIN
flight_num: INT32 SNAPPY DO:83456393 FPO:83488603 SZ:10216514/11474301/1.12
VC:20636601 ENC:PLAIN_DICTIONARY,RLE,PLAIN
...
Note:
Impala INSERT statements write Parquet data files using an HDFS block size that matches the data
file size, to ensure that each data file is represented by a single HDFS block, and the entire file can be
processed on a single node without requiring any remote reads.
If you create Parquet data files outside of Impala, such as through a MapReduce or Pig job, ensure
that the HDFS block size is greater than or equal to the file size, so that the “one file per block”
relationship is maintained. Set the dfs.block.size or the dfs.blocksize property large enough
that each file fits within a single HDFS block, even if that size is larger than the normal HDFS block size.
If the block size is reset to a lower value during a file copy, you will see lower performance for queries
involving those files, and the PROFILE statement will reveal that some I/O is being done suboptimally,
through remote reads. See Example of Copying Parquet Data Files on page 540 for an example showing
how to preserve the block size when copying Parquet data files.
When Impala retrieves or tests the data for a particular column, it opens all the data files, but only reads the portion
of each file containing the values for that column. The column values are stored consecutively, minimizing the I/O
required to process the values within a single column. If other columns are named in the SELECT list or WHERE clauses,
the data for all columns in the same row is available within that same data file.
If an INSERT statement brings in less than one Parquet block's worth of data, the resulting data file is smaller than
ideal. Thus, if you do split up an ETL job to use multiple INSERT statements, try to keep the volume of data for each
INSERT statement to approximately 256 MB, or a multiple of 256 MB.
Here are techniques to help you produce large data files in Parquet INSERT operations, and to compact existing
too-small data files:
• When inserting into a partitioned Parquet table, use statically partitioned INSERT statements where the partition
key values are specified as constant values. Ideally, use a separate INSERT statement for each partition.
• You might set the NUM_NODES option to 1 briefly, during INSERT or CREATE TABLE AS SELECT statements.
Normally, those statements produce one or more data files per data node. If the write operation involves small
amounts of data, a Parquet table, and/or a partitioned table, the default behavior could produce many small files
when intuitively you might expect only a single output file. SET NUM_NODES=1 turns off the “distributed” aspect
of the write operation, making it more likely to produce only one or a few data files.
• Be prepared to reduce the number of partition key columns from what you are used to with traditional analytic
database systems.
• Do not expect Impala-written Parquet files to fill up the entire Parquet block size. Impala estimates on the
conservative side when figuring out how much data to write to each Parquet file. Typically, the of uncompressed
data in memory is substantially reduced on disk by the compression and encoding techniques in the Parquet file
format. The final data file size varies depending on the compressibility of the data. Therefore, it is not an indication
of a problem if 256 MB of text data is turned into 2 Parquet data files, each less than 256 MB.
• If you accidentally end up with a table with many small data files, consider using one or more of the preceding
techniques and copying all the data into a new Parquet table, either through CREATE TABLE AS SELECT or
INSERT ... SELECT statements.
To avoid rewriting queries to change table names, you can adopt a convention of always running important queries
against a view. Changing the view definition immediately switches any subsequent queries to use the new underlying
tables:
• If you use ALTER TABLE ... REPLACE COLUMNS to define additional columns at the end, when the original
data files are used in a query, these final columns are considered to be all NULL values.
• If you use ALTER TABLE ... REPLACE COLUMNS to define fewer columns than before, when the original data
files are used in a query, the unused columns still present in the data file are ignored.
• Parquet represents the TINYINT, SMALLINT, and INT types the same internally, all stored in 32-bit integers.
– That means it is easy to promote a TINYINT column to SMALLINT or INT, or a SMALLINT column to INT.
The numbers are represented exactly the same in the data file, and the columns being promoted would not
contain any out-of-range values.
– If you change any of these column types to a smaller type, any values that are out-of-range for the new type
are returned incorrectly, typically as negative numbers.
– You cannot change a TINYINT, SMALLINT, or INT column to BIGINT, or the other way around. Although
the ALTER TABLE succeeds, any attempt to query those columns results in conversion errors.
– Any other type conversion for columns produces a conversion error during queries. For example, INT to
STRING, FLOAT to DOUBLE, TIMESTAMP to STRING, DECIMAL(9,0) to DECIMAL(5,2), and so on.
Primitive types:
Logical types:
Complex types:
For the complex types (ARRAY, MAP, and STRUCT) available in CDH 5.5 / Impala 2.3 and higher, Impala only supports
queries against those types in Parquet tables.
File Type Format Compression Codecs Impala Can CREATE? Impala Can INSERT?
Avro Structured Snappy, gzip, deflate, Yes, in Impala 1.4.0 and higher. No. Import data by using LOAD
bzip2 Before that, create the table using DATA on data files already in the
Hive. right format, or use INSERT in
Hive followed by REFRESH
table_name in Impala.
Note:
Currently, Avro tables cannot contain TIMESTAMP columns. If you need to store date and time values
in Avro tables, as a workaround you can use a STRING representation of the values, convert the values
to BIGINT with the UNIX_TIMESTAMP() function, or create separate numeric columns for individual
date and time fields using the EXTRACT() function.
The following examples demonstrate creating an Avro table in Impala, using either an inline column specification or
one taken from a JSON file stored in HDFS:
Each field of the record becomes a column of the table. Note that any other information, such as the record name, is
ignored.
Note: For nullable Avro columns, make sure to put the "null" entry before the actual type name.
In Impala, all columns are nullable; Impala currently does not have a NOT NULL clause. Any non-nullable
property is only enforced on the Avro side.
Most column types map directly from Avro to Impala under the same names. These are the exceptions and special
cases to consider:
• The DECIMAL type is defined in Avro as a BYTE type with the logicalType property set to "decimal" and a
specified precision and scale. Use DECIMAL in Avro tables only under CDH 5. The infrastructure and components
under CDH 4 do not have reliable DECIMAL support.
• The Avro long type maps to BIGINT in Impala.
If you create the table through Hive, switch back to impala-shell and issue an INVALIDATE METADATA table_name
statement. Then you can run queries for that table through impala-shell.
In rare instances, a mismatch could occur between the Avro schema and the column definitions in the metastore
database. In CDH 5.5 / Impala 2.3 and higher, Impala checks for such inconsistencies during a CREATE TABLE statement
and each time it loads the metadata for a table (for example, after INVALIDATE METADATA). Impala uses the following
rules to determine how to treat mismatching columns, a process known as schema reconciliation:
• If there is a mismatch in the number of columns, Impala uses the column definitions from the Avro schema.
• If there is a mismatch in column name or type, Impala uses the column definition from the Avro schema. Because
a CHAR or VARCHAR column in Impala maps to an Avro STRING, this case is not considered a mismatch and the
column is preserved as CHAR or VARCHAR in the reconciled schema.
• An Impala TIMESTAMP column definition maps to an Avro STRING and is presented as a STRING in the reconciled
schema, because Avro has no binary TIMESTAMP representation. As a result, no Avro table can have a TIMESTAMP
column; this restriction is the same as in earlier CDH and Impala releases.
Complex type considerations: Although you can create tables in this file format using the complex types (ARRAY,
STRUCT, and MAP) available in CDH 5.5 / Impala 2.3 and higher, currently, Impala can query these types only in Parquet
tables.
tblproperties ('avro.schema.url'='hdfs//your-name-node:port/path/to/schema.json');
To copy data from another table, issue any INSERT statements through Hive. For information about loading data into
Avro tables through Hive, see Avro page on the Hive wiki.
If you already have data files in Avro format, you can also issue LOAD DATA in either Impala or Hive. Impala can move
existing Avro data files into an Avro table, it just cannot create new Avro data files.
When Avro data files or columns are not consulted during a query, Impala does not check for consistency. Thus, if you
issue SELECT c1, c2 FROM t1, Impala does not return any error if the column c3 changed in an incompatible way.
If a query retrieves data from some partitions but not others, Impala does not check the data files for the unused
partitions.
In the Hive DDL statements, you can specify an avro.schema.literal table property (if the schema definition is
short) or an avro.schema.url property (if the schema definition is long, or to allow convenient editing for the
definition).
For example, running the following SQL code in the Hive shell creates a table using the Avro file format and puts some
sample data into it:
"name": "my_record",
"fields": [
{"name": "a", "type": "int"},
{"name": "b", "type": "string"}
]}');
Once the Avro table is created and contains data, you can query it through the impala-shell command:
Now in the Hive shell, you change the type of a column and add a new column with a default value:
-- Promote column "a" from INT to FLOAT (no need to update Avro schema)
ALTER TABLE avro_table CHANGE A A FLOAT;
Once again in impala-shell, you can query the Avro table based on its latest schema definition. Because the table
metadata was changed outside of Impala, you issue a REFRESH statement first so that Impala has up-to-date metadata
for the table.
Primitive Types
---------------
STRING -> STRING
INT -> INT
BOOLEAN -> BOOLEAN
LONG -> BIGINT
FLOAT -> FLOAT
DOUBLE -> DOUBLE
Logical Types
-------------
BYTES + logicalType = "decimal" -> DECIMAL
------------------------------------
RECORD, MAP, ARRAY, UNION, ENUM, FIXED, NULL
The Avro specification allows string values up to 2**64 bytes in length. Impala queries for Avro tables use 32-bit integers
to hold string lengths. In CDH 5.7 / Impala 2.5 and higher, Impala truncates CHAR and VARCHAR values in Avro tables
to (2**31)-1 bytes. If a query encounters a STRING value longer than (2**31)-1 bytes in an Avro table, the query fails.
In earlier releases, encountering such long values in an Avro table could cause a crash.
File Type Format Compression Codecs Impala Can CREATE? Impala Can INSERT?
RCFile Structured Snappy, gzip, deflate, Yes. No. Import data by using LOAD
bzip2 DATA on data files already in the
right format, or use INSERT in
Hive followed by REFRESH
table_name in Impala.
Because Impala can query some kinds of tables that it cannot currently write to, after creating tables of certain file
formats, you might use the Hive shell to load the data. See How Impala Works with Hadoop File Formats on page 527
for details. After loading data into a table through Hive or other mechanism outside of Impala, issue a REFRESH
table_name statement the next time you connect to the Impala node, before querying the table, to make Impala
recognize the new data.
Important: See Known Issues and Workarounds in Impala on page 635 for potential compatibility
issues with RCFile tables created in Hive 0.12, due to a change in the default RCFile SerDe for Hive.
For example, here is how you might create some RCFile tables in Impala (by specifying the columns explicitly, or cloning
the structure of another table), load data through Hive, and query them through Impala:
$ impala-shell -i localhost
[localhost:21000] > create table rcfile_table (x int) stored as rcfile;
[localhost:21000] > create table rcfile_clone like some_other_table stored as rcfile;
[localhost:21000] > quit;
$ hive
hive> insert into table rcfile_table select x from some_other_table;
3 Rows loaded to rcfile_table
Time taken: 19.015 seconds
hive> quit;
$ impala-shell -i localhost
[localhost:21000] > select * from rcfile_table;
Returned 0 row(s) in 0.23s
[localhost:21000] > -- Make Impala recognize the data loaded through Hive;
[localhost:21000] > refresh rcfile_table;
[localhost:21000] > select * from rcfile_table;
+---+
| x |
+---+
| 1 |
| 2 |
| 3 |
+---+
Returned 3 row(s) in 0.23s
Complex type considerations: Although you can create tables in this file format using the complex types (ARRAY,
STRUCT, and MAP) available in CDH 5.5 / Impala 2.3 and higher, currently, Impala can query these types only in Parquet
tables.
If you are converting partitioned tables, you must complete additional steps. In such a case, specify additional settings
similar to the following:
Remember that Hive does not require that you specify a source format for it. Consider the case of converting a table
with two partition columns called year and month to a Snappy compressed RCFile. Combining the components outlined
previously to complete this table conversion, you would specify settings similar to the following:
hive> CREATE TABLE tbl_rc (int_col INT, string_col STRING) STORED AS RCFILE;
hive> SET hive.exec.compress.output=true;
hive> SET mapred.max.split.size=256000000;
hive> SET mapred.output.compression.type=BLOCK;
hive> SET mapred.output.compression.codec=org.apache.hadoop.io.compress.SnappyCodec;
hive> SET hive.exec.dynamic.partition.mode=nonstrict;
hive> SET hive.exec.dynamic.partition=true;
hive> INSERT OVERWRITE TABLE tbl_rc SELECT * FROM tbl;
To complete a similar process for a table that includes partitions, you would specify settings similar to the following:
hive> CREATE TABLE tbl_rc (int_col INT, string_col STRING) PARTITIONED BY (year INT)
STORED AS RCFILE;
hive> SET hive.exec.compress.output=true;
hive> SET mapred.max.split.size=256000000;
hive> SET mapred.output.compression.type=BLOCK;
hive> SET mapred.output.compression.codec=org.apache.hadoop.io.compress.SnappyCodec;
hive> SET hive.exec.dynamic.partition.mode=nonstrict;
hive> SET hive.exec.dynamic.partition=true;
hive> INSERT OVERWRITE TABLE tbl_rc PARTITION(year) SELECT * FROM tbl;
Note:
The compression type is specified in the following command:
SET
mapred.output.compression.codec=org.apache.hadoop.io.compress.SnappyCodec;
File Type Format Compression Codecs Impala Can CREATE? Impala Can INSERT?
SequenceFile Structured Snappy, gzip, deflate, Yes. No. Import data by using LOAD
bzip2 DATA on data files already in the
right format, or use INSERT in
Hive followed by REFRESH
table_name in Impala.
Because Impala can query some kinds of tables that it cannot currently write to, after creating tables of certain file
formats, you might use the Hive shell to load the data. See How Impala Works with Hadoop File Formats on page 527
for details. After loading data into a table through Hive or other mechanism outside of Impala, issue a REFRESH
table_name statement the next time you connect to the Impala node, before querying the table, to make Impala
recognize the new data.
For example, here is how you might create some SequenceFile tables in Impala (by specifying the columns explicitly,
or cloning the structure of another table), load data through Hive, and query them through Impala:
$ impala-shell -i localhost
[localhost:21000] > create table seqfile_table (x int) stored as sequencefile;
[localhost:21000] > create table seqfile_clone like some_other_table stored as
sequencefile;
[localhost:21000] > quit;
$ hive
hive> insert into table seqfile_table select x from some_other_table;
3 Rows loaded to seqfile_table
Time taken: 19.047 seconds
hive> quit;
$ impala-shell -i localhost
[localhost:21000] > select * from seqfile_table;
Returned 0 row(s) in 0.23s
[localhost:21000] > -- Make Impala recognize the data loaded through Hive;
[localhost:21000] > refresh seqfile_table;
[localhost:21000] > select * from seqfile_table;
+---+
| x |
+---+
| 1 |
| 2 |
| 3 |
+---+
Returned 3 row(s) in 0.23s
Complex type considerations: Although you can create tables in this file format using the complex types (ARRAY,
STRUCT, and MAP) available in CDH 5.5 / Impala 2.3 and higher, currently, Impala can query these types only in Parquet
tables.
If you are converting partitioned tables, you must complete additional steps. In such a case, specify additional settings
similar to the following:
Remember that Hive does not require that you specify a source format for it. Consider the case of converting a table
with two partition columns called year and month to a Snappy compressed SequenceFile. Combining the components
outlined previously to complete this table conversion, you would specify settings similar to the following:
hive> create table TBL_SEQ (int_col int, string_col string) STORED AS SEQUENCEFILE;
hive> SET hive.exec.compress.output=true;
hive> SET mapred.max.split.size=256000000;
hive> SET mapred.output.compression.type=BLOCK;
hive> SET mapred.output.compression.codec=org.apache.hadoop.io.compress.SnappyCodec;
hive> SET hive.exec.dynamic.partition.mode=nonstrict;
hive> SET hive.exec.dynamic.partition=true;
hive> INSERT OVERWRITE TABLE tbl_seq SELECT * FROM tbl;
To complete a similar process for a table that includes partitions, you would specify settings similar to the following:
hive> CREATE TABLE tbl_seq (int_col INT, string_col STRING) PARTITIONED BY (year INT)
STORED AS SEQUENCEFILE;
hive> SET hive.exec.compress.output=true;
hive> SET mapred.max.split.size=256000000;
hive> SET mapred.output.compression.type=BLOCK;
hive> SET mapred.output.compression.codec=org.apache.hadoop.io.compress.SnappyCodec;
hive> SET hive.exec.dynamic.partition.mode=nonstrict;
hive> SET hive.exec.dynamic.partition=true;
hive> INSERT OVERWRITE TABLE tbl_seq PARTITION(year) SELECT * FROM tbl;
Note:
The compression type is specified in the following command:
SET
mapred.output.compression.codec=org.apache.hadoop.io.compress.SnappyCodec;
<property>
<name>hbase.client.retries.number</name>
<value>3</value>
</property>
<property>
<name>hbase.rpc.timeout</name>
<value>3000</value>
</property>
Currently, Cloudera Manager does not have an Impala-only override for HBase settings, so any HBase configuration
change you make through Cloudera Manager would take affect for all HBase applications. Therefore, this change is
not recommended on systems managed by Cloudera Manager.
only applies to a single row, HBase will locate and return that row. Conversely, if a non-key predicate is used, even if
it only applies to a single row, HBase must still scan the entire table to find the correct result.
Interpreting EXPLAIN Output for HBase Queries
For example, here are some queries against the following Impala table, which is mapped to an HBase table. The
examples show excerpts from the output of the EXPLAIN statement, demonstrating what things to look for to indicate
an efficient or inefficient query against an HBase table.
The first column (cust_id) was specified as the key column in the CREATE EXTERNAL TABLE statement; for
performance, it is important to declare this column as STRING. Other columns, such as BIRTH_YEAR and
NEVER_LOGGED_ON, are also declared as STRING, rather than their “natural” types of INT or BOOLEAN, because Impala
can optimize those types more effectively in HBase tables. For comparison, we leave one column, YEAR_REGISTERED,
as INT to show that filtering on this column is inefficient.
describe hbase_table;
Query: describe hbase_table
+-----------------------+--------+---------+
| name | type | comment |
+-----------------------+--------+---------+
| cust_id | string | |
| birth_year | string | |
| never_logged_on | string | |
| private_email_address | string | |
| year_registered | int | |
+-----------------------+--------+---------+
The best case for performance involves a single row lookup using an equality comparison on the column defined as
the row key:
Another type of efficient query involves a range lookup on the row key column, using SQL operators such as greater
than (or equal), less than (or equal), or BETWEEN. This example also includes an equality test on a non-key column;
because that column is a STRING, Impala can let HBase perform that test, indicated by the hbase filters: line in
the EXPLAIN output. Doing the filtering within HBase is more efficient than transmitting all the data to Impala and
doing the filtering on the Impala side.
explain select count(*) from hbase_table where cust_id between 'a' and 'b'
and never_logged_on = 'true';
+------------------------------------------------------------------------------------+
| Explain String |
+------------------------------------------------------------------------------------+
...
| 01:AGGREGATE |
| | output: count(*) |
| | |
The query is less efficient if Impala has to evaluate any of the predicates, because Impala must scan the entire HBase
table. Impala can only push down predicates to HBase for columns declared as STRING. This example tests a column
declared as INT, and the predicates: line in the EXPLAIN output indicates that the test is performed after the data
is transmitted to Impala.
| 01:AGGREGATE |
| | output: count(*) |
| | |
| 00:SCAN HBASE [hbase.hbase_table] |
| predicates: year_registered = 2010 |
+------------------------------------------------------------------------------------+
The same inefficiency applies if the key column is compared to any non-constant value. Here, even though the key
column is a STRING, and is tested using an equality operator, Impala must scan the entire HBase table because the
key column is compared to another column value rather than a constant.
| 01:AGGREGATE |
| | output: count(*) |
| | |
| 00:SCAN HBASE [hbase.hbase_table] |
| predicates: cust_id = private_email_address |
+------------------------------------------------------------------------------------+
Currently, tests on the row key using OR or IN clauses are not optimized into direct lookups either. Such limitations
might be lifted in the future, so always check the EXPLAIN output to be sure whether a particular SQL construct results
in an efficient query or not for HBase tables.
| 01:AGGREGATE
|
| | output: count(*)
|
| |
|
| 00:SCAN HBASE [hbase.hbase_table]
|
| predicates: cust_id = 'some_user@example.com' OR cust_id = 'other_user@example.com'
|
+----------------------------------------------------------------------------------------+
| Explain String |
+------------------------------------------------------------------------------------+
...
| 01:AGGREGATE |
| | output: count(*) |
| | |
| 00:SCAN HBASE [hbase.hbase_table] |
| predicates: cust_id IN ('some_user@example.com', 'other_user@example.com') |
+------------------------------------------------------------------------------------+
Either rewrite into separate queries for each value and combine the results in the application, or combine the single-row
queries using UNION ALL:
explain
select count(*) from hbase_table where cust_id = 'some_user@example.com'
union all
select count(*) from hbase_table where cust_id = 'other_user@example.com';
+------------------------------------------------------------------------------------+
| Explain String |
+------------------------------------------------------------------------------------+
...
| | 04:AGGREGATE |
| | | output: count(*) |
| | | |
| | 03:SCAN HBASE [hbase.hbase_table] |
| | start key: other_user@example.com |
| | stop key: other_user@example.com\0 |
| | |
| 10:MERGE |
...
| 02:AGGREGATE |
| | output: count(*) |
| | |
| 01:SCAN HBASE [hbase.hbase_table] |
| start key: some_user@example.com |
| stop key: some_user@example.com\0 |
+------------------------------------------------------------------------------------+
Or update the impalad defaults file /etc/default/impala and include settings for HBASE_CACHE_BLOCKS and/or
HBASE_CACHING in the -default_query_options setting for IMPALA_SERVER_ARGS. See Modifying Impala Startup
Options on page 43 for details.
Note: In Impala 2.0 and later, these options are settable through the JDBC or ODBC interfaces using
the SET statement.
• If you issue a CREATE TABLE LIKE statement for a table mapped to an HBase table, the new table is also an
HBase table, but inherits the same underlying HBase table name as the original. The new table is effectively an
alias for the old one, not a new table with identical column structure. Avoid using CREATE TABLE LIKE for HBase
tables, to avoid any confusion.
• Copying data into an HBase table using the Impala INSERT ... SELECT syntax might produce fewer new rows
than are in the query result set. If the result set contains multiple rows with the same value for the key column,
each row supercedes any previous rows with the same key value. Because the order of the inserted rows is
unpredictable, you cannot rely on this technique to preserve the “latest” version of a particular key value.
• Because the complex data types (ARRAY, STRUCT, and MAP) available in CDH 5.5 / Impala 2.3 and higher are
currently only supported in Parquet tables, you cannot use these types in HBase tables that are queried through
Impala.
• The LOAD DATA statement cannot be used with HBase tables.
$ hbase shell
15/02/10 16:07:45
HBase Shell; enter 'help<RETURN>' for list of supported commands.
Type "exit<RETURN>" to leave the HBase Shell
Version 0.94.2-cdh4.2.0, rUnknown, Fri Feb 15 11:51:18 PST 2013
Issue the following CREATE TABLE statement in the Hive shell. (The Impala CREATE TABLE statement currently does
not support the STORED BY clause, so you switch into Hive to create the table, then back to Impala and the
impala-shell interpreter to issue the queries.)
This example creates an external table mapped to the HBase table, usable by both Impala and Hive. It is defined as an
external table so that when dropped by Impala or Hive, the original HBase table is not touched at all.
The WITH SERDEPROPERTIES clause specifies that the first column (ID) represents the row key, and maps the
remaining columns of the SQL table to HBase column families. The mapping relies on the ordinal order of the columns
in the table, not the column names in the CREATE TABLE statement. The first column is defined to be the lookup key;
the STRING data type produces the fastest key-based lookups for HBase tables.
Note: For Impala with HBase tables, the most important aspect to ensure good performance is to
use a STRING column as the row key, as shown in this example.
$ hive
Logging initialized using configuration in file:/etc/hive/conf.dist/hive-log4j.properties
Hive history file=/tmp/cloudera/hive_job_log_cloudera_201502101610_1980712808.txt
hive> use hbase;
OK
Time taken: 4.095 seconds
hive> CREATE EXTERNAL TABLE hbasestringids (
> id string,
> bool_col boolean,
Once you have established the mapping to an HBase table, you can issue DML statements and queries from Impala.
The following example shows a series of INSERT statements followed by a query. The ideal kind of query from a
performance standpoint retrieves a row from the table based on a row key mapped to a string column. An initial
INVALIDATE METADATA table_name statement makes the table created through Hive visible to Impala.
(Shell build version: Impala Shell v2.1.0-cdh4 (d520a9c) built on Mon Dec 8 21:41:17
PST 2014)
Query: use `hbase`
[localhost:21000] > invalidate metadata hbasestringids;
Fetched 0 row(s) in 0.09s
[localhost:21000] > desc hbasestringids;
+-----------------+-----------+---------+
| name | type | comment |
+-----------------+-----------+---------+
| id | string | |
| bool_col | boolean | |
| double_col | double | |
| float_col | float | |
| bigint_col | bigint | |
| int_col | int | |
| smallint_col | smallint | |
| tinyint_col | tinyint | |
| date_string_col | string | |
| string_col | string | |
| timestamp_col | timestamp | |
+-----------------+-----------+---------+
Fetched 11 row(s) in 0.02s
[localhost:21000] > insert into hbasestringids values
('0001',true,3.141,9.94,1234567,32768,4000,76,'2014-12-31','Hello world',now());
Inserted 1 row(s) in 0.26s
[localhost:21000] > insert into hbasestringids values
('0002',false,2.004,6.196,1500,8000,129,127,'2014-01-01','Foo bar',now());
Inserted 1 row(s) in 0.12s
[localhost:21000] > select * from hbasestringids where id = '0001';
+------+----------+------------+-------------------+------------+---------+--------------+-------------+-----------------+-------------+-------------------------------+
| id | bool_col | double_col | float_col | bigint_col | int_col | smallint_col
| tinyint_col | date_string_col | string_col | timestamp_col |
+------+----------+------------+-------------------+------------+---------+--------------+-------------+-----------------+-------------+-------------------------------+
| 0001 | true | 3.141 | 9.939999580383301 | 1234567 | 32768 | 4000
| 76 | 2014-12-31 | Hello world | 2015-02-10 16:36:59.764838000 |
+------+----------+------------+-------------------+------------+---------+--------------+-------------+-----------------+-------------+-------------------------------+
Fetched 1 row(s) in 0.54s
Note: After you create a table in Hive, such as the HBase mapping table in this example, issue an
INVALIDATE METADATA table_name statement the next time you connect to Impala, make Impala
aware of the new table. (Prior to Impala 1.2.4, you could not specify the table name if Impala was not
aware of the table yet; in Impala 1.2.4 and higher, specifying the table name avoids reloading the
metadata for other tables that are not changed.)
Important:
Impala query functionality for Amazon S3 is included beginning in CDH 5.4. From CDH 5.4 through
CDH 5.7, the S3 functionality for Impala is not supported or recommended for production use. In CDH
5.8 and higher, this functionality is supported and production-ready. For the most current information,
see the latest documentation for using Impala with S3.
You can use Impala to query data residing on the Amazon S3 filesystem. This capability allows convenient access to a
storage system that is remotely managed, accessible from anywhere, and integrated with various cloud-based services.
Impala can query files in any supported file format from S3. The S3 storage location can be for an entire table or
individual partitions in a partitioned table.
The default Impala tables use data files stored on HDFS, which are ideal for bulk loads and queries using full-table
scans. In contrast, queries against S3 data are less performant, making S3 suitable for holding “cold” data that is only
queried occasionally, while more frequently accessed “hot” data resides in HDFS. In a partitioned table, you can set
the LOCATION attribute for individual partitions to put some partitions on HDFS and others on S3, typically depending
on the age of the data.
<property>
<name>fs.s3a.access.key</name>
<value>your_access_key</value>
</property>
<property>
<name>fs.s3a.secret.key</name>
<value>your_secret_key</value>
</property>
As of CDH 5.4.0, these settings do not have corresponding controls in the Cloudera Manager user interface. Specify
them in the HDFS Client Advanced Configuration Snippet (Safety Valve) for hdfs-site.xml field. After specifying the
credentials, restart both the Impala and Hive services. (Restarting Hive is required because Impala queries, CREATE
TABLE statements, and so on go through the Hive metastore.)
Important: Although you can specify the access key ID and secret key as part of the s3a:// URL in
the LOCATION attribute, doing so makes this sensitive information visible in many places, such as
DESCRIBE FORMATTED output and Impala log files. Therefore, specify this information centrally in
the hdfs-site.xml file, and restrict read access to that file to only trusted users.
After you upload data files to a location already mapped to an Impala table or partition, or if you delete files in S3 from
such a location, issue the REFRESH table_name statement to make Impala aware of the new set of data files.
For a partitioned table, either specify a separate LOCATION clause for each new partition, or specify a base LOCATION
for the table and set up a directory structure in S3 to mirror the way Impala partitioned tables are structured in HDFS.
Although, strictly speaking, S3 filenames do not have directory paths, Impala treats S3 filenames with / characters the
same as HDFS pathnames that include directories.
To point a nonpartitioned table or an individual partition at S3 involves specifying a single directory path in S3, which
could be any arbitrary directory. To replicate the structure of an entire Impala partitioned table or database in S3
requires more care, with directories and subdirectories nested and named to match the equivalent directory tree in
HDFS. Consider setting up an empty staging area if necessary in HDFS, and recording the complete directory structure
so that you can replicate it in S3.
For convenience when working with multiple tables with data files stored in S3, you can create a database with a
LOCATION attribute pointing to an S3 path. Specify a URL of the form s3a://bucket/root/path/for/database
for the LOCATION attribute of the database. Any tables created inside that database automatically create directories
underneath the one specified by the database LOCATION attribute.
For example, the following session creates a partitioned table where only a single partition resides on S3. The partitions
for years 2013 and 2014 are located on HDFS. The partition for year 2015 includes a LOCATION attribute with an
s3a:// URL, and so refers to data residing on S3, under a specific path underneath the bucket impala-demo.
The following session creates a database and two partitioned tables residing entirely on S3, one partitioned by a single
column and the other partitioned by multiple columns. Because a LOCATION attribute with an s3a:// URL is specified
for the database, the tables inside that database are automatically created on S3 underneath the database directory.
To see the names of the associated subdirectories, including the partition key values, we use an S3 client tool to examine
how the directory structure is organized on S3. For example, Impala partition directories such as month=1 do not
include leading zeroes such sometimes appear in partitioned tables created through Hive.
(year=2015,month=1,day=1);
[localhost:21000] > alter table partitioned_multiple_keys add partition
(year=2015,month=1,day=31);
[localhost:21000] > alter table partitioned_multiple_keys add partition
(year=2015,month=2,day=28);
The CREATE DATABASE and CREATE TABLE statements create the associated directory paths if they do not already
exist. You can specify multiple levels of directories, and the CREATE statement creates all appropriate levels, similar
to using mkdir -p.
Use the standard S3 file upload methods to actually put the data files into the right locations. You can also put the
directory paths and data files in place before creating the associated Impala databases or tables, and Impala automatically
uses the data from the appropriate location after the associated databases and tables are created.
You can switch whether an existing table or partition points to data in HDFS or S3. For example, if you have an Impala
table or partition pointing to data files in HDFS or S3, and you later transfer those data files to the other filesystem,
use an ALTER TABLE statement to adjust the LOCATION attribute of the corresponding table or partition to reflect
that change. Because Impala does not have an ALTER DATABASE statement, this location-switching technique is not
practical for entire databases that have a custom LOCATION attribute.
| id | smallint | |
| city | string | |
| state | string | |
+-------+----------+---------+
-- Now from a web browser, upload the same data file(s) to S3 as in the HDFS table,
under the relevant bucket and path.
-- If you already have the data in S3, you would point the table LOCATION at an existing
path.
In this case, we have already uploaded a Parquet file with a million rows of data to the sample_data directory
underneath the impala-demo bucket on S3. This session creates a table with matching column settings pointing to
the corresponding location in S3, then queries the table. Because the data is already in place on S3 when the table is
created, no REFRESH statement is required.
[localhost:21000] > create table sample_data_s3 (id int, id bigint, val int, zerofill
string,
> name string, assertion boolean, city string, state string)
> stored as parquet location 's3a://impala-demo/sample_data';
[localhost:21000] > select count(*) from sample_data_s3;;
+----------+
| count(*) |
+----------+
| 1000000 |
+----------+
[localhost:21000] > select count(*) howmany, assertion from sample_data_s3 group by
assertion;
+---------+-----------+
| howmany | assertion |
+---------+-----------+
| 667149 | true |
| 332851 | false |
+---------+-----------+
Impala does not support the old s3:// block-based and s3n:// filesystem schemes, only s3a://.
Although S3 is often used to store JSON-formatted data, the current Impala support for S3 does not include directly
querying JSON data. For Impala queries, use data files in one of the file formats listed in How Impala Works with Hadoop
File Formats on page 527. If you have data in JSON format, you can prepare a flattened version of that data for querying
by Impala as part of your ETL cycle.
You cannot use the ALTER TABLE ... SET CACHED statement for tables or partitions that are located in S3.
The typical use case for Impala and Isilon together is to use Isilon for the default filesystem, replacing HDFS entirely.
In this configuration, when you create a database, table, or partition, the data always resides on Isilon storage and you
do not need to specify any special LOCATION attribute. If you do specify a LOCATION attribute, its value refers to a
path within the Isilon filesystem. For example:
Impala can write to, delete, and rename data files and database, table, and partition directories on Isilon storage.
Therefore, Impala statements such as CREATE TABLE, DROP TABLE, CREATE DATABASE, DROP DATABASE, ALTER
TABLE, and INSERT work the same with Isilon storage as with HDFS.
When the Impala spill-to-disk feature is activated by a query that approaches the memory limit, Impala writes all the
temporary data to a local (not Isilon) storage device. Because the I/O bandwidth for the temporary data depends on
the number of local disks, and clusters using Isilon storage might not have as many local disks attached, pay special
attention on Isilon-enabled clusters to any queries that use the spill-to-disk feature. Where practical, tune the queries
or allocate extra memory for Impala to avoid spilling. Although you can specify an Isilon storage device as the destination
for the temporary data for the spill-to-disk feature, that configuration is not recommended due to the need to transfer
the data both ways using remote I/O.
When tuning Impala queries on HDFS, you typically try to avoid any remote reads. When the data resides on Isilon
storage, all the I/O consists of remote reads. Do not be alarmed when you see non-zero numbers for remote read
measurements in query profile output. The benefit of the Impala and Isilon integration is primarily convenience of not
having to move or copy large volumes of data to HDFS, rather than raw query performance. You can increase the
performance of Impala I/O for Isilon systems by increasing the value for the num_remote_hdfs_io_threads
configuration parameter, in the Cloudera Manager user interface for clusters using Cloudera Manager, or through the
--num_remote_hdfs_io_threads startup option for the impalad daemon on clusters not using Cloudera Manager.
For information about managing Isilon storage devices through Cloudera Manager, see
http://www.cloudera.com/documentation/enterprise/latest/topics/cm_mc_isilon_service.html.
Required Configurations
Specify the following configurations in Cloudera Manager on the Clusters > Isilon Service > Configuration tab:
• In HDFS Client Advanced Configuration Snippet (Safety Valve) for hdfs-site.xml hdfs-site.xml and the
Cluster-wide Advanced Configuration Snippet (Safety Valve) for core-site.xml properties for the Isilon service,
set the value of the dfs.client.file-block-storage-locations.timeout.millis property to 10000.
• In the Isilon Cluster-wide Advanced Configuration Snippet (Safety Valve) for core-site.xml property for the Isilon
service, set the value of the hadoop.security.token.service.use_ip property to FALSE.
• If you see errors that reference the .Trash directory, make sure that the Use Trash property is selected.
Note:
Formerly, the logs contained the query profile for each query, showing low-level details of how the
work is distributed among nodes and how intermediate and final results are transmitted across the
network. To save space, those query profiles are now stored in zlib-compressed files in
/var/log/impala/profiles. You can access them through the Impala web user interface. For
example, at http://impalad-node-hostname:25000/queries, each query is followed by a
Profile link leading to a page showing extensive analytical data for the query execution.
The auditing feature introduced in Impala 1.1.1 produces a separate set of audit log files when enabled.
See Auditing Impala Operations on page 112 for details.
The lineage feature introduced in Impala 2.2.0 produces a separate lineage log file when enabled. See
Viewing Lineage Information for Impala Data on page 114 for details.
Impala stores information using the glog_v logging system. You will see some messages referring to C++ file names.
Logging is affected by:
• The GLOG_v environment variable specifies which types of messages are logged. See Setting Logging Levels on
page 576 for details.
• The -logbuflevel startup flag for the impalad daemon specifies how often the log information is written to
disk. The default is 0, meaning that the log is immediately flushed to disk when Impala outputs an important
messages such as a warning or an error, but less important messages such as informational ones are buffered in
memory rather than being flushed to disk immediately.
• Cloudera Manager has an Impala configuration setting that sets the -logbuflevel startup option.
Note:
The web interface limits the amount of logging information displayed. To view every log entry, access
the log files directly through the file system.
You can view the contents of the impalad.INFO log file in the file system. With the default configuration settings,
the start of the log file appears as follows:
Note: The preceding example shows only a small part of the log file. Impala log files are often several
megabytes in size.
export GLOG_v=1
Note: For performance reasons, Cloudera highly recommends not enabling the most verbose logging
level of 3.
For more information on how to configure GLOG, including how to set variable logging levels for different system
components, see documentation for the glog project on github.
Troubleshooting Impala
Troubleshooting for Impala requires being able to diagnose and debug problems with performance, network connectivity,
out-of-memory conditions, disk space usage, and crash or hang conditions in any of the Impala-related daemons.
On modern hardware, a throughput rate of less than 100 MB/s typically indicates a performance issue with the storage
device. Correct the hardware problem before continuing with Impala tuning or benchmarking.
Queries return Impala metadata may be outdated after changes Where possible, use the appropriate Impala
incorrect are performed in Hive. statement (INSERT, LOAD DATA, CREATE TABLE,
results. ALTER TABLE, COMPUTE STATS, and so on)
rather than switching back and forth between
Impala and Hive. Impala automatically broadcasts
the results of DDL and DML operations to all
Impala nodes in the cluster, but does not
automatically recognize when such changes are
made through Hive. After inserting data, adding
a partition, or other operation in Hive, refresh the
metadata for the table as described in REFRESH
Statement on page 277.
Queries are Some impalad instances may not have started. Ensure Impala is installed on all DataNodes. Start
slow to return Using a browser, connect to the host running the any impalad instances that are not running.
results. Impala state store. Connect using an address of
the form http://hostname:port/metrics.
Queries are Impala may not be configured to use native Ensure Impala is configured to use native
slow to return checksumming. Native checksumming uses checksumming as described in Post-Installation
results. machine-specific instructions to compute Configuration for Impala on page 30.
checksums over HDFS data very quickly. Review
Impala logs. If you find instances of "INFO
util.NativeCodeLoader: Loaded the
Queries are Impala may not be configured to use data locality Test Impala for data locality tracking and make
slow to return tracking. configuration changes as necessary. Information
results. on this process can be found in Post-Installation
Configuration for Impala on page 30.
Attempts to This can be the result of permissions issues. For In general, ensure the Impala user has sufficient
complete example, you could use the Hive shell as the hive permissions. In the preceding example, ensure
Impala tasks user to create a table. After creating this table, the Impala user has sufficient permissions to the
such as you could attempt to complete some action, such table that the Hive user created.
executing as an INSERT-SELECT on the table. Because the
INSERT-SELECT table was created using one user and the
actions fail. The INSERT-SELECT is attempted by another, this
Impala logs action may fail due to permissions issues.
include notes
that files could
not be opened
due to
permission
denied.
Impala fails to A large number of databases, tables, partitions, Configure the statestore timeout value and
start up, with and so on can require metadata synchronization, possibly other settings related to the frequency
the impalad particularly on startup, that takes longer than the of statestore updates and metadata loading. See
logs referring default timeout for the statestore service. Increasing the Statestore Timeout on page 84 and
to errors Scalability Considerations for the Impala
connecting to Statestore on page 516.
the statestore
service and
attempts to
re-register.
daemon within any cluster, you view the web UI only on the particular host that serves as the Impala Catalog
Server.
Note:
The web user interface is primarily for problem diagnosis and troubleshooting. The items listed and
their formats are subject to change. To monitor Impala health, particularly across the entire cluster
at once, use the Cloudera Manager interface.
Note: To get a convenient picture of the health of all Impala nodes in a cluster, use the Cloudera
Manager interface, which collects the low-level operational information from all Impala nodes, and
presents a unified view of the entire cluster.
Main Page
By default, the main page of the debug web UI is at http://impala-server-hostname:25000/ (non-secure cluster)
or https://impala-server-hostname:25000/ (secure cluster).
This page lists the version of the impalad daemon, plus basic hardware and software information about the
corresponding host, such as information about the CPU, memory, disks, and operating system version.
Backends Page
By default, the backends page of the debug web UI is at http://impala-server-hostname:25000/backends
(non-secure cluster) or https://impala-server-hostname:25000/backends (secure cluster).
This page lists the host and port info for each of the impalad nodes in the cluster. Because each impalad daemon
knows about every other impalad daemon through the statestore, this information should be the same regardless
of which node you select. Links take you to the corresponding debug web pages for any of the other nodes in the
cluster.
Catalog Page
By default, the catalog page of the debug web UI is at http://impala-server-hostname:25000/catalog
(non-secure cluster) or https://impala-server-hostname:25000/catalog (secure cluster).
This page displays a list of databases and associated tables recognized by this instance of impalad. You can use this
page to locate which database a table is in, check the exact spelling of a database or table name, look for identical
table names in multiple databases, and so on.
Logs Page
By default, the logs page of the debug web UI is at http://impala-server-hostname:25000/logs (non-secure
cluster) or https://impala-server-hostname:25000/logs (secure cluster).
This page shows the last portion of the impalad.INFO log file, the most detailed of the info, warning, and error logs
for the impalad daemon. You can refer here to see the details of the most recent operations, whether the operations
succeeded or encountered errors. This central page can be more convenient than looking around the filesystem for
the log files, which could be in different locations on clusters that use Cloudera Manager or not.
Memz Page
By default, the memz page of the debug web UI is at http://impala-server-hostname:25000/memz (non-secure
cluster) or https://impala-server-hostname:25000/memz (secure cluster).
This page displays summary and detailed information about memory usage by the impalad daemon. You can see the
memory limit in effect for the node, and how much of that memory Impala is currently using.
Metrics Page
By default, the metrics page of the debug web UI is at http://impala-server-hostname:25000/metrics
(non-secure cluster) or https://impala-server-hostname:25000/metrics (secure cluster).
This page displays the current set of metrics: counters and flags representing various aspects of impalad internal
operation. For the meanings of these metrics, see Impala Metrics in the Cloudera Manager documentation.
Queries Page
By default, the queries page of the debug web UI is at http://impala-server-hostname:25000/queries
(non-secure cluster) or https://impala-server-hostname:25000/queries (secure cluster).
This page lists all currently running queries, plus any completed queries whose details still reside in memory. The
queries are listed in reverse chronological order, with the most recent at the top. (You can control the amount of
memory devoted to completed queries by specifying the --query_log_size startup option for impalad.)
On this page, you can see at a glance how many SQL statements are failing (State value of EXCEPTION), how large
the result sets are (# rows fetched), and how long each statement took (Start Time and End Time).
Each query has an associated link that displays the detailed query profile, which you can examine to understand the
performance characteristics of that query. See Using the Query Profile for Performance Tuning on page 509 for details.
Sessions Page
By default, the sessions page of the debug web UI is at http://impala-server-hostname:25000/sessions
(non-secure cluster) or https://impala-server-hostname:25000/sessions (secure cluster).
This page displays information about the sessions currently connected to this impalad instance. For example, sessions
could include connections from the impala-shell command, JDBC or ODBC applications, or the Impala Query UI in
the Hue web interface.
Threadz Page
By default, the threadz page of the debug web UI is at http://impala-server-hostname:25000/threadz
(non-secure cluster) or https://impala-server-hostname:25000/threadz (secure cluster).
This page displays information about the threads used by this instance of impalad, and shows which categories they
are grouped into. Making use of this information requires substantial knowledge about Impala internals.
Varz Page
By default, the varz page of the debug web UI is at http://impala-server-hostname:25000/varz (non-secure
cluster) or https://impala-server-hostname:25000/varz (secure cluster).
This page shows the configuration settings in effect when this instance of impalad communicates with other Hadoop
components such as HDFS and YARN. These settings are collected from a set of configuration files; Impala might not
actually make use of all settings.
The bottom of this page also lists all the command-line settings in effect for this instance of impalad. See Modifying
Impala Startup Options on page 43 for information about modifying these values.
Impala Daemon Impala Daemon Frontend Port 21050 External Used to transmit commands
and receive results by
applications, such as Business
Intelligence tools, using JDBC,
the Beeswax query editor in
Hue, and version 2.0 or higher
of the Cloudera ODBC driver.
Impala Daemon Impala Daemon Backend Port 22000 Internal Internal use only. Impala
daemons use this port to
communicate with each other.
Impala Daemon StateStoreSubscriber Service 23000 Internal Internal use only. Impala
Port daemons listen on this port for
updates from the statestore
daemon.
Catalog Daemon StateStoreSubscriber Service 23020 Internal Internal use only. The catalog
Port daemon listens on this port for
updates from the statestore
daemon.
Impala Daemon Impala Daemon HTTP Server 25000 External Impala web interface for
Port administrators to monitor and
troubleshoot.
Impala StateStore StateStore HTTP Server Port 25010 External StateStore web interface for
Daemon administrators to monitor and
troubleshoot.
Impala Catalog Catalog HTTP Server Port 25020 External Catalog service web interface
Daemon for administrators to monitor
and troubleshoot. New in
Impala 1.2 and higher.
Impala StateStore StateStore Service Port 24000 Internal Internal use only. The
Daemon statestore daemon listens on
this port for
registration/unregistration
requests.
Impala Daemon Llama Callback Port 28000 Internal Internal use only. Impala
daemons use to communicate
with Llama. New in CDH 5.0.0
and higher.
Impala Llama Llama Thrift Admin Port 15002 Internal Internal use only. New in CDH
ApplicationMaster 5.0.0 and higher.
Impala Llama Llama Thrift Port 15000 Internal Internal use only. New in CDH
ApplicationMaster 5.0.0 and higher.
Impala Llama Llama HTTP Port 15001 External Llama service web interface for
ApplicationMaster administrators to monitor and
troubleshoot. New in CDH 5.0.0
and higher.
external
false
fields
fileformat
finalize_fn
first
float
following
for
format
formatted
from
full
function
functions
grant
group
having
if
in
incremental
init_fn
inner
inpath
insert
int
integer
intermediate
interval
into
invalidate
is
join
last
left
like
limit
lines
load
location
merge_fn
metadata
not
null
nulls
offset
on
or
order
outer
over
overwrite
parquet
parquetfile
partition
partitioned
partitions
preceding
prepare_fn
produced
purge
range
rcfile
real
refresh
regexp
rename
replace
restrict
returns
revoke
right
rlike
role
roles
row
rows
schema
schemas
select
semi
sequencefile
serdeproperties
serialize_fn
set
show
smallint
stats
stored
straight_join
string
symbol
table
tables
tblproperties
terminated
textfile
then
timestamp
tinyint
to
true
truncate
unbounded
uncached
union
update_fn
use
using
values
varchar
view
when
where
with
any
authorization
backup
begin
break
browse
bulk
cascade
check
checkpoint
close
clustered
coalesce
collate
commit
constraint
contains
continue
convert
current
current_date
current_time
current_timestamp
current_user
cursor
dbcc
deallocate
declare
default
delete
deny
disk
distributed
dump
errlvl
escape
except
exec
execute
exit
fetch
file
fillfactor
for
foreign
freetext
goto
holdlock
identity
index
intersect
key
kill
lineno
merge
national
nocheck
nonclustered
nullif
of
off
offsets
open
option
percent
pivot
plan
precision
primary
print
proc
procedure
public
raiserror
read
readtext
reconfigure
references
replication
restore
restrict
return
revert
rollback
rowcount
rule
save
securityaudit
session_user
setuser
shutdown
some
statistics
system_user
tablesample
textsize
then
top
tran
transaction
trigger
try_convert
unique
unpivot
update
updatetext
user
varying
waitfor
while
within
writetext
Trying Impala
How do I try Impala out?
To look at the core features and functionality on Impala, the easiest way to try out Impala is to download the Cloudera
QuickStart VM and start the Impala service through Cloudera Manager, then use impala-shell in a terminal window
or the Impala Query UI in the Hue web interface.
To do performance testing and try out the management features for Impala on a cluster, you need to move beyond
the QuickStart VM with its virtualized single-node environment. Ideally, download the Cloudera Manager software to
set up the cluster, then install the Impala software through Cloudera Manager.
and your cluster has 50 nodes, then each of those 50 nodes will transmit a maximum of 1000 rows back to the
coordinator node. The coordinator node needs enough memory to sort (LIMIT * cluster_size) rows, although in
the end the final result set is at most LIMIT rows, 1000 in this case.
Likewise, if you execute the query:
then each node filters out a set of rows matching the WHERE conditions, sorts the results (with no size limit), and
sends the sorted intermediate rows back to the coordinator node. The coordinator node might need substantial
memory to sort the final result set, and so might use a temporary disk work area for that final phase of the query.
• Whether the query contains any join clauses, GROUP BY clauses, analytic functions, or DISTINCT operators. These
operations all require some in-memory work areas that vary depending on the volume and distribution of data.
In Impala 2.0 and later, these kinds of operations utilize temporary disk work areas if memory usage grows too
large to handle. See SQL Operations that Spill to Disk on page 517 for details.
• The size of the result set. When intermediate results are being passed around between nodes, the amount of data
depends on the number of columns returned by the query. For example, it is more memory-efficient to query
only the columns that are actually needed in the result set rather than always issuing SELECT *.
• The mechanism by which work is divided for a join query. You use the COMPUTE STATS statement, and query
hints in the most difficult cases, to help Impala pick the most efficient execution plan. See Performance
Considerations for Join Queries on page 482 for details.
See Hardware Requirements on page 24 for more details and recommendations about Impala hardware prerequisites.
• ODBC: Impala is certified to run against MicroStrategy and Tableau, with restrictions. For more information, see
Configuring Impala to Work with ODBC on page 31.
• Querying data stored in HDFS and HBase in a single query. See Using Impala to Query HBase Tables on page 557
for details.
• In Impala 2.2.0 and higher, querying data stored in the Amazon Simple Storage Service (S3). See Using Impala to
Query the Amazon S3 Filesystem on page 566 for details.
• Concurrent client requests. Each Impala daemon can handle multiple concurrent client requests. The effects on
performance depend on your particular hardware and workload.
• Kerberos authentication. For more information, see Impala Security on page 91.
• Partitions. With Impala SQL, you can create partitioned tables with the CREATE TABLE statement, and add and
drop partitions with the ALTER TABLE statement. Impala also takes advantage of the partitioning present in Hive
tables. See Partitioning for Impala Tables on page 522 for details.
What features from relational databases or Hive are not available in Impala?
• Querying streaming data.
• Deleting individual rows. You delete data in bulk by overwriting an entire table or partition, or by dropping a table.
• Indexing (not currently). LZO-compressed text files can be indexed outside of Impala, as described in Using
LZO-Compressed Text Files on page 531.
• Full text search on text fields. The Cloudera Search product is appropriate for this use case.
• Custom Hive Serializer/Deserializer classes (SerDes). Impala supports a set of common native file formats that
have built-in SerDes in CDH. See How Impala Works with Hadoop File Formats on page 527 for details.
• Checkpointing within a query. That is, Impala does not save intermediate results to disk during long-running
queries. Currently, Impala cancels a running query if any host on which that query is executing fails. When one or
more hosts are down, Impala reroutes future queries to only use the available hosts, and Impala detects when
the hosts come back up and begins using them again. Because a query can be submitted through any Impala node,
there is no single point of failure. In the future, we will consider adding additional work allocation features to
Impala, so that a running query would complete even in the presence of host failures.
• Hive indexes.
• Non-Hadoop data stores, such as relational databases.
For the detailed list of features that are different between Impala and HiveQL, see SQL Differences Between Impala
and Hive on page 461.
Is Avro supported?
Yes, Avro is supported. Impala has always been able to query Avro tables. You can use the Impala LOAD DATA statement
to load existing Avro data files into a table. Starting with Impala 1.4, you can create Avro tables with Impala. Currently,
you still use the INSERT statement in Hive to copy data from another table into an Avro table. See Using the Avro File
Format with Impala Tables on page 547 for details.
How do I?
How do I prevent users from seeing the text of SQL queries?
For instructions on making the Impala log files unreadable by unprivileged users, see Securing Impala Data and Log
Files on page 92.
For instructions on password-protecting the web interface to the Impala log files and other internal server information,
see Securing the Impala Web User Interface on page 93.
In Impala 2.2 / CDH 5.4 and higher, you can use the log redaction feature to obfuscate sensitive information in Impala
log files. See http://www.cloudera.com/documentation/enterprise/latest/topics/sg_redaction.html for details.
statestore.live-backends:3
statestore.live-backends.list:[host1:22000, host1:26000, host2:22000]
The number of impalad nodes is the number of list items referring to port 22000, in this case two. (Typically, this
number is one less than the number reported by the statestore.live-backends line.) If an impalad node became
unavailable or came back after an outage, the information reported on this page would change appropriately.
Impala Performance
Are results returned as they become available, or all at once when a query completes?
Impala streams results whenever they are available, when possible. Certain SQL operations (aggregation or ORDER
BY) require all of the input to be ready before Impala can return results.
- BytesRead: 180.33 MB
- BytesReadLocal: 180.33 MB
- BytesReadShortCircuit: 180.33 MB
If BytesReadLocal is lower than BytesRead, something in your cluster is misconfigured, such as the impalad
daemon not running on all the data nodes. If BytesReadShortCircuit is lower than BytesRead, short-circuit
reads are not enabled properly on that node; see Post-Installation Configuration for Impala on page 30 for
instructions.
• If the table was just created, or this is the first query that accessed the table after an INVALIDATE METADATA
statement or after the impalad daemon was restarted, there might be a one-time delay while the metadata for
the table is loaded and cached. Check whether the slowdown disappears when the query is run again. When doing
performance comparisons, consider issuing a DESCRIBE table_name statement for each table first, to make
sure any timings only measure the actual query time and not the one-time wait to load the table metadata.
• Is the table data in uncompressed text format? Check by issuing a DESCRIBE FORMATTED table_name statement.
A text table is indicated by the line:
InputFormat: org.apache.hadoop.mapred.TextInputFormat
Although uncompressed text is the default format for a CREATE TABLE statement with no STORED AS clauses,
it is also the bulkiest format for disk storage and consequently usually the slowest format for queries. For data
where query performance is crucial, particularly for tables that are frequently queried, consider starting with or
converting to a compact binary file format such as Parquet, Avro, RCFile, or SequenceFile. For details, see How
Impala Works with Hadoop File Formats on page 527.
• If your table has many columns, but the query refers to only a few columns, consider using the Parquet file format.
Its data files are organized with a column-oriented layout that lets queries minimize the amount of I/O needed
to retrieve, filter, and aggregate the values for specific columns. See Using the Parquet File Format with Impala
Tables on page 535 for details.
• If your query involves any joins, are the tables in the query ordered so that the tables or subqueries are ordered
with the one returning the largest number of rows on the left, followed by the smallest (most selective), the second
smallest, and so on? That ordering allows Impala to optimize the way work is distributed among the nodes and
how intermediate results are routed from one node to another. For example, all other things being equal, the
following join order results in an efficient query:
See Performance Considerations for Join Queries on page 482 for performance tips for join queries.
• Also for join queries, do you have table statistics for the table, and column statistics for the columns used in the
join clauses? Column statistics let Impala better choose how to distribute the work for the various pieces of a join
query. See Table and Column Statistics on page 489 for details about gathering statistics.
• Does your table consist of many small data files? Impala works most efficiently with data files in the multi-megabyte
range; Parquet, a format optimized for data warehouse-style queries, uses large files (originally 1 GB, now 256
MB in Impala 2.0 and higher) with a block size matching the file size. Use the DESCRIBE FORMATTED table_name
statement in impala-shell to see where the data for a table is located, and use the hadoop fs -ls or hdfs
dfs -ls Unix commands to see the files and their sizes. If you have thousands of small data files, that is a signal
that you should consolidate into a smaller number of large files. Use an INSERT ... SELECT statement to copy
the data to a new table, reorganizing into new data files as part of the process. Prefer to construct large data files
and import them in bulk through the LOAD DATA or CREATE EXTERNAL TABLE statements, rather than issuing
many INSERT ... VALUES statements; each INSERT ... VALUES statement creates a separate tiny data file.
If you have thousands of files all in the same directory, but each one is megabytes in size, consider using a partitioned
table so that each partition contains a smaller number of files. See the following point for more on partitioning.
• If your data is easy to group according to time or geographic region, have you partitioned your table based on the
corresponding columns such as YEAR, MONTH, and/or DAY? Partitioning a table based on certain columns allows
queries that filter based on those same columns to avoid reading the data files for irrelevant years, postal codes,
and so on. (Do not partition down to too fine a level; try to structure the partitions so that there is still sufficient
data in each one to take advantage of the multi-megabyte HDFS block size.) See Partitioning for Impala Tables on
page 522 for details.
Does Impala performance improve as it is deployed to more hosts in a cluster in much the same way that Hadoop
performance does?
Yes. Impala scales with the number of hosts. It is important to install Impala on all the DataNodes in the cluster, because
otherwise some of the nodes must do remote reads to retrieve data not available for local reads. Data locality is an
important architectural aspect for Impala performance. See this Impala performance blog post for background. Note
that this blog post refers to benchmarks with Impala 1.1.1; Impala has added even more performance features in the
1.2.x series.
Is MapReduce required for Impala? Will Impala continue to work as expected if MapReduce is stopped?
Impala does not use MapReduce at all.
Is Impala intended to handle real time queries in low-latency applications or is it for ad hoc queries for the purpose of
data exploration?
Ad-hoc queries are the primary use case for Impala. We anticipate it being used in many other situations where
low-latency is required. Whether Impala is appropriate for any particular use-case depends on the workload, data size
and query volume. See Impala Benefits on page 16 for the primary benefits you can expect when using Impala.
Can I use Impala to query data already loaded into Hive and HBase?
There are no additional steps to allow Impala to query tables managed by Hive, whether they are stored in HDFS or
HBase. Make sure that Impala is configured to access the Hive metastore correctly and you should be ready to go.
Keep in mind that impalad, by default, runs as the impala user, so you might need to adjust some file permissions
depending on how strict your permissions are currently.
See Using Impala to Query HBase Tables on page 557 for details about querying data in HBase.
Impala Availability
Is Impala production ready?
Impala has finished its beta release cycle, and the 1.0, 1.1, and 1.2 GA releases are production ready. The 1.1.x series
includes additional security features for authorization, an important requirement for production use in many
organizations. The 1.2.x series includes important performance features, particularly for large join queries. Some
Cloudera customers are already using Impala for large workloads.
The Impala 1.3.0 and higher releases are bundled with corresponding levels of CDH 5. The number of new features
grows with each release. See New Features in Apache Impala on page 604 for a full list.
Can Impala and MapReduce jobs run on the same cluster without resource contention?
Yes. See Controlling Impala Resource Usage on page 500 for how to control Impala resource usage using the Linux
cgroup mechanism, and Integrated Resource Management with YARN on page 83 for how to use Impala with the YARN
resource management framework. Impala is designed to run on the DataNode hosts. Any contention depends mostly
on the cluster setup and workload.
For a detailed example of configuring a cluster to share resources between Impala queries and MapReduce jobs, see
Setting up a Multi-tenant Cluster for Impala and MapReduce
Impala Internals
On which hosts does Impala run?
Cloudera strongly recommends running the impalad daemon on each DataNode for good performance. Although this
topology is not a hard requirement, if there are data blocks with no Impala daemons running on any of the hosts
containing replicas of those blocks, queries involving that data could be very inefficient. In that case, the data must be
transmitted from one host to another for processing by “remote reads”, a condition Impala normally tries to avoid.
See Impala Concepts and Architecture on page 18 for details about the Impala architecture. Impala schedules query
fragments on all hosts holding data relevant to the query, if possible.
Impala uses a more efficient execution engine by taking advantage of modern hardware and technologies:
• Impala generates runtime code. Impala uses LLVM to generate assembly code for the query that is being run.
Individual queries do not have to pay the overhead of running on a system that needs to be able to execute
arbitrary queries.
• Impala uses available hardware instructions when possible. Impala uses the supplemental SSE3 (SSSE3) instructions
which can offer tremendous speedups in some cases. (Impala 2.0 and 2.1 required the SSE4.1 instruction set;
Impala 2.2 and higher relax the restriction again so only SSSE3 is required.)
• Impala uses better I/O scheduling. Impala is aware of the disk location of blocks and is able to schedule the order
to process blocks to keep all disks busy.
• Impala is designed for performance. A lot of time has been spent in designing Impala with sound
performance-oriented fundamentals, such as tight inner loops, inlined function calls, minimal branching, better
use of cache, and minimal memory usage.
SQL
Is there an UPDATE statement?
Impala does not currently have an UPDATE statement, which would typically be used to change a single row, a small
group of rows, or a specific column. The HDFS-based files used by typical Impala queries are optimized for bulk operations
across many megabytes of data at a time, making traditional UPDATE operations inefficient or impractical.
You can use the following techniques to achieve the same goals as the familiar UPDATE statement, in a way that
preserves efficient file layouts for subsequent queries:
• Replace the entire contents of a table or partition with updated data that you have already staged in a different
location, either using INSERT OVERWRITE, LOAD DATA, or manual HDFS file operations followed by a REFRESH
statement for the table. Optionally, you can use built-in functions and expressions in the INSERT statement to
transform the copied data in the same way you would normally do in an UPDATE statement, for example to turn
a mixed-case string into all uppercase or all lowercase.
• To update a single row, use an HBase table, and issue an INSERT ... VALUES statement using the same key as
the original row. Because HBase handles duplicate keys by only returning the latest row with a particular key
value, the newly inserted row effectively hides the previous one.
Why do I have to use REFRESH and INVALIDATE METADATA, what do they do?
In Impala 1.2 and higher, there is much less need to use the REFRESH and INVALIDATE METADATA statements:
• The new impala-catalog service, represented by the catalogd daemon, broadcasts the results of Impala DDL
statements to all Impala nodes. Thus, if you do a CREATE TABLE statement in Impala while connected to one
node, you do not need to do INVALIDATE METADATA before issuing queries through a different node.
• The catalog service only recognizes changes made through Impala, so you must still issue a REFRESH statement
if you load data through Hive or by manipulating files in HDFS, and you must issue an INVALIDATE METADATA
statement if you create a table, alter a table, add or drop partitions, or do other DDL statements in Hive.
• Because the catalog service broadcasts the results of REFRESH and INVALIDATE METADATA statements to all
nodes, in the cases where you do still need to issue those statements, you can do that on a single node rather
than on every node, and the changes will be automatically recognized across the cluster, making it more convenient
to load balance by issuing queries through arbitrary Impala nodes rather than always using the same coordinator
node.
When Impala deletes files and they are moved to the HDFS trashcan, they go into an HDFS directory owned by the
impala user. If the impala user does not have an HDFS home directory where a trashcan can be created, the files
are not deleted or moved, as a safety measure. If you issue a DROP TABLE statement and find that the table data files
are left in their original location, create an HDFS directory /user/impala, owned and writeable by the impala user.
For example, you might find that /user/impala is owned by the hdfs user, in which case you would switch to the
hdfs user and issue a command such as:
select 2+2;
select substr('hello',2,1);
select pow(10,6);
Partitioned Tables
How do I load a big CSV file into a partitioned table?
To load a data file into a partitioned table, when the data file includes fields like year, month, and so on that correspond
to the partition key columns, use a two-stage process. First, use the LOAD DATA or CREATE EXTERNAL TABLE
statement to bring the data into an unpartitioned text table. Then use an INSERT ... SELECT statement to copy
the data from the unpartitioned table to a partitioned one. Include a PARTITION clause in the INSERT statement to
specify the partition key columns. The INSERT operation splits up the data into separate data files for each partition.
For examples, see Partitioning for Impala Tables on page 522. For details about loading data into partitioned Parquet
tables, a popular choice for high-volume data, see Loading Data into Parquet Tables on page 536.
HBase
What kinds of Impala queries or data are best suited for HBase?
HBase tables are ideal for queries where normally you would use a key-value store. That is, where you retrieve a single
row or a few rows, by testing a special unique key column using the = or IN operators.
HBase tables are not suitable for queries that produce large result sets with thousands of rows. HBase tables are also
not suitable for queries that perform full table scans because the WHERE clause does not request specific values from
the unique key column.
Use HBase tables for data that is inserted one row or a few rows at a time, such as by the INSERT ... VALUES syntax.
Loading data piecemeal like this into an HDFS-backed table produces many tiny files, which is a very inefficient layout
for HDFS data files.
If the lack of an UPDATE statement in Impala is a problem for you, you can simulate single-row updates by doing an
INSERT ... VALUES statement using an existing value for the key column. The old row value is hidden; only the new
row value is seen by queries.
HBase tables are often wide (containing many columns) and sparse (with most column values NULL). For example, you
might record hundreds of different data points for each user of an online service, such as whether the user had registered
for an online game or enabled particular account features. With Impala and HBase, you could look up all the information
for a specific customer efficiently in a single query. For any given customer, most of these columns might be NULL,
because a typical customer might not make use of most features of an online service.
Note: Starting in April 2016, future release note updates are being consolidated in a single location
to avoid duplication of stale or incomplete information. You can view online the Impala New Features,
Incompatible Changes, Known Issues, and Fixed Issues. You can view or print all of these by downloading
the latest Impala PDF.
Note: Impala 2.4.0 is available as part of CDH 5.6.0 and is not available for CDH 4. Cloudera does not
intend to release future versions of Impala for CDH 4 outside patch and maintenance releases if
required. Given the end-of-maintenance status for CDH 4, Cloudera recommends all customers to
migrate to a recent CDH 5 release.
Note: Impala 2.3.0 is available as part of CDH 5.5.0 and is not available for CDH 4. Cloudera does not
intend to release future versions of Impala for CDH 4 outside patch and maintenance releases if
required. Given the end-of-maintenance status for CDH 4, Cloudera recommends all customers to
migrate to a recent CDH 5 release.
The following are the major new features in Impala 2.3.x. This major release, available as part of CDH 5.5.x, contains
improvements to SQL syntax (particularly new support for complex types), performance, manageability, security.
• Complex data types: STRUCT, ARRAY, and MAP. These types can encode multiple named fields, positional items,
or key-value pairs within a single column. You can combine these types to produce nested types with arbitrarily
deep nesting, such as an ARRAY of STRUCT values, a MAP where each key-value pair is an ARRAY of other MAP
values, and so on. Currently, complex data types are only supported for the Parquet file format.
• Column-level authorization lets you define access to particular columns within a table, rather than the entire table.
This feature lets you reduce the reliance on creating views to set up authorization schemes for subsets of
information.
• The TRUNCATE TABLE statement removes all the data from a table without removing the table itself.
• Nested loop join queries. Some join queries that formerly required equality comparisons can now use operators
such as < or >=. This same join mechanism is used internally to optimize queries that retrieve values from complex
type columns.
• Reduced memory usage and improved performance and robustness for spill-to-disk feature.
• Performance improvements for querying Parquet data files containing multiple row groups and multiple data
blocks:
– For files written by Hive, SparkSQL, and other Parquet MR writers and spanning multiple HDFS blocks. Impala
now scans the extra data blocks locally when possible, rather than using remote reads.
– Impala queries benefit from the improved alignment of row groups with HDFS blocks for Parquet files written
by Hive, MapReduce, and other components in CDH 5.5 and higher. (Impala itself never writes multiblock
Parquet files, so the alignment change does not apply to Parquet files produced by Impala.) These Parquet
writers now add padding to Parquet files that they write to align row groups with HDFS blocks. The
parquet.writer.max-padding setting specifies the maximum number of bytes, by default 8 megabytes,
that can be added to the file between row groups to fill the gap at the end of one block so that the next row
group starts at the beginning of the next block. If the gap is larger than this size, the writer attempts to fit
another entire row group in the remaining space. Include this setting in the hive-site configuration file to
influence Parquet files written by Hive, or the hdfs-site configuration file to influence Parquet files written
by all non-Impala components.
–
• Many new built-in scalar functions, for convenience and enhanced portability of SQL that uses common industry
extensions.
Math functions:
– ATAN2
– COSH
– COT
– DCEIL
– DEXP
– DFLOOR
– DLOG10
– DPOW
– DROUND
– DSQRT
– DTRUNC
– FACTORIAL, and corresponding ! operator
– FPOW
– RADIANS
– RANDOM
– SINH
– TANH
String functions:
– BTRIM
– CHR
– REGEXP_LIKE
– SPLIT_PART
Date and time functions:
– INT_MONTHS_BETWEEN
– MONTHS_BETWEEN
– TIMEOFDAY
– TIMESTAMP_CMP
Currently, you can only use one of server-to-server TLS/SSL encryption or Kerberos authentication. This limitation
is tracked by the issue IMPALA-2598.
• Improved flexibility for intermediate data types in user-defined aggregate functions (UDAFs).
In CDH 5.5.2 / Impala 2.3.2, the bug fix for IMPALA-2598 removes the restriction on using both Kerberos and SSL for
internal communication between Impala components.
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
Note: Impala 2.2.0 is available as part of CDH 5.4.0 and is not available for CDH 4. Cloudera does not
intend to release future versions of Impala for CDH 4 outside patch and maintenance releases if
required. Given the end-of-maintenance status for CDH 4, Cloudera recommends all customers to
migrate to a recent CDH 5 release.
The following are the major new features in Impala 2.2.0. This major release, available as part of CDH 5.4.0, contains
improvements to performance, manageability, security, and SQL syntax.
• Several improvements to date and time features enable higher interoperability with Hive and other database
systems, provide more flexibility for handling time zones, and future-proof the handling of TIMESTAMP values:
– The WITH REPLICATION clause for the CREATE TABLE and ALTER TABLE statements lets you control the
replication factor for HDFS caching for a specific table or partition. By default, each cached block is only
present on a single host, which can lead to CPU contention if the same host processes each cached block.
Increasing the replication factor lets Impala choose different hosts to process different cached blocks, to
better distribute the CPU load.
– Startup flags for the impalad daemon enable a higher level of compatibility with TIMESTAMP values written
by Hive, and more flexibility for working with date and time data using the local time zone instead of UTC.
To enable these features, set the impalad startup flags
-use_local_tz_for_unix_timestamp_conversions=true and
-convert_legacy_hive_parquet_utc_timestamps=true.
• The SHOW FILES statement lets you view the names and sizes of the files that make up an entire table or a specific
partition. See SHOW FILES Statement on page 322 for details.
• Impala can now run queries against Parquet data containing columns with complex or nested types, as long as
the query only refers to columns with scalar types.
• Performance improvements for queries that include IN() operators and involve partitioned tables.
• The new -max_log_files configuration option specifies how many log files to keep at each severity level. The
default value is 10, meaning that Impala preserves the latest 10 log files for each severity level (INFO, WARNING,
and ERROR) for each Impala-related daemon (impalad, statestored, and catalogd). Impala checks to see if
any old logs need to be removed based on the interval specified in the logbufsecs setting, every 5 seconds by
default. See Rotating Impala Logs on page 575 for details.
• Redaction of sensitive data from Impala log files. This feature protects details such as credit card numbers or tax
IDs from administrators who see the text of SQL statements in the course of monitoring and troubleshooting a
Hadoop cluster. See Redacting Sensitive Information from Impala Log Files on page 577 for background information
for Impala users, and http://www.cloudera.com/documentation/enterprise/latest/topics/sg_redaction.html for
usage details.
• Lineage information is available for data created or queried by Impala. This feature lets you track who has accessed
data through Impala SQL statements, down to the level of specific columns, and how data has been propagated
between tables. See Viewing Lineage Information for Impala Data on page 114 for background information for
Impala users,
http://www.cloudera.com/documentation/enterprise/latest/topics/datamgmt_impala_lineage_log.html for usage
details, and http://www.cloudera.com/documentation/enterprise/latest/topics/cn_iu_lineage.html. for how to
interpret the lineage information.
• Impala tables and partitions can now be located on the Amazon Simple Storage Service (S3) filesystem, for
convenience in cases where data is already located in S3 and you prefer to query it in-place. Queries might have
lower performance than when the data files reside on HDFS, because Impala uses some HDFS-specific optimizations.
Impala can query data in S3, but cannot write to S3. Therefore, statements such as INSERT and LOAD DATA are
not available when the destination table or partition is in S3. See Using Impala to Query the Amazon S3 Filesystem
on page 566 for details.
Important:
Impala query functionality for Amazon S3 is included beginning in CDH 5.4. From CDH 5.4 through
CDH 5.7, the S3 functionality for Impala is not supported or recommended for production use.
In CDH 5.8 and higher, this functionality is supported and production-ready. For the most current
information, see the latest documentation for using Impala with S3.
• Improved support for HDFS encryption. The LOAD DATA statement now works when the source directory and
destination table are in different encryption zones.
• Additional arithmetic function mod(). See Impala Mathematical Functions on page 339 for details.
• Flexibility to interpret TIMESTAMP values using the UTC time zone (the traditional Impala behavior) or using the
local time zone (for compatibility with TIMESTAMP values produced by Hive).
• Enhanced support for ETL using tools such as Flume. Impala ignores temporary files typically produced by these
tools (filenames with suffixes .copying and .tmp).
• The CPU requirement for Impala, which had become more restrictive in Impala 2.0.x and 2.1.x, has now been
relaxed.
The prerequisite for CPU architecture has been relaxed in Impala 2.2.0 and higher. From this release onward,
Impala works on CPUs that have the SSSE3 instruction set. The SSE4 instruction set is no longer required. This
relaxed requirement simplifies the upgrade planning from Impala 1.x releases, which also worked on SSSE3-enabled
processors.
• Enhanced support for CHAR and VARCHAR types in the COMPUTE STATS statement.
• The amount of memory required during setup for “spill to disk” operations is greatly reduced. This enhancement
reduces the chance of a memory-intensive join or aggregation query failing with an out-of-memory error.
• Several new conditional functions provide enhanced compatibility when porting code that uses industry extensions.
The new functions are: isfalse(), isnotfalse(), isnottrue(), istrue(), nonnullvalue(), and
nullvalue(). See Impala Conditional Functions on page 391 for details.
• The Impala debug web UI now can display a visual representation of the query plan. On the /queries tab, select
Details for a particular query. The Details page includes a Plan tab with a plan diagram that you can zoom in or
out (using scroll gestures through mouse wheel or trackpad).
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: Impala 2.1.3 is available as part of CDH 5.3.3, not under CDH 4.
Note: Impala 2.1.2 is available as part of CDH 5.3.2, not under CDH 4.
Note: Impala 2.0.5 is available as part of CDH 5.2.6, not under CDH 4.
Note: Impala 2.0.4 is available as part of CDH 5.2.5, not under CDH 4.
Note: Impala 2.0.3 is available as part of CDH 5.2.4, not under CDH 4.
Note: Impala 2.0.2 is available as part of CDH 5.2.3, not under CDH 4.
For details about subqueries, see Subqueries in Impala SELECT Statements on page 297 For information about new
and improved operators, see EXISTS Operator on page 193 and IN Operator on page 196.
• Analytic functions such as RANK(), LAG(), LEAD(), and FIRST_VALUE() let you analyze sequences of rows with
flexible ordering and grouping. Existing aggregate functions such as MAX(), SUM(), and COUNT() can also be used
in an analytic context. See Impala Analytic Functions on page 429 for details. See Impala Aggregate Functions on
page 403 for enhancements to existing aggregate functions.
• New data types provide greater compatibility with source code from traditional database systems:
– VARCHAR is like the STRING data type, but with a maximum length. See VARCHAR Data Type (CDH 5.2 or
higher only) on page 154 for details.
– CHAR is like the STRING data type, but with a precise length. Short values are padded with spaces on the
right. See CHAR Data Type (CDH 5.2 or higher only) on page 122 for details.
• Security enhancements:
• Formerly, Impala was restricted to using either Kerberos or LDAP / Active Directory authentication within a
cluster. Now, Impala can freely accept either kind of authentication request, allowing you to set up some
hosts with Kerberos authentication and others with LDAP or Active Directory. See Using Multiple Authentication
Methods with Impala on page 111 for details.
• GRANT statement. See GRANT Statement (CDH 5.2 or higher only) on page 263 for details.
• REVOKE statement. See REVOKE Statement (CDH 5.2 or higher only) on page 279 for details.
• CREATE ROLE statement. See CREATE ROLE Statement (CDH 5.2 or higher only) on page 235 for details.
• DROP ROLE statement. See DROP ROLE Statement (CDH 5.2 or higher only) on page 254 for details.
• SHOW ROLES and SHOW ROLE GRANT statements. See SHOW Statement on page 322 for details.
• To complement the HDFS encryption feature, a new Impala configuration option, --disk_spill_encryption
secures sensitive data from being observed or tampered with when temporarily stored on disk.
The new security-related SQL statements work along with the Sentry authorization framework. See Enabling Sentry
Authorization for Impala on page 95 for details.
• Impala can now read compressed text files compressed by gzip, bzip, or Snappy. These files do not require any
special table settings to work in an Impala text table. Impala recognizes the compression type automatically based
on file extensions of .gz, .bz2, and .snappy respectively. These types of compressed text files are intended for
convenience with existing ETL pipelines. Their non-splittable nature means they are not optimal for
high-performance parallel queries. See Using gzip, bzip2, or Snappy-Compressed Text Files on page 534 for details.
• Query hints can now use comment notation, /* +hint_name */ or -- +hint_name, at the same places in the
query where the hints enclosed by [ ] are recognized. This enhancement makes it easier to reuse Impala queries
on other database systems. See Query Hints in Impala SELECT Statements on page 301 for details.
• A new query option, QUERY_TIMEOUT_S, lets you specify a timeout period in seconds for individual queries.
The working of the --idle_query_timeout configuration option is extended. If no QUERY_OPTION_S query
option is in effect, --idle_query_timeout works the same as before, setting the timeout interval. When the
QUERY_OPTION_S query option is specified, its maximum value is capped by the value of the
--idle_query_timeout option.
That is, the system administrator sets the default and maximum timeout through the --idle_query_timeout
startup option, and then individual users or applications can set a lower timeout value if desired through the
QUERY_TIMEOUT_S query option. See Setting Timeout Periods for Daemons, Queries, and Sessions on page 84
and QUERY_TIMEOUT_S Query Option (CDH 5.2 or higher only) on page 320 for details.
• New functions VAR_SAMP() and VAR_POP() are aliases for the existing VARIANCE_SAMP() and VARIANCE_POP()
functions.
• A new date and time function, DATE_PART(), provides similar functionality to EXTRACT(). You can also call the
EXTRACT() function using the SQL-99 syntax, EXTRACT(unit FROM timestamp). These enhancements simplify
the porting process for date-related code from other systems. See Impala Date and Time Functions on page 363
for details.
• New approximation features provide a fast way to get results when absolute precision is not required:
– The APPX_COUNT_DISTINCT query option lets Impala rewrite COUNT(DISTINCT) calls to use NDV() instead,
which speeds up the operation and allows multiple COUNT(DISTINCT) operations in a single query. See
APPX_COUNT_DISTINCT Query Option (CDH 5.2 or higher only) on page 306 for details.
The APPX_MEDIAN() aggregate function produces an estimate for the median value of a column by using sampling.
See APPX_MEDIAN Function on page 403 for details.
• Impala now supports a DECODE() function. This function works as a shorthand for a CASE() expression, and
improves compatibility with SQL code containing vendor extensions. See Impala Conditional Functions on page
391 for details.
• The STDDEV(), STDDEV_POP(), STDDEV_SAMP(), VARIANCE(), VARIANCE_POP(), VARIANCE_SAMP(), and
NDV() aggregate functions now all return DOUBLE results rather than STRING. Formerly, you were required to
CAST() the result to a numeric type before using it in arithmetic operations.
• The default settings for Parquet block size, and the associated PARQUET_FILE_SIZE query option, are changed.
Now, Impala writes Parquet files with a size of 256 MB and an HDFS block size of 256 MB. Previously, Impala
attempted to write Parquet files with a size of 1 GB and an HDFS block size of 1 GB. In practice, Impala used a
conservative estimate of the disk space needed for each Parquet block, leading to files that were typically 512 MB
anyway. Thus, this change will make the file size more accurate if you specify a value for the PARQUET_FILE_SIZE
query option. It also reduces the amount of memory reserved during INSERT into Parquet tables, potentially
avoiding out-of-memory errors and improving scalability when inserting data into Parquet tables.
• Anti-joins are now supported, expressed using the LEFT ANTI JOIN and RIGHT ANTI JOIN clauses. These
clauses returns results from one table that have no match in the other table. You might use this type of join in
the same sorts of use cases as the NOT EXISTS and NOT IN operators. See Joins in Impala SELECT Statements
on page 282 for details.
• The SET command in impala-shell has been promoted to a real SQL statement. You can now set query options
such as PARQUET_FILE_SIZE, MEM_LIMIT, and SYNC_DDL within JDBC, ODBC, or any other kind of application
that submits SQL without going through the impala-shell interpreter. See SET Statement on page 304 for details.
• The impala-shell interpreter now reads settings from an optional configuration file, named $HOME/.impalarc
by default. See impala-shell Configuration File on page 473 for details.
• The library used for regular expression parsing has changed from Boost to Google RE2. This implementation change
adds support for non-greedy matches using the .*? notation. This and other changes in the way regular expressions
are interpreted means you might need to re-test queries that use functions such as regexp_extract() or
regexp_replace(), or operators such as REGEXP or RLIKE. See Incompatible Changes and Limitations in Apache
Impala on page 625 for those details.
Note: Impala 1.4.4 is available as part of CDH 5.1.5, not under CDH 4.
Note: Impala 1.4.3 is available as part of CDH 5.1.4, and under CDH 4.
Note: Impala 1.4.2 is only available as part of CDH 5.1.3, not under CDH 4.
• For interoperability with Parquet files created through other Hadoop components, such as Pig or MapReduce
jobs, you can create an Impala table that automatically sets up the column definitions based on the layout of an
existing Parquet data file.
• ORDER BY queries no longer require a LIMIT clause. If the size of the result set to be sorted exceeds the memory
available to Impala, Impala uses a temporary work space on disk to perform the sort operation.
• LDAP connections can be secured through either SSL or TLS.
• The following new built-in scalar and aggregate functions are available:
– A new built-in function, EXTRACT(), returns one date or time field from a TIMESTAMP value.
– A new built-in function, TRUNC(), truncates date/time values to a particular granularity, such as year, month,
day, hour, and so on.
– ADD_MONTHS() built-in function, an alias for the existing MONTHS_ADD() function.
– A new built-in function, ROUND(), rounds DECIMAL values to a specified number of fractional digits.
– Several built-in aggregate functions for computing properties for statistical distributions: STDDEV(),
STDDEV_SAMP(), STDDEV_POP(), VARIANCE(), VARIANCE_SAMP(), and VARIANCE_POP().
– Several new built-in functions, such as MAX_INT(), MIN_SMALLINT(), and so on, let you conveniently check
whether data values are in an expected range. You might be able to switch a column to a smaller type, saving
memory during processing.
– New built-in functions, IS_INF() and IS_NAN(), check for the special values infinity and “not a number”.
These values could be specified as inf or nan in text data files, or be produced by certain arithmetic
expressions.
• The SHOW PARTITIONS statement displays information about the structure of a partitioned table.
• New configuration options for the impalad daemon let you specify initial memory usage for all queries. The initial
resource requests handled by Llama and YARN can be expanded later if needed, avoiding unnecessary over-allocation
and reducing the chance of out-of-memory conditions.
• Impala can take advantage of the Llama high availability feature in CDH 5.1, for improved reliability of resource
management through YARN.
• The Impala CREATE TABLE statement now has a STORED AS AVRO clause, allowing you to create Avro tables
through Impala.
• New impalad configuration options let you fine-tune the calculations Impala makes to estimate resource
requirements for each query. These options can help avoid problems due to overconsumption due to too-low
estimates, or underutilization due to too-high estimates.
• A new SUMMARY command in the impala-shell interpreter provides a high-level summary of the work performed
at each stage of the explain plan. The summary is also included in output from the PROFILE command.
• Performance improvements for the COMPUTE STATS statement:
– The NDV function is speeded up through native code generation.
– Because the NULL count is not currently used by the Impala query planner, in Impala 1.4.0 and higher, COMPUTE
STATS does not count the NULL values for each column. (The #Nulls field of the stats table is left as -1,
signifying that the value is unknown.)
• Performance improvements for partition pruning. This feature reduces the time spent in query planning, for
partitioned tables with thousands of partitions. Previously, Impala typically queried tables with up to approximately
3000 partitions. With the performance improvement in partition pruning, now Impala can comfortably handle
tables with tens of thousands of partitions.
• The documentation provides additional guidance for planning tasks.
• The impala-shell interpreter now supports UTF-8 characters for input and output. You can control whether
impala-shell ignores invalid Unicode code points through the --strict_unicode option. (Although this
option is removed in Impala 2.0.)
Note: Impala 1.3.3 is only available as part of CDH 5.0.5, not under CDH 4.
Note: Impala 1.3.2 is only available as part of CDH 5.0.4, not under CDH 4.
Note:
• The Impala 1.3.1 release is available for both CDH 4 and CDH 5. This is the first release in the 1.3.x
series for CDH 4.
• A new impalad startup option, --insert_inherit_permissions, causes Impala INSERT statements to create
each new partition with the same HDFS permissions as its parent directory. By default, INSERT statements create
directories for new partitions using default HDFS permissions. See INSERT Statement on page 263 for examples of
INSERT statements for partitioned tables.
• The SHOW FUNCTIONS statement now displays the return type of each function, in addition to the types of its
arguments. See SHOW Statement on page 322 for examples.
• You can now specify the clause FIELDS TERMINATED BY '\0' with a CREATE TABLE statement to use text
data files that use ASCII 0 (nul) characters as a delimiter. See Using Text Data Files with Impala Tables on page
528 for details.
• In Impala 1.3.1 and higher, the REGEXP and RLIKE operators now match a regular expression string that occurs
anywhere inside the target string, the same as if the regular expression was enclosed on each side by .*. See
REGEXP Operator on page 202 for examples. Previously, these operators only succeeded when the regular expression
matched the entire target string. This change improves compatibility with the regular expression support for
popular database systems. There is no change to the behavior of the regexp_extract() and regexp_replace()
built-in functions.
Note:
• The Impala 1.3.1 release is available for both CDH 4 and CDH 5. This is the first release in the 1.3.x
series for CDH 4.
• The admission control feature lets you control and prioritize the volume and resource consumption of concurrent
queries. This mechanism reduces spikes in resource usage, helping Impala to run alongside other kinds of workloads
on a busy cluster. It also provides more user-friendly conflict resolution when multiple memory-intensive queries
are submitted concurrently, avoiding resource contention that formerly resulted in out-of-memory errors. See
Admission Control and Query Queuing on page 75 for details.
• Enhanced EXPLAIN plans provide more detail in an easier-to-read format. Now there are four levels of verbosity:
the EXPLAIN_LEVEL option can be set from 0 (most concise) to 3 (most verbose). See EXPLAIN Statement on
page 260 for syntax and Understanding Impala Query Performance - EXPLAIN Plans and Query Profiles on page 507
for usage information.
• The TIMESTAMP data type accepts more kinds of input string formats through the UNIX_TIMESTAMP function,
and produces more varieties of string formats through the FROM_UNIXTIME function. The documentation now
also lists more functions for date arithmetic, used for adding and subtracting INTERVAL expressions from
TIMESTAMP values. See Impala Date and Time Functions on page 363 for details.
• New conditional functions, NULLIF(), NULLIFZERO(), and ZEROIFNULL(), simplify porting SQL containing
vendor extensions to Impala. See Impala Conditional Functions on page 391 for details.
• New utility function, CURRENT_DATABASE(). See Impala Miscellaneous Functions on page 403 for details.
• Integration with the YARN resource management framework. Only available in combination with CDH 5. This
feature makes use of the underlying YARN service, plus an additional service (Llama) that coordinates requests
to YARN for Impala resources, so that the Impala query only proceeds when all requested resources are available.
See Integrated Resource Management with YARN on page 83 for full details.
On the Impala side, this feature involves some new startup options for the impalad daemon:
– -enable_rm
– -llama_host
– -llama_port
– -llama_callback_port
– -cgroup_hierarchy_path
For details of these startup options, see Modifying Impala Startup Options on page 43.
This feature also involves several new or changed query options that you can set through the impala-shell
interpreter and apply within a specific session:
– MEM_LIMIT: the function of this existing option changes when Impala resource management is enabled.
– REQUEST_POOL: a new option. (Renamed to RESOURCE_POOL in Impala 1.3.0.)
– V_CPU_CORES: a new option.
– RESERVATION_REQUEST_TIMEOUT: a new option.
For details of these query options, see impala-shell Query Options for Resource Management on page 84.
Note: Impala 1.2.4 works with CDH 4. It is primarily a bug fix release for Impala 1.2.3, plus some
performance enhancements for the catalog server to minimize startup and DDL wait times for Impala
deployments with large numbers of databases, tables, and partitions.
• On Impala startup, the metadata loading and synchronization mechanism has been improved and optimized, to
give more responsiveness when starting Impala on a system with a large number of databases, tables, or partitions.
The initial metadata loading happens in the background, allowing queries to be run before the entire process is
finished. When a query refers to a table whose metadata is not yet loaded, the query waits until the metadata
for that table is loaded, and the load operation for that table is prioritized to happen first.
• Formerly, if you created a new table in Hive, you had to issue the INVALIDATE METADATA statement (with no
table name) which was an expensive operation that reloaded metadata for all tables. Impala did not recognize
the name of the Hive-created table, so you could not do INVALIDATE METADATA new_table to get the metadata
for just that one table. Now, when you issue INVALIDATE METADATA table_name, Impala checks to see if that
name represents a table created in Hive, and if so recognizes the new table and loads the metadata for it.
Additionally, if the new table is in a database that was newly created in Hive, Impala also recognizes the new
database.
• If you issue INVALIDATE METADATA table_name and the table has been dropped through Hive, Impala will
recognize that the table no longer exists.
• New startup options let you control the parallelism of the metadata loading during startup for the catalogd
daemon:
– --load_catalog_in_background makes Impala load and cache metadata using background threads after
startup. It is true by default. Previously, a system with a large number of databases, tables, or partitions
could be unresponsive or even time out during startup.
– --num_metadata_loading_threads determines how much parallelism Impala devotes to loading metadata
in the background. The default is 16. You might increase this value for systems with huge numbers of databases,
tables, or partitions. You might lower this value for busy systems that are CPU-constrained due to jobs from
components other than Impala.
Note: Impala 1.2.3 works with CDH 4 and with CDH 5 beta 2. The resource management feature
requires CDH 5 beta.
Impala 1.2.3 contains exactly the same feature set as Impala 1.2.2. Its only difference is one additional fix for compatibility
with Parquet files generated outside of Impala by components such as Hive, Pig, or MapReduce. If you are upgrading
from Impala 1.2.1 or earlier, see New Features in Impala Version 1.2.2 on page 618 for the latest added features.
Note: Impala 1.2.2 works with CDH 4. Its feature set is a superset of features in the Impala 1.2.0 beta,
with the exception of resource management, which relies on CDH 5.
Impala 1.2.2 includes new features for performance, security, and flexibility. The major enhancements over 1.2.1 are
performance related, primarily for join queries.
New user-visible features include:
• Join order optimizations. This highly valuable feature automatically distributes and parallelizes the work for a join
query to minimize disk I/O and network traffic. The automatic optimization reduces the need to use query hints
or to rewrite join queries with the tables in a specific order based on size or cardinality. The new COMPUTE STATS
statement gathers statistical information about each table that is crucial for enabling the join optimizations. See
Performance Considerations for Join Queries on page 482 for details.
• COMPUTE STATS statement to collect both table statistics and column statistics with a single statement. Intended
to be more comprehensive, efficient, and reliable than the corresponding Hive ANALYZE TABLE statement, which
collects statistics in multiple phases through MapReduce jobs. These statistics are important for query planning
for join queries, queries on partitioned tables, and other types of data-intensive operations. For optimal planning
of join queries, you need to collect statistics for each table involved in the join. See COMPUTE STATS Statement
on page 226 for details.
• Reordering of tables in a join query can be overridden by the STRAIGHT_JOIN operator, allowing you to fine-tune
the planning of the join query if necessary, by using the original technique of ordering the joined tables in descending
order of size. See Overriding Join Reordering with STRAIGHT_JOIN on page 483 for details.
• The CROSS JOIN clause in the SELECT statement to allow Cartesian products in queries, that is, joins without an
equality comparison between columns in both tables. Because such queries must be carefully checked to avoid
accidental overconsumption of memory, you must use the CROSS JOIN operator to explicitly select this kind of
join. See Cross Joins and Cartesian Products with the CROSS JOIN Operator on page 61 for examples.
• The ALTER TABLE statement has new clauses that let you fine-tune table statistics. You can use this technique
as a less-expensive way to update specific statistics, in case the statistics become stale, or to experiment with the
effects of different data distributions on query planning.
• LDAP username/password authentication in JDBC/ODBC. See Enabling LDAP Authentication for Impala on page
109 for details.
• GROUP_CONCAT() aggregate function to concatenate column values across all rows of a result set.
• The INSERT statement now accepts hints, [SHUFFLE] and [NOSHUFFLE], to influence the way work is redistributed
during INSERT...SELECT operations. The hints are primarily useful for inserting into partitioned Parquet tables,
where using the [SHUFFLE] hint can avoid problems due to memory consumption and simultaneous open files
in HDFS, by collecting all the new data for each partition on a specific node.
• Several built-in functions and operators are now overloaded for more numeric data types, to reduce the requirement
to use CAST() for type coercion in INSERT statements. For example, the expression 2+2 in an INSERT statement
formerly produced a BIGINT result, requiring a CAST() to be stored in an INT variable. Now, addition, subtraction,
and multiplication only produce a result that is one step “bigger” than their arguments, and numeric and conditional
functions can return SMALLINT, FLOAT, and other smaller types rather than always BIGINT or DOUBLE.
• New fnv_hash() built-in function for constructing hashed values. See Impala Mathematical Functions on page
339 for details.
• The clause STORED AS PARQUET is accepted as an equivalent for STORED AS PARQUETFILE. This more concise
form is recommended for new code.
Because Impala 1.2.2 builds on a number of features introduced in 1.2.1, if you are upgrading from an older 1.1.x
release straight to 1.2.2, also review New Features in Impala Version 1.2.1 on page 619 to see features such as the SHOW
TABLE STATS and SHOW COLUMN STATS statements, and user-defined functions (UDFs).
Note: Impala 1.2.1 works with CDH 4. Its feature set is a superset of features in the Impala 1.2.0 beta,
with the exception of resource management, which relies on CDH 5.
Impala 1.2.1 includes new features for security, performance, and flexibility.
New user-visible features include:
• SHOW TABLE STATS table_name and SHOW COLUMN STATS table_name statements, to verify that statistics
are available and to see the values used during query planning.
• CREATE TABLE AS SELECT syntax, to create a new table and transfer data into it in a single operation.
• OFFSET clause, for use with the ORDER BY and LIMIT clauses to produce “paged” result sets such as items 1-10,
then 11-20, and so on.
• NULLS FIRST and NULLS LAST clauses to ensure consistent placement of NULL values in ORDER BY queries.
For even more precise synchronization, you can enable the SYNC_DDL query option before issuing a DDL, INSERT,
or LOAD DATA statement. This option causes the statement to wait, returning only after the catalog service has
broadcast the applicable changes to all Impala nodes in the cluster.
Note:
Because the catalog service only monitors operations performed through Impala, INVALIDATE
METADATA and REFRESH are still needed on the Impala side after creating new tables or loading
data through the Hive shell or by manipulating data files directly in HDFS. Because the catalog
service broadcasts the result of the REFRESH and INVALIDATE METADATA statements to all
Impala nodes, when you do need to use those statements, you can do so a single time rather
than on every Impala node.
This service is implemented by the catalogd daemon. See The Impala Catalog Service on page 19 for details.
• CREATE TABLE ... AS SELECT syntax, to create a table and copy data into it in a single operation. See CREATE
TABLE Statement on page 235 for details.
• The CREATE TABLE and ALTER TABLE statements have new clauses TBLPROPERTIES and WITH
SERDEPROPERTIES. The TBLPROPERTIES clause lets you associate arbitrary items of metadata with a particular
table as key-value pairs. The WITH SERDEPROPERTIES clause lets you specify the serializer/deserializer (SerDes)
classes that read and write data for a table; although Impala does not make use of these properties, sometimes
particular values are needed for Hive compatibility. See CREATE TABLE Statement on page 235 and ALTER TABLE
Statement on page 215 for details.
• Delegation support lets you authorize certain OS users associated with applications (for example, hue), to submit
requests using the credentials of other users. Only available in combination with CDH 5. See Configuring Impala
Delegation for Hue and BI Tools on page 112 for details.
• Enhancements to EXPLAIN output. In particular, when you enable the new EXPLAIN_LEVEL query option, the
EXPLAIN and PROFILE statements produce more verbose output showing estimated resource requirements and
whether table and column statistics are available for the applicable tables and columns. See EXPLAIN Statement
on page 260 for details.
• SHOW CREATE TABLE summarizes the effects of the original CREATE TABLE statement and any subsequent
ALTER TABLE statements, giving you a CREATE TABLE statement that will re-create the current structure and
layout for a table.
• The LIMIT clause for queries now accepts an arithmetic expression, in addition to numeric literals.
Note: The Impala 1.2.0 beta release only works in combination with the beta version of CDH 5. The
Impala 1.2.0 software is bundled together with the CDH 5 beta 1 download.
The Impala 1.2.0 beta includes new features for security, performance, and flexibility.
New user-visible features include:
• User-defined functions (UDFs). This feature lets you transform data in very flexible ways, which is important when
using Impala as part of an ETL or ELT pipeline. Prior to Impala 1.2, using UDFs required switching into Hive. Impala
1.2 can run scalar UDFs and user-defined aggregate functions (UDAs). Impala can run high-performance functions
written in C++, or you can reuse existing Hive functions written in Java.
You create UDFs through the CREATE FUNCTION statement and drop them through the DROP FUNCTION
statement. See Impala User-Defined Functions (UDFs) on page 447 for instructions about coding, building, and
deploying UDFs, and CREATE FUNCTION Statement on page 232 and DROP FUNCTION Statement on page 254 for
related SQL syntax.
• A new service automatically propagates changes to table data and metadata made by one Impala node, sending
the new or updated metadata to all the other Impala nodes. The automatic synchronization mechanism eliminates
the need to use the INVALIDATE METADATA and REFRESH statements after issuing Impala statements such as
CREATE TABLE, ALTER TABLE, DROP TABLE, INSERT, and LOAD DATA.
Note:
Because this service only monitors operations performed through Impala, INVALIDATE METADATA
and REFRESH are still needed on the Impala side after creating new tables or loading data through
the Hive shell or by manipulating data files directly in HDFS. Because the catalog service broadcasts
the result of the REFRESH and INVALIDATE METADATA statements to all Impala nodes, when
you do need to use those statements, you can do so a single time rather than on every Impala
node.
This service is implemented by the catalogd daemon. See The Impala Catalog Service on page 19 for details.
• Integration with the YARN resource management framework. Only available in combination with CDH 5. This
feature makes use of the underlying YARN service, plus an additional service (Llama) that coordinates requests
to YARN for Impala resources, so that the Impala query only proceeds when all requested resources are available.
See Integrated Resource Management with YARN on page 83 for full details.
On the Impala side, this feature involves some new startup options for the impalad daemon:
– -enable_rm
– -llama_host
– -llama_port
– -llama_callback_port
– -cgroup_hierarchy_path
For details of these startup options, see Modifying Impala Startup Options on page 43.
This feature also involves several new or changed query options that you can set through the impala-shell
interpreter and apply within a specific session:
– MEM_LIMIT: the function of this existing option changes when Impala resource management is enabled.
– YARN_POOL: a new option. (Renamed to RESOURCE_POOL in Impala 1.3.0.)
– V_CPU_CORES: a new option.
– RESERVATION_REQUEST_TIMEOUT: a new option.
For details of these query options, see impala-shell Query Options for Resource Management on page 84.
• CREATE TABLE ... AS SELECT syntax, to create a table and copy data into it in a single operation. See CREATE
TABLE Statement on page 235 for details.
• The CREATE TABLE and ALTER TABLE statements have a new TBLPROPERTIES clause that lets you associate
arbitrary items of metadata with a particular table as key-value pairs. See CREATE TABLE Statement on page 235
and ALTER TABLE Statement on page 215 for details.
• Delegation support lets you authorize certain OS users associated with applications (for example, hue), to submit
requests using the credentials of other users. Only available in combination with CDH 5. See Configuring Impala
Delegation for Hue and BI Tools on page 112 for details.
• Enhancements to EXPLAIN output. In particular, when you enable the new EXPLAIN_LEVEL query option, the
EXPLAIN and PROFILE statements produce more verbose output showing estimated resource requirements and
whether table and column statistics are available for the applicable tables and columns. See EXPLAIN Statement
on page 260 for details.
• Performance is improved through a number of automatic optimizations. Resource consumption is also reduced
for Impala queries. These improvements apply broadly across all kinds of workloads and file formats. The major
areas of performance enhancement include:
– Improved disk and thread scheduling, which applies to all queries.
– Improved hash join and aggregation performance, which applies to queries with large build tables or a large
number of groups.
– Dictionary encoding with Parquet, which applies to Parquet tables with short string columns.
– Improved performance on systems with SSDs, which applies to all queries and file formats.
• Some new built-in functions are implemented: translate() to substitute characters within strings, user() to check
the login ID of the connected user.
• The new WITH clause for SELECT statements lets you simplify complicated queries in a way similar to creating a
view. The effects of the WITH clause only last for the duration of one query, unlike views, which are persistent
schema objects that can be used by multiple sessions or applications. See WITH Clause on page 300.
• An enhancement to DESCRIBE statement, DESCRIBE FORMATTED table_name, displays more detailed
information about the table. This information includes the file format, location, delimiter, ownership, external or
internal, creation and access times, and partitions. The information is returned as a result set that can be interpreted
and used by a management or monitoring application. See DESCRIBE Statement on page 245.
• You can now insert a subset of columns for a table, with other columns being left as all NULL values. Or you can
specify the columns in any order in the destination table, rather than having to match the order of the corresponding
columns in the source. VALUES clause. This feature is known as “column permutation”. See INSERT Statement on
page 263.
• The new LOAD DATA statement lets you load data into a table directly from an HDFS data file. This technique lets
you minimize the number of steps in your ETL process, and provides more flexibility. For example, you can bring
data into an Impala table in one step. Formerly, you might have created an external table where the data files are
not entirely under your control, or copied the data files to Impala data directories manually, or loaded the original
data into one table and then used the INSERT statement to copy it to a new table with a different file format,
partitioning scheme, and so on. See LOAD DATA Statement on page 274.
• Improvements to Impala-HBase integration:
– New query options for HBase performance: HBASE_CACHE_BLOCKS and HBASE_CACHING.
– Support for binary data types in HBase tables. See Supported Data Types for HBase Columns on page 558 for
details.
• You can issue REFRESH as a SQL statement through any of the programming interfaces that Impala supports.
REFRESH formerly had to be issued as a command through the impala-shell interpreter, and was not available
through a JDBC or ODBC API call. As part of this change, the functionality of the REFRESH statement is divided
between two statements. In Impala 1.1, REFRESH requires a table name argument and immediately reloads the
metadata; the new INVALIDATE METADATA statement works the same as the Impala 1.0 REFRESH did: the table
name argument is optional, and the metadata for one or all tables is marked as stale, but not actually reloaded
until the table is queried. When you create a new table in the Hive shell or through a different Impala node, you
must enter INVALIDATE METADATA with no table parameter before you can see the new table in impala-shell.
See REFRESH Statement on page 277 and INVALIDATE METADATA Statement on page 271.
benchmark tests, it is also faster to produce plain rather than pretty-printed results, and write to a file rather than
to the screen, giving a more accurate picture of the actual query time.
• Several bug fixes. See Issues Fixed in the 1.0.1 Release on page 679 for details.
• The Impala debug webserver now has the ability to serve static files from ${IMPALA_HOME}/www. This can be
disabled by setting --enable_webserver_doc_root=false on the command line. As a result, Impala now
uses the Twitter Bootstrap library to style its debug webpages, and the /queries page now tracks the last 25
queries run by each Impala daemon.
• Additional metrics available on the Impala Debug Webpage.
-default_query_options='key=value;key=value'
Note: Starting in April 2016, future release note updates are being consolidated in a single location
to avoid duplication of stale or incomplete information. You can view online the Impala New Features,
Incompatible Changes, Known Issues, and Fixed Issues. You can view or print all of these by downloading
the latest Impala PDF.
Note: Though Impala can be used together with YARN via simple configuration of Static Service Pools
in Cloudera Manager, the use of the general-purpose component Llama for integrated resource
management within YARN is no longer supported with CDH 5.5 / Impala 2.3 and higher.
• If Impala encounters a Parquet file that is invalid because of an incorrect magic number, the query skips the file.
This change is caused by the fix for issue IMPALA-2130. Previously, Impala would attempt to read the file despite
the possibility that the file was corrupted.
• Previously, calls to overloaded built-in functions could treat parameters as DOUBLE or FLOAT when no overload
had a signature that matched the exact argument types. Now Impala prefers the function signature with DECIMAL
parameters in this case. This change avoids a possible loss of precision in function calls such as greatest(0,
99999.8888); now both parameters are treated as DECIMAL rather than DOUBLE, avoiding any loss of precision
in the fractional value. This could cause slightly different results than in previous Impala releases for certain
function calls.
• Formerly, adding or subtracting a large interval value to a TIMESTAMP could produce a nonsensical result. Now
when the result goes outside the range of TIMESTAMP values, Impala returns NULL.
• Formerly, it was possible to accidentally create a table with identical row and column delimiters. This could happen
unintentionally, when specifying one of the delimiters and using the default value for the other. Now an attempt
to use identical delimiters still succeeds, but displays a warning message.
• Formerly, Impala could include snippets of table data in log files by default, for example when reporting conversion
errors for data values. Now any such log messages are only produced at higher logging levels that you would
enable only during debugging.
Incompatible Changes Introduced in Impala for CDH 5.4.x (CDH 5.4.1 through CDH 5.4.9)
No incompatible changes. CDH maintenance releases such as 5.4.1, 5.4.2, and so on are exclusively bug fix releases.
See Incompatible Changes Introduced in Impala 2.2.0 / CDH 5.4.0 on page 626 for the most Impala feature release,
which are the releases that typically include incompatible changes.
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
Note: Impala 2.2.0 is available as part of CDH 5.4.0 and is not available for CDH 4. Cloudera does not
intend to release future versions of Impala for CDH 4 outside patch and maintenance releases if
required. Given the end-of-maintenance status for CDH 4, Cloudera recommends all customers to
migrate to a recent CDH 5 release.
Changes to Prerequisites
The prerequisite for CPU architecture has been relaxed in Impala 2.2.0 and higher. From this release onward, Impala
works on CPUs that have the SSSE3 instruction set. The SSE4 instruction set is no longer required. This relaxed
requirement simplifies the upgrade planning from Impala 1.x releases, which also worked on SSSE3-enabled processors.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: Impala 2.1.3 is available as part of CDH 5.3.3, not under CDH 4.
Note: Impala 2.1.2 is available as part of CDH 5.3.2, not under CDH 4.
Changes to Prerequisites
Currently, Impala 2.1.x does not function on CPUs without the SSE4.1 instruction set. This minimum CPU requirement
is higher than in previous versions, which relied on the older SSSE3 instruction set. Check the CPU level of the hosts in
your cluster before upgrading to Impala 2.1.x or CDH 5.3.x.
Note: Impala 2.0.5 is available as part of CDH 5.2.6, not under CDH 4.
Note: Impala 2.0.4 is available as part of CDH 5.2.5, not under CDH 4.
Note: Impala 2.0.3 is available as part of CDH 5.2.4, not under CDH 4.
Note: Impala 2.0.2 is available as part of CDH 5.2.3, not under CDH 4.
Changes to Prerequisites
Currently, Impala 2.0.x does not function on CPUs without the SSE4.1 instruction set. This minimum CPU requirement
is higher than in previous versions, which relied on the older SSSE3 instruction set. Check the CPU level of the hosts in
your cluster before upgrading to Impala 2.0.x or CDH 5.2.x.
The --strict_unicode option of impala-shell was removed. To avoid problems with Unicode values in
impala-shell, define the following locale setting before running impala-shell:
export LC_CTYPE=en_US.UTF-8
Note: Impala 1.4.4 is available as part of CDH 5.1.5, not under CDH 4.
Note: Impala 1.4.3 is available as part of CDH 5.1.4, and under CDH 4.
Note: Impala 1.4.2 is only available as part of CDH 5.1.3, not under CDH 4.
– CACHED
– CLASS
– PARTITIONS
– PRODUCED
– UNCACHED
The following were formerly reserved keywords, but are no longer reserved:
– COUNT
– GROUP_CONCAT
– NDV
– SUM
• The fix for issue IMPALA-973 changes the behavior of the INVALIDATE METADATA statement regarding nonexistent
tables. In Impala 1.4.0 and higher, the statement returns an error if the specified table is not in the metastore
database at all. It completes successfully if the specified table is in the metastore database but not yet recognized
by Impala, for example if the table was created through Hive. Formerly, you could issue this statement for a
completely nonexistent table, with no error.
Note: Impala 1.3.3 is only available as part of CDH 5.0.5, not under CDH 4.
Note: Impala 1.3.2 is only available as part of CDH 5.0.4, not under CDH 4.
Note: Although the DECIMAL keyword is a reserved word, currently Impala does not support
DECIMAL as a data type for columns.
• The query option named YARN_POOL during the CDH 5 beta period is now named REQUEST_POOL to reflect its
broader use with the Impala admission control feature.
• There are some changes to the list of reserved words.
– The names of aggregate functions are no longer reserved words, so you can have databases, tables, columns,
or other objects named AVG, MIN, and so on without any name conflicts.
– The internal function names DISTINCTPC and DISTINCTPCSA are no longer reserved words, although
DISTINCT is still a reserved word.
• Formerly, a DROP DATABASE statement in Impala would not remove the top-level HDFS directory for that database.
The DROP DATABASE has been enhanced to remove that directory. (You still need to drop all the tables inside
the database first; this change only applies to the top-level directory for the entire database.)
• The keyword PARQUET is introduced as a synonym for PARQUETFILE in the CREATE TABLE and ALTER TABLE
statements, because that is the common name for the file format. (As opposed to SequenceFile and RCFile where
the “File” suffix is part of the name.) Documentation examples have been changed to prefer the new shorter
keyword. The PARQUETFILE keyword is still available for backward compatibility with older Impala versions.
• New overloads are available for several operators and built-in functions, allowing you to insert their result values
into smaller numeric columns such as INT, SMALLINT, TINYINT, and FLOAT without using a CAST() call. If you
remove the CAST() calls from INSERT statements, those statements might not work with earlier versions of
Impala.
Because many users are likely to upgrade straight from Impala 1.x to Impala 1.2.2, also read Incompatible Changes
Introduced in Impala 1.2.1 on page 633 for things to note about upgrading to Impala 1.2.x in general.
In a Cloudera Manager environment, the catalog service is not recognized or managed by Cloudera Manager versions
prior to 4.8. Cloudera Manager 4.8 and higher require the catalog service to be present for Impala. Therefore, if you
upgrade to Cloudera Manager 4.8 or higher, you must also upgrade Impala to 1.2.1 or higher. Likewise, if you upgrade
Impala to 1.2.1 or higher, you must also upgrade Cloudera Manager to 4.8 or higher.
Impala 1.2.1 goes along with CDH 4.5 and Cloudera Manager 4.8. If you used the beta version Impala 1.2.0 that came
with the beta of CDH 5, Impala 1.2.1 includes all the features of Impala 1.2.0 except for resource management, which
relies on the YARN framework from CDH 5.
The new catalogd service might require changes to any user-written scripts that stop, start, or restart Impala services,
install or upgrade Impala packages, or issue REFRESH or INVALIDATE METADATA statements:
• See Installing Impala on page 27, Upgrading Impala on page 38 and Starting Impala on page 42, for usage
information for the catalogd daemon.
• The REFRESH and INVALIDATE METADATA statements are no longer needed when the CREATE TABLE, INSERT,
or other table-changing or data-changing operation is performed through Impala. These statements are still needed
if such operations are done through Hive or by manipulating data files directly in HDFS, but in those cases the
statements only need to be issued on one Impala node rather than on all nodes. See REFRESH Statement on page
277 and INVALIDATE METADATA Statement on page 271 for the latest usage information for those statements.
• See The Impala Catalog Service on page 19 for background information on the catalogd service.
In a Cloudera Manager environment, the catalog service is not recognized or managed by Cloudera Manager versions
prior to 4.8. Cloudera Manager 4.8 and higher require the catalog service to be present for Impala. Therefore, if you
upgrade to Cloudera Manager 4.8 or higher, you must also upgrade Impala to 1.2.1 or higher. Likewise, if you upgrade
Impala to 1.2.1 or higher, you must also upgrade Cloudera Manager to 4.8 or higher.
Because Impala 1.2.0 is bundled with the CDH 5 beta download and depends on specific levels of Apache Hadoop
components supplied with CDH 5, you can only install it in combination with the CDH 5 beta.
The new catalogd service might require changes to any user-written scripts that stop, start, or restart Impala services,
install or upgrade Impala packages, or issue REFRESH or INVALIDATE METADATA statements:
• See Installing Impala on page 27, Upgrading Impala on page 38 and Starting Impala on page 42, for usage
information for the catalogd daemon.
• The REFRESH and INVALIDATE METADATA statements are no longer needed when the CREATE TABLE, INSERT,
or other table-changing or data-changing operation is performed through Impala. These statements are still needed
if such operations are done through Hive or by manipulating data files directly in HDFS, but in those cases the
statements only need to be issued on one Impala node rather than on all nodes. See REFRESH Statement on page
277 and INVALIDATE METADATA Statement on page 271 for the latest usage information for those statements.
• See The Impala Catalog Service on page 19 for background information on the catalogd service.
The new resource management feature interacts with both YARN and Llama services, which are available in CDH 5.
These services are set up for you automatically in a Cloudera Manager (CM) environment. For information about setting
up the YARN and Llama services, see the instructions for YARN and Llama in the CDH 5 Documentation.
If you are running a level of Impala that is older than 1.1.1, do the metadata update through Hive:
Impala 1.1.1 and higher can reuse Parquet data files created by Hive, without any action required.
As usual, make sure to upgrade the Impala LZO support package to the latest level at the same time as you upgrade
the Impala server.
• Cloudera Manager 4.5.2 and higher only supports Impala 1.0 and higher, and vice versa. If you upgrade to Impala
1.0 or higher managed by Cloudera Manager, you must also upgrade Cloudera Manager to version 4.5.2 or higher.
If you upgrade from an earlier version of Cloudera Manager, and were using Impala, you must also upgrade Impala
to version 1.0 or higher. The beta versions of Impala are no longer supported as of the release of Impala 1.0.
Incompatible Change Introduced in Version 0.7 of the Cloudera Impala Beta Release
• The defaults for the -nn and -nn_port flags have changed and are now read from core-site.xml. Impala
prints the values of -nn and -nn_port to the log when it starts. The ability to set -nn and -nn_port on the
command line is deprecated in 0.7 and may be removed in Impala 0.8.
Incompatible Change Introduced in Version 0.6 of the Cloudera Impala Beta Release
• Cloudera Manager 4.5 supports only version 0.6 of the Cloudera Impala Beta Release. It does not support the
earlier beta versions. If you upgrade your Cloudera Manager installation, you must also upgrade Impala to beta
version 0.6. If you upgrade Impala to beta version 0.6, you must upgrade Cloudera Manager to 4.5.
Incompatible Change Introduced in Version 0.4 of the Cloudera Impala Beta Release
• Cloudera Manager 4.1.3 supports only version 0.4 of the Cloudera Impala Beta Release. It does not support the
earlier beta versions. If you upgrade your Cloudera Manager installation, you must also upgrade Impala to beta
version 0.4. If you upgrade Impala to beta version 0.4, you must upgrade Cloudera Manager to 4.1.3.
Incompatible Change Introduced in Version 0.3 of the Cloudera Impala Beta Release
• Cloudera Manager 4.1.2 supports only version 0.3 of the Cloudera Impala Beta Release. It does not support the
earlier beta versions. If you upgrade your Cloudera Manager installation, you must also upgrade Impala to beta
version 0.3. If you upgrade Impala to beta version 0.3, you must upgrade Cloudera Manager to 4.1.2.
Note: The online issue tracking system for Impala contains comprehensive information and is updated
in real time. To verify whether an issue you are experiencing has already been reported, or which
release an issue is fixed in, search on the issues.cloudera.org JIRA tracker.
For issues fixed in various Impala releases, see Fixed Issues in Apache Impala on page 643.
Note: Starting in April 2016, future release note updates are being consolidated in a single location
to avoid duplication of stale or incomplete information. You can view online the Impala New Features,
Incompatible Changes, Known Issues, and Fixed Issues. You can view or print all of these by downloading
the latest Impala PDF.
Impalad is crashing if udf jar is not available in hdfs location for first time
If the JAR file corresponding to a Java UDF is removed from HDFS after the Impala CREATE FUNCTION statement is
issued, the impalad daemon crashes.
Bug: IMPALA-2365
Severity: High
ImpalaODBC: Can not get the value in the SQLGetData(m-x th column) after the SQLBindCol(m th column)
If the ODBC SQLGetData is called on a series of columns, the function calls must follow the same order as the columns.
For example, if data is fetched from column 2 then column 1, the SQLGetData call for column 1 returns NULL.
Bug: IMPALA-1792
Severity: High
Workaround: Fetch columns in the same order they are defined in the table.
Unable to import the python 'ssl' module. It is required for an SSL-secured connection.
Resolution: Customers are less likely to experience this issue over time, because ssl module is included in newer
Python releases packaged with recent Linux releases.
Workaround: To use SSL with impala-shell on these platform versions, install the ssh Python module:
Process mem limit does not account for the JVM's memory usage
Some memory allocated by the JVM used internally by Impala is not counted against the memory limit for the impalad
daemon.
Bug: IMPALA-691
Severity: High
Workaround: To monitor overall memory usage, use the top command, or add the memory figures in the Impala web
UI /memz tab to JVM memory usage shown on the /metrics tab.
Fix issues with the legacy join and agg nodes using --enable_partitioned_hash_join=false and
--enable_partitioned_aggregation=false
Bug: IMPALA-2375
Severity: High
Workaround: Transition away from the “old-style” join and aggregation mechanism if practical.
% escaping does not work correctly when occurs at the end in a LIKE clause
If the final character in the RHS argument of a LIKE operator is an escaped \% character, it does not match a % final
character of the LHS argument.
Bug: IMPALA-2422
Severity: High
Duplicated column in inline view causes dropping null slots during scan
If the same column is queried twice within a view, NULL values for that column are omitted. For example, the result
of COUNT(*) on the view could be less than expected.
Bug: IMPALA-2643
Severity: High
Workaround: Avoid selecting the same column twice within an inline view.
Crash: impala::Coordinator::ValidateCollectionSlots
A query could encounter a serious error if includes multiple nested levels of INNER JOIN clauses involving subqueries.
Bug: IMPALA-2603
Severity: High
Incorrect assignment of On-clause predicate inside inline view with an outer join.
A query might return incorrect results due to wrong predicate assignment in the following scenario:
1. There is an inline view that contains an outer join
2. That inline view is joined with another table in the enclosing query block
3. That join has an On-clause containing a predicate that only references columns originating from the outer-joined
tables inside the inline view
Bug: IMPALA-2665
Severity: High
Wrong plan of NOT IN aggregate subquery when a constant is used in subquery predicate
A NOT IN operator with a subquery that calls an aggregate function, such as NOT IN (SELECT SUM(...)), could
return incorrect results.
Bug: IMPALA-2093
Severity: High
<property>
<name>hbase.client.retries.number</name>
<value>3</value>
</property>
<property>
<name>hbase.rpc.timeout</name>
<value>3000</value>
</property>
Currently, Cloudera Manager does not have an Impala-only override for HBase settings, so any HBase configuration
change you make through Cloudera Manager would take affect for all HBase applications. Therefore, this change is
not recommended on systems managed by Cloudera Manager.
Can't update stats manually via alter table after upgrading to CDH 5.2
Bug: IMPALA-1420
Severity: High
Workaround: On CDH 5.2, when adjusting table statistics manually by setting the numRows, you must also enable the
Boolean property STATS_GENERATED_VIA_STATS_TASK. For example, use a statement like the following to set both
properties with a single ALTER TABLE statement:
Resolution: The underlying cause is the issue HIVE-8648 that affects the metastore in Hive 0.13. The workaround is
only needed until the fix for this issue is incorporated into a CDH release.
If Hue and Impala are installed on the same host, and if you configure Hue Beeswax in CDH 4.1 to execute Impala
queries, Beeswax cannot list Hive tables and shows an error on Beeswax startup.
Hue requires Beeswaxd to be running in order to list the Hive tables. Because of a port conflict bug in Hue in CDH4.1,
when Hue and Impala are installed on the same host, an error page is displayed when you start the Beeswax application,
and when you open the Tables page in Beeswax.
Severity: High
Anticipated Resolution: Fixed in an upcoming CDH4 release
Workarounds: Choose one of the following workarounds (but only one):
• Install Hue and Impala on different hosts. OR
• Upgrade to CDH4.1.2 and add the following property in the beeswax section of the /etc/hue/hue.ini
configuration file:
beeswax_meta_server_only=9004
OR
• If you are using CDH4.1.1 and you want to install Hue and Impala on the same host, change the code in this file:
/usr/share/hue/apps/beeswax/src/beeswax/management/commands/beeswax_server.py
str(beeswax.conf.BEESWAX_SERVER_PORT.get()),
'8004',
Note:
If you used Cloudera Manager to install Impala, refer to the Cloudera Manager release notes for
information about using an equivalent workaround by specifying the
beeswax_meta_server_only=9004 configuration value in the options field for Hue. In Cloudera
Manager 4, these fields are labelled Safety Valve; in Cloudera Manager 5, they are called Advanced
Configuration Snippet.
Deviation from Hive behavior: Impala does not do implicit casts between string and numeric and boolean types.
Severity: Low
Anticipated Resolution: None
Workaround: Use explicit casts.
Deviation from Hive behavior: Out of range values float/double values are returned as maximum allowed value of type
(Hive returns NULL)
Impala behavior differs from Hive with respect to out of range float/double values. Out of range values are returned
as maximum allowed value of type (Hive returns NULL).
Severity: Low
Workaround: None
Impala incorrectly handles text data when the new line character \n\r is split between different HDFS block
If a carriage return / newline pair of characters in a text table is split between HDFS data blocks, Impala incorrectly
processes the row following the \n\r pair twice.
Bug: IMPALA-1578
Severity: High
Workaround: Use the Parquet format for large volumes of data where practical.
A failed CTAS does not drop the table if the insert fails.
If a CREATE TABLE AS SELECT operation successfully creates the target table but an error occurs while querying
the source table or copying the data, the new table is left behind rather than being dropped.
Bug: IMPALA-2005
Severity: High
Workaround: Drop the new table manually after a failed CREATE TABLE AS SELECT.
Bug: IMPALA-1821
Severity: High
Catalogd may crash when loading metadata for tables with many partitions, many columns and with incremental stats
Incremental stats use up about 400 bytes per partition for each column. For example, for a table with 20K partitions
and 100 columns, the memory overhead from incremental statistics is about 800 MB. When serialized for transmission
across the network, this metadata exceeds the 2 GB Java array size limit and leads to a catalogd crash.
Bugs: IMPALA-2647, IMPALA-2648, IMPALA-2649
Severity: Low. This does not occur frequently.
Workaround: If feasible, compute full stats periodically and avoid computing incremental stats for that table. The
scalability of incremental stats computation is a continuing work item.
Impala Parser issue when using fully qualified table names that start with a number.
A fully qualified table name starting with a number could cause a parsing error. In a name such as db.571_market,
the decimal point followed by digits is interpreted as a floating-point number.
Bug: IMPALA-941
Severity: Low
Workaround: Surround each part of the fully qualified name with backticks (``).
Note: Starting in April 2016, future release note updates are being consolidated in a single location
to avoid duplication of stale or incomplete information. You can view online the Impala New Features,
Incompatible Changes, Known Issues, and Fixed Issues. You can view or print all of these by downloading
the latest Impala PDF.
Free local allocations once for every row batch when building hash tables
A join query could encounter a serious error due to an internal failure to allocate memory, which resulted in
dereferencing a NULL pointer.
Bug: IMPALA-2612
Severity: High
Throw AnalysisError if table properties are too large (for the Hive metastore)
CREATE TABLE or ALTER TABLE statements could fail with metastore database errors due to length limits on the
SERDEPROPERTIES and TBLPROPERTIES clauses. (The limit on key size is 256, while the limit on value size is 4000.)
The fix makes Impala handle these error conditions more cleanly, by detecting too-long values rather than passing
them to the metastore database.
Bug: IMPALA-2226
Severity: High
Bug: IMPALA-2273
Severity: High
Bug: IMPALA-2473
Severity: High
Handle error when distinct and aggregates are used with a having clause
A query that included a DISTINCT operator and a HAVING clause, but no aggregate functions or GROUP BY, would
fail with an uninformative error message.
Bug: IMPALA-2113
Severity: High
Handle error when star based select item and aggregate are incorrectly used
A query that included * in the SELECT list, in addition to an aggregate function call, would fail with an uninformative
message if the query had no GROUP BY clause.
Bug: IMPALA-2225
Severity: High
Severity: High
Severity: High
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
For the full list of fixed issues, see the CDH 5.4.x release notes.
HBase scan node uses 2-4x memory after upgrade to Impala 2.2.8
Queries involving HBase tables used substantially more memory than in earlier Impala versions. The problem occurred
starting in Impala 2.2.8, as a result of the changes for IMPALA-2284. The fix for this issue involves removing a separate
memory work area for HBase queries and reusing other memory that was already allocated.
Bug: IMPALA-2731
Severity: High
Bug: IMPALA-2273
Severity: High
Fix spilling sorts with var-len slots that are NULL or empty.
A query that activated the spill-to-disk mechanism could fail if it contained a sort expression involving certain
combinations of fixed-length or variable-length types.
Bug: IMPALA-2357
Severity: High
Impala throws IllegalStateException when inserting data into a partition while select subquery group by partition
columns
An INSERT ... SELECT operation into a partitioned table could fail if the SELECT query included a GROUP BY clause
referring to the partition key columns.
Bug: IMPALA-2533
Severity: High
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
For the full list of fixed issues, see the CDH 5.4.x release notes.
Impala is unable to read hive tables created with the "STORED AS AVRO" clause
Impala could not read Avro tables created in Hive with the STORED AS AVRO clause.
Bug: IMPALA-1136, IMPALA-2161
Severity: High
make Parquet scanner fail query if the file size metadata is stale
If a Parquet file in HDFS was overwritten by a smaller file, Impala could encounter a serious error. Issuing a INVALIDATE
METADATA statement before a subsequent query would avoid the error. The fix allows Impala to handle such
inconsistencies in Parquet file length cleanly regardless of whether the table metadata is up-to-date.
Bug: IMPALA-2213
Severity: High
The catalog does not close the connection to HMS during table invalidation
A communication error could occur between Impala and the Hive metastore database, causing Impala operations that
update table metadata to fail.
Bug: IMPALA-2348
Severity: High
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
For the full list of fixed issues, see the CDH 5.4.x release notes.
Severity: High
Set the output smap of an EmptySetNode produced from an empty inline view.
A query could encounter a serious error if it included an inline view whose subquery had no FROM clause.
Bug: IMPALA-2216
Severity: High
Set an InsertStmt's result exprs from the source statement's result exprs.
A CREATE TABLE AS SELECT or INSERT ... SELECT statement could produce different results than a SELECT
statement, for queries including a FULL JOIN clause and including literal values in the select list.
Bug: IMPALA-2203
Severity: High
Row count not set for empty partition when spec is used with compute incremental stats
A COMPUTE INCREMENTAL STATS statement could leave the row count for an emptyp partition as -1, rather than
initializing the row count to 0. The missing statistic value could result in reduced query performance.
Bug: IMPALA-2199
Severity: High
Bug: IMPALA-1987
Severity: High
Substitute an InsertStmt's partition key exprs with the root node's smap.
An INSERT statement could encounter a serious error if the SELECT portion called an analytic function.
Bug: IMPALA-1737
Severity: High
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
For the full list of fixed issues, see the CDH 5.4.x release notes.
Impala DML/DDL operations corrupt table metadata leading to Hive query failures
When the Impala COMPUTE STATS statement was run on a partitioned Parquet table that was created in Hive, the
table subsequently became inaccessible in Hive. The table was still accessible to Impala. Regaining access in Hive
required a workaround of creating a new table. The error displayed in Hive was:
Error: Error while compiling statement: FAILED: SemanticException Class not found:
com.cloudera.impala.hive.serde.ParquetInputFormat (state=42000,code=40000)
Bug: IMPALA-2048
Severity: High
Bug: IMPALA-2018
Severity: High
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
For the full list of fixed issues, see the CDH 5.4.1 release notes.
Note: The Impala 2.2.x maintenance releases now use the CDH 5.4.x numbering system rather than
increasing the Impala version numbers. Impala 2.2 and higher are not available under CDH 4.
For the full list of fixed issues, see the CDH 5.4.1 release notes.
Note: Impala 2.2.0 is available as part of CDH 5.4.0 and is not available for CDH 4. Cloudera does not
intend to release future versions of Impala for CDH 4 outside patch and maintenance releases if
required. Given the end-of-maintenance status for CDH 4, Cloudera recommends all customers to
migrate to a recent CDH 5 release.
Altering a column's type causes column stats to stop sticking for that column
When the type of a column was changed in either Hive or Impala through ALTER TABLE CHANGE COLUMN, the
metastore database did not correctly propagate that change to the table that contains the column statistics. The
statistics (particularly the NDV) for that column were permanently reset and could not be changed by Impala's COMPUTE
STATS command. The underlying cause is a Hive bug (HIVE-9866).
Bug: IMPALA-1607
Severity: Major
Resolution: Resolved by incorporating the fix for HIVE-9866.
Workaround: On systems without the corresponding Hive fix, change the column back to its original type. The stats
reappear and you can recompute or drop them.
Bug: IMPALA-1712
Severity: High
DROP TABLE fails after COMPUTE STATS and ALTER TABLE RENAME to a different database.
When a table was moved from one database to another, the column statistics were not pointed to the new database.i
This could result in lower performance for queries due to unavailable statistics, and also an inability to drop the table.
Bug: IMPALA-1711
Severity: High
Impala incorrectly handles text data missing a newline on the last line
Some queries did not recognize the final line of a text data file if the line did not end with a newline character. This
could lead to inconsistent results, such as a different number of rows for SELECT COUNT(*) as opposed to SELECT
*.
Bug: IMPALA-1476
Severity: High
Impala's ACLs check do not consider all group ACLs, only checked first one.
If the HDFS user ID associated with the impalad process had read or write access in HDFS based on group membership,
Impala statements could still fail with HDFS permission errors if that group was not the first listed group for that user
ID.
Bug: IMPALA-1805
Severity: High
Cannot write Parquet files when values are larger than 64KB
Impala could sometimes fail to INSERT into a Parquet table if a column value such as a STRING was larger than 64 KB.
Bug: IMPALA-1705
Severity: High
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Constant filter expressions are not checked for errors and state cleanup on exception / DCHECK on destroying an
ExprContext
An invalid constant expression in a WHERE clause (for example, an invalid regular expression pattern) could fail to clean
up internal state after raising a query error. Therefore, certain combinations of invalid expressions in a query could
cause a crash, or cause a query to continue when it should halt with an error.
Bug: IMPALA-1756, IMPALA-2514
Severity: High
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Note: This Impala maintenance release is only available as part of CDH 5, not under CDH 4.
Crash: impala::TupleIsNullPredicate::Prepare
When expressions that tested for NULL were used in combination with analytic functions, an error could occur. (The
original crash issue was fixed by an earlier patch.)
Bug: IMPALA-1519
Severity: High
Note: Impala 2.1.3 is available as part of CDH 5.3.3, not under CDH 4.
Severity: High
Bug: IMPALA-1865
Severity: High
Impala's ACLs check do not consider all group ACLs, only checked first one.
If the HDFS user ID associated with the impalad process had read or write access in HDFS based on group membership,
Impala statements could still fail with HDFS permission errors if that group was not the first listed group for that user
ID.
Bug: IMPALA-1805
Severity: High
Bug: IMPALA-1712
Severity: High
Note: Impala 2.1.2 is available as part of CDH 5.3.2, not under CDH 4.
Impala incorrectly handles double numbers with more than 19 significant decimal digits
When a floating-point value was read from a text file and interpreted as a FLOAT or DOUBLE value, it could be incorrectly
interpreted if it included more than 19 significant digits.
Bug: IMPALA-1622
Severity: High
Bug: IMPALA-1535
Severity: High
Fetch column stats in bulk using new (Hive .13) HMS APIs
The performance of the COMPUTE STATS statement and queries was improved, particularly for wide tables.
Bug: IMPALA-1120
Severity: High
Note: Impala 2.0.5 is available as part of CDH 5.2.6, not under CDH 4.
Note: Impala 2.0.4 is available as part of CDH 5.2.5, not under CDH 4.
IoMgr infinite loop opening/closing file when shorter than cached metadata size
If a table data file was replaced by a shorter file outside of Impala, such as with INSERT OVERWRITE in Hive producing
an empty output file, subsequent Impala queries could hang.
Bug: IMPALA-1794
Severity: High
Note: Impala 2.0.3 is available as part of CDH 5.2.4, not under CDH 4.
Bug: IMPALA-1535
Severity: High
Fetch column stats in bulk using new (Hive .13) HMS APIs
The performance of the COMPUTE STATS statement and queries was improved, particularly for wide tables.
Bug: IMPALA-1120
Severity: High
Note: Impala 2.0.2 is available as part of CDH 5.2.3, not under CDH 4.
Queries fail with metastore exception after upgrade and compute stats
After running the COMPUTE STATS statement on an Impala table, subsequent queries on that table could fail with the
exception message Failed to load metadata for table: default.stats_test.
Bug: https://issues.cloudera.org/browse/IMPALA-1416 IMPALA-1416
Severity: High
Workaround: Upgrading to CDH 5.2.1, or another level of CDH that includes the fix for HIVE-8627, prevents the problem
from affecting future COMPUTE STATS statements. On affected levels of CDH, or for Impala tables that have become
inaccessible, the workaround is to disable the hive.metastore.try.direct.sql setting in the Hive metastore
hive-site.xml file and issue the INVALIDATE METADATA statement for the affected table. You do not need to
rerun the COMPUTE STATS statement for the table.
Incorrect plan after reordering predicates (inner join following outer join)
Potential wrong results for some types of queries.
Bug: IMPALA-1118"
Severity: High
Combining fragments with compatible data partitions can lead to incorrect results due to type incompatibilities (missing
casts).
Potential wrong results for some types of queries.
Bug: IMPALA-1123"
Severity: High
Bug: IMPALA-1109"
Severity: High
Bug: IMPALA-1121"
Severity: High
Allow creating Avro tables without column definitions. Allow COMPUTE STATS to always work on Impala-created Avro
tables.
Hive-created Avro tables with columns specified by a JSON file or literal could produce errors when queried in Impala,
and could not be used with the COMPUTE STATS statement. Now you can create such tables in Impala to avoid such
errors.
Bug: IMPALA-1104"
Severity: High
Impala does not employ ACLs when checking path permissions for LOAD and INSERT
Certain INSERT and LOAD DATA statements could fail unnecessarily, if the target directories in HDFS had restrictive
HDFS permissions, but those permissions were overridden by HDFS extended ACLs.
Bug: IMPALA-1279"
Severity: High
Note: Impala 1.4.4 is available as part of CDH 5.1.5, not under CDH 4.
Note: Impala 1.4.3 is available as part of CDH 5.1.4, and under CDH 4.
Note: Impala 1.4.3 is available as part of CDH 5.1.4, and under CDH 4.
Note: Impala 1.4.1 is only available as part of CDH 5.1.2, not under CDH 4.
boost::exception_detail::clone_impl
<boost::exception_detail::error_info_injector<boost::lock_error> >
Severity: High
Severity: High
Impalad catalog updates can fail with error: "IllegalArgumentException: fromKey out of range" at
com.cloudera.impala.catalog.CatalogDeltaLog
The error in the title could occur following a DDL statement. This issue was discovered during internal testing and has
not been reported in customer environments.
Bug: IMPALA-1093
Severity: High
"Total" time counter does not capture all the network transmit time
The time for some network operations was not counted in the report of total time for a query, making it difficult to
diagnose network-related performance issues.
Bug: IMPALA-1131
Severity: High
Impala will crash when reading certain Avro files containing bytes data
Certain Avro fields for byte data could cause Impala to be unable to read an Avro data file, even if the field was not
part of the Impala table definition. With this fix, Impala can now read these Avro data files, although Impala queries
cannot refer to the “bytes” fields.
Bug: IMPALA-1149
Severity: High
The issue was due to the use of HDFS caching with data files accessed by Impala. Support for HDFS caching in Impala
was introduced in Impala 1.4.0 for CDH 5.1.0. The fix for this issue was backported to Impala 1.3.x, and is the only
change in Impala 1.3.2 for CDH 5.0.4.
Bug: IMPALA-1019
Severity: High
Workaround: On CDH 5.0.x, upgrade to CDH 5.0.4 with Impala 1.3.2, where this issue is fixed. In Impala 1.3.0 or 1.3.1
on CDH 5.0.x, do not use HDFS caching for Impala data files in Impala internal or external tables. If some of these data
files are cached (for example because they are used by other components that take advantage of HDFS caching), set
the query option DISABLE_CACHED_READS=true. To set that option for all Impala queries across all sessions, start
impalad with the -default_query_options option and include this setting in the option argument, or on a cluster
managed by Cloudera Manager, fill in this option setting on the Impala Daemon options page.
Resolution: This issue is fixed in Impala 1.3.2 for CDH 5.0.4. The addition of HDFS caching support in Impala 1.4 means
that this issue does not apply to any new level of Impala on CDH 5.
Severity: High
The extended view definition SQL text in Views created by Impala should always have fully-qualified table names
When a view was accessed while inside a different database, references to tables were not resolved unless the names
were fully qualified when the view was created.
Bug: IMPALA-962
Severity: High
When I run CREATE TABLE new_table LIKE avro_table, the schema does not get mapped properly from an avro schema
to a hive schema
After a CREATE TABLE LIKE statement using an Avro table as the source, the new table could have incorrect metadata
and be inaccessible, depending on how the original Avro table was created.
Bug: IMPALA-185
Severity: High
Note: Impala 1.3.3 is only available as part of CDH 5.0.5, not under CDH 4.
Note: Impala 1.3.3 is only available as part of CDH 5.0.5, not under CDH 4.
The issue was due to the use of HDFS caching with data files accessed by Impala. Support for HDFS caching in Impala
was introduced in Impala 1.4.0 for CDH 5.1.0. The fix for this issue was backported to Impala 1.3.x, and is the only
change in Impala 1.3.2 for CDH 5.0.4.
Bug: IMPALA-1019
Severity: High
Workaround: On CDH 5.0.x, upgrade to CDH 5.0.4 with Impala 1.3.2, where this issue is fixed. In Impala 1.3.0 or 1.3.1
on CDH 5.0.x, do not use HDFS caching for Impala data files in Impala internal or external tables. If some of these data
files are cached (for example because they are used by other components that take advantage of HDFS caching), set
the query option DISABLE_CACHED_READS=true. To set that option for all Impala queries across all sessions, start
impalad with the -default_query_options option and include this setting in the option argument, or on a cluster
managed by Cloudera Manager, fill in this option setting on the Impala Daemon options page.
Resolution: This issue is fixed in Impala 1.3.2 for CDH 5.0.4. The addition of HDFS caching support in Impala 1.4 means
that this issue does not apply to any new level of Impala on CDH 5.
Impalad crashes when left joining inline view that has aggregate using distinct
Impala could encounter a severe error in a query combining a left outer join with an inline view containing a
COUNT(DISTINCT) operation.
Bug: IMPALA-904
Severity: High
Incorrect result with group by query with null value in group by data
If the result of a GROUP BY operation is NULL, the resulting row might be omitted from the result set. This issue depends
on the data values and data types in the table.
Bug: IMPALA-901
Severity: High
Text data with carriage returns generates wrong results for count(*)
A COUNT(*) operation could return the wrong result for text tables using nul characters (ASCII value 0) as delimiters.
Bug: IMPALA-13
Severity: High
Workaround: Impala adds support for ASCII 0 characters as delimiters through the clause FIELDS TERMINATED BY
'\0'.
IO Mgr should take instance memory limit into account when creating io buffers
Impala could allocate more memory than necessary during certain operations.
Bug: IMPALA-488
Severity: High
Workaround: Before issuing a COMPUTE STATS statement for a Parquet table, reduce the number of threads used in
that operation by issuing SET NUM_SCANNER_THREADS=2 in impala-shell. Then issue UNSET
NUM_SCANNER_THREADS before continuing with queries.
Impala should provide an option for new sub directories to automatically inherit the permissions of the parent directory
When new subdirectories are created underneath a partitioned table by an INSERT statement, previously the new
subdirectories always used the default HDFS permissions for the impala user, which might not be suitable for directories
intended to be read and written by other components also.
Bug: IMPALA-827
Severity: High
Resolution: In Impala 1.3.1 and higher, you can specify the --insert_inherit_permissions configuration when
starting the impalad daemon.
Illegal state exception (or crash) in query with UNION in inline view
Impala could encounter a severe error in a query where the FROM list contains an inline view that includes a UNION.
The exact type of the error varies.
Bug: IMPALA-888
Severity: High
Using distinct inside aggregate function may cause incorrect result when using having clause
A query could return incorrect results if it combined an aggregate function call, a DISTINCT operator, and a HAVING
clause, without a GROUP BY clause.
Bug: IMPALA-845
Severity: High
Wrong expression may be used in aggregate query if there are multiple similar expressions
If a GROUP BY query referenced the same columns multiple times using different operators, result rows could contain
multiple copies of the same expression.
Bug: IMPALA-817
Severity: High
Incorrect results when changing the order of aggregates in the select list with codegen enabled
Referencing the same columns in both a COUNT() and a SUM() call in the same query, or some other combinations
of aggregate function calls, could incorrectly return a result of 0 from one of the aggregate functions. This issue affected
references to TINYINT and SMALLINT columns, but not INT or BIGINT columns.
Bug: IMPALA-765
Severity: High
Workaround: Setting the query option DISABLE_CODEGEN=TRUE prevented the incorrect results. Switching the order
of the function calls could also prevent the issue from occurring.
Union queries give Wrong result in a UNION followed by SIGSEGV in another union
A UNION query could produce a wrong result, followed by a serious error for a subsequent UNION query.
Bug: IMPALA-723
Severity: High
Compute stats need to use quotes with identifiers that are Impala keywords
Using a column or table name that conflicted with Impala keywords could prevent running the COMPUTE STATS
statement for the table.
Bug: IMPALA-777
Severity: High
Fail early (in analysis) when COMPUTE STATS is run against Avro table with no columns
If the columns for an Avro table were all defined in the TBLPROPERTIES or SERDEPROPERTIES clauses, the COMPUTE
STATS statement would fail after completely analyzing the table, potentially causing a long delay. Although the COMPUTE
STATS statement still does not work for such tables, now the problem is detected and reported immediately.
Bug: IMPALA-867
Severity: High
Workaround: Re-create the Avro table with columns defined in SQL style, using the output of SHOW CREATE TABLE.
(See the JIRA page for detailed steps.)
The Catalog Server exits with an OOM error after a certain number of CREATE statements
A large number of concurrent CREATE TABLE statements can cause the catalogd process to consume excessive
memory, and potentially be killed due to an out-of-memory condition.
Bug: IMPALA-818
Severity: High
Workaround: Restart the catalogd service and re-try the DDL operations that failed.
Statestore seems to send concurrent heartbeats to the same subscriber leading to repeated "Subscriber 'hostname'
is registering with statestore, ignoring update" messages
Impala nodes could produce repeated error messages after recovering from a communication error with the statestore
service.
Bug: IMPALA-809
Severity: High
Impala cannot load tables with more than Short.MAX_VALUE number of partitions
If a table had more than 32,767 partitions, Impala would not recognize the partitions above the 32K limit and query
results could be incomplete.
Bug: IMPALA-749
Severity: High
Severity: Medium
Anticipated Resolution: Fixed in Impala 1.2.2.
Workaround: In Impala 1.2.2 and higher, use the COMPUTE STATS statement to gather statistics for each table involved
in the join query, after data is loaded. Prior to Impala 1.2.2, modify the query, if possible, to join the largest table first.
For example:
Parquet in CDH4.5 writes data files that are sometimes unreadable by Impala
Some Parquet files could be generated by other components that Impala could not read.
Bug: IMPALA-694
Severity: High
Resolution: The underlying issue is being addressed by a fix in the CDH Parquet libraries. Impala 1.2.2 works around
the problem and reads the existing data files.
Severity: High
Scanners use too much memory when reading past scan range
While querying a table with long column values, Impala could over-allocate memory leading to an out-of-memory
error. This problem was observed most frequently with tables using uncompressed RCFile or text data files.
Bug: IMPALA-525
Severity: High
Resolution: Fixed in 1.2.1
Excessive memory consumption when query tables with 1k columns (Parquet file)
Impala could encounter an out-of-memory condition setting up work areas for Parquet tables with many columns. The
fix reduces the size of the allocated memory when not actually needed to hold table data.
Bug: IMPALA-652
Severity: High
Resolution: Fixed in 1.2.1
"block size is too big" error with Snappy-compressed RCFile containing null
Queries could fail with a “block size is too big” error, due to NULL values in RCFile tables using Snappy compression.
Bug: IMPALA-482
Severity: High
Cannot query RC file for table that has more columns than the data file
Queries could fail if an Impala RCFile table was defined with more columns than in the corresponding RCFile data files.
Bug: IMPALA-510
Severity: High
Update the serde name we write into the metastore for Parquet tables
The SerDes class string written into Parquet data files created by Impala was updated for compatibility with Parquet
support in Hive. See Incompatible Changes Introduced in Impala 1.1.1 on page 634 for the steps to update older Parquet
data files for Hive compatibility.
Bug: IMPALA-485
Severity: High
Impala continues to allocate more memory even though it has exceed its mem-limit
Queries could allocate substantially more memory than specified in the impalad -mem_limit startup option. The fix
causes more frequent checking of the limit during query execution.
Bug: IMPALA-520
Severity: High
10-20% perf regression for most queries across all table formats
This issue is due to a performance tradeoff between systems running many queries concurrently, and systems running
a single query. Systems running only a single query could experience lower performance than in early beta releases.
Systems running many queries simultaneously should experience higher performance than in the beta releases.
Severity: High
planner fails with "Join requires at least one equality predicate between the two tables" when "from" table order does
not match "where" join order
A query could fail if it involved 3 or more tables and the last join table was specified as a subquery.
Bug: IMPALA-85
Severity: High
Comments in impala-shell in interactive mode are not handled properly causing syntax errors or wrong results
The impala-shell interpreter did not accept comment entered at the command line, making it problematic to copy
and paste from scripts or other code examples.
Bug: IMPALA-192
Severity: Low
Cancelled queries sometimes aren't removed from the inflight query list
The Impala web UI would sometimes display a query as if it were still running, after the query was cancelled.
Bug: IMPALA-364
Severity: High
Impala's 1.0.1 Shell Broke Python 2.4 Compatibility (AttributeError: 'module' object has no attribute 'field_size_limit)
The impala-shell command in Impala 1.0.1 does not work with Python 2.4, which is the default on Red Hat 5.
For the impala-shell command in Impala 1.0, the -o option (pipe output to a file) does not work with Python 2.4.
Bug: IMPALA-396
Severity: High
Impala parquet scanner cannot read all data files generated by other frameworks
Impala might issue an erroneous error message when processing a Parquet data file produced by a non-Impala Hadoop
component.
Bug: IMPALA-333
Severity: High
Resolution: Fixed
Impala is unable to query RCFile tables which describe fewer columns than the file's header.
If an RCFile table definition had fewer columns than the fields actually in the data files, queries would fail.
Bug: IMPALA-293
Severity: High
Resolution: Fixed
Bug: IMPALA-349
Severity: High
Resolution: Fixed
Impala returns 0 for bad time values in UNIX_TIMESTAMP, Hive returns NULL
Impala returns 0 for bad time values in UNIX_TIMESTAMP, Hive returns NULL.
Impala:
Hive:
Bug: IMPALA-16
Severity: Low
Anticipated Resolution: Fixed
Undeterministically receive "ERROR: unknown row bach destination..." and "ERROR: Invalid query handle" from impala
shell when running union query
A query containing both UNION and LIMIT clauses could intermittently cause the impalad process to halt with a
segmentation fault.
Bug: IMPALA-183
Severity: High
Resolution: Fixed
Add some library version validation logic to impalad when loading impala-lzo shared library
No validation was done to check that the impala-lzo shared library was compatible with the version of Impala,
possibly leading to a crash when using LZO-compressed text files.
Bug: IMPALA-234
Severity: High
Resolution: Fixed
Workaround: Always upgrade the impala-lzo library at the same time as you upgrade Impala itself.
Problems inserting into tables with TIMESTAMP partition columns leading table metadata loading failures and failed
dchecks
INSERT statements for tables partitioned on columns involving datetime types could appear to succeed, but cause
errors for subsequent queries on those tables. The problem was especially serious if an improperly formatted timestamp
value was specified for the partition key.
Bug: IMPALA-238
Severity: Critical
Resolution: Fixed
Ctrl-C sometimes interrupts shell in system call, rather than cancelling query
Pressing Ctrl-C in the impala-shell interpreter could sometimes display an error and return control to the shell,
making it impossible to cancel the query.
Bug: IMPALA-243
Severity: Critical
Resolution: Fixed
Excessive mem usage for certain queries which are very selective
Some queries that returned very few rows experienced unnecessary memory usage.
Bug: IMPALA-288
Severity: High
Resolution: Fixed
Right outer Join includes NULLs as well and hence wrong result count
The result set from a right outer join query could include erroneous rows containing NULL values.
Bug: IMPALA-90
Severity: High
Resolution: Fixed
DDL statements (CREATE/ALTER/DROP TABLE) are not supported in the Impala Beta Release
Severity: Medium
Resolution: Fixed in 0.7
Impala does not currently allow limiting the memory consumption of a single query
It is currently not possible to limit the memory consumption of a single query. All tables on the right hand side of JOIN
statements need to be able to fit in memory. If they do not, Impala may crash due to out of memory errors.
Severity: High
Resolution: Fixed in 0.7
Aggregate of a subquery result set returns wrong results if the subquery contains a 'limit' and data is distributed across
multiple nodes
Aggregate of a subquery result set returns wrong results if the subquery contains a 'limit' clause and data is distributed
across multiple nodes. From the query plan, it looks like we are just summing the results from each worker node.
Bug: IMPALA-20
Severity: Low
Resolution: Fixed in 0.7
Partition pruning for arbitrary predicates that are fully bound by a particular partition column
We currently cannot utilize a predicate like "country_code in ('DE', 'FR', 'US')" to do partitioning pruning, because that
requires an equality predicate or a binary comparison.
We should create a superclass of planner.ValueRange, ValueSet, that can be constructed with an arbitrary predicate,
and whose isInRange(analyzer, valueExpr) constructs a literal predicate by substitution of the valueExpr into the
predicate.
Bug: IMPALA-144
Severity: Medium
Resolution: Fixed in 0.7
Queries may fail on secure environment due to impalad Kerberos ticket expiration
Queries may fail on secure environment due to impalad Kerberos tickets expiring. This can happen if the Impala
-kerberos_reinit_interval flag is set to a value ten minutes or less. This may lead to an impalad requesting a
ticket with a lifetime that is less than the time to the next ticket renewal.
Bug: IMPALA-64
Severity: Medium
Resolution: Fixed in 0.6
Concurrent queries may fail when Impala uses Thrift to communicate with the Hive Metastore
Concurrent queries may fail when Impala is using Thrift to communicate with part of the Hive Metastore such as the
Hive Metastore Service. In such a case, the error get_fields failed: out of sequence response" may occur
because Impala shared a single Hive Metastore Client connection across threads. With Impala 0.6, a separate connection
is used for each metadata request.
Bug: IMPALA-48
Severity: Low
Resolution: Fixed in 0.6
Bug: IMPALA-44
Severity: Medium
Resolution: Fixed in 0.6
Backend client connections not getting cached causes an observable latency in secure clusters
Backend impalads do not cache connections to the coordinator. On a secure cluster, this introduces a latency proportional
to the number of backend clients involved in query execution, as the cost of establishing a secure connection is much
higher than in the non-secure case.
Bug: IMPALA-38
Severity: Medium
Resolution: Fixed in 0.5
Concurrent queries may fail with error: "Table object has not been been initialised : `PARTITIONS`"
Concurrent queries may fail with error: "Table object has not been been initialised : `PARTITIONS`".
This was due to a lack of locking in the Impala table/database metadata cache.
Bug: IMPALA-30
Severity: Medium
Resolution: Fixed in 0.5
UNIX_TIMESTAMP format behaviour deviates from Hive when format matches a prefix of the time value
The Impala UNIX_TIMESTAMP(val, format) operation compares the length of format and val and returns NULL if they
do not match. Hive instead effectively truncates val to the length of the format parameter.
Bug: IMPALA-15
Severity: Medium
Resolution: Fixed in 0.5
Order by on a string column produces incorrect results if there are empty strings
Severity: Low
Resolution: Fixed in 0.4
Severity: Mild
Resolution: Fixed in 0.3
Impala cannot read from HBase tables that are not created as external tables in the hive metastore.
Attempting to select from these tables fails.
Severity: Medium
Resolution: Fixed in 0.3
Certain queries that contain OUTER JOINs may return incorrect results
Queries that contain OUTER JOINs may not return the correct results if there are predicates referencing any of the
joined tables in the WHERE clause.
Severity: Medium
Resolution: Fixed in 0.3.
SELECT * FROM (SELECT sum(col1) FROM some_table GROUP BY col1) t1 JOIN other_table ON
(...);
Severity: Medium
Resolution: Fixed in 0.2
An insert with a limit that runs as more than one query fragment inserts more rows than the limit.
For example:
Severity: Medium
Resolution: Fixed in 0.2
Severity: Medium
Resolution: Fixed in 0.2
Impala server raises a null pointer exception when running an HBase query.
When querying an HBase table whose row-key is string type, the Impala server may raise a null pointer exception.
Severity: Medium
Resolution: Fixed in 0.2
Apache License
Version 2.0, January 2004
http://www.apache.org/licenses/
Cloudera | 689
Appendix: Apache License, Version 2.0
licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their
Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against
any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated
within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under
this License for that Work shall terminate as of the date such litigation is filed.
4. Redistribution.
You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without
modifications, and in Source or Object form, provided that You meet the following conditions:
1. You must give any other recipients of the Work or Derivative Works a copy of this License; and
2. You must cause any modified files to carry prominent notices stating that You changed the files; and
3. You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark,
and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part
of the Derivative Works; and
4. If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute
must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices
that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE
text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along
with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party
notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify
the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or
as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be
construed as modifying the License.
You may add Your own copyright statement to Your modifications and may provide additional or different license
terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as
a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated
in this License.
5. Submission of Contributions.
Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the
Licensor shall be under the terms and conditions of this License, without any additional terms or conditions.
Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement
you may have executed with Licensor regarding such Contributions.
6. Trademarks.
This License does not grant permission to use the trade names, trademarks, service marks, or product names of the
Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing
the content of the NOTICE file.
7. Disclaimer of Warranty.
Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides
its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied,
including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or
FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or
redistributing the Work and assume any risks associated with Your exercise of permissions under this License.
8. Limitation of Liability.
In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required
by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable
to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising
as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss
of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even
if such Contributor has been advised of the possibility of such damages.
9. Accepting Warranty or Additional Liability.
690 | Cloudera
Appendix: Apache License, Version 2.0
While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance
of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in
accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any
other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability
incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional
liability.
END OF TERMS AND CONDITIONS
http://www.apache.org/licenses/LICENSE-2.0
Cloudera | 691