提到Hive通常都會想到,Hive是數據倉庫,支持類SQL查詢,有不少語法支持,能夠嵌套MR,寫Transform、寫UDF/UDAF等,可是,不支持更新操做。因此Hive的常見也通常都是一次寫入,頻繁讀取。從Hive 0.13開始,加入了ACID的新feature,可是0.13的時候還不支持insert、update和delete操做,我也並無欣然的當小白鼠。html
目前咱們平臺使用hive1.2.1的社區版,業務上也遇到了須要更新的場景。也是繼續調研Transaction的特性。Transaction有幾個依賴條件:java
一、只支持ORCFile數據庫
二、默認關閉,須要本身開啓apache
三、表必須分桶session
四、0.14開始支持insert、update和deleteapp
五、必須加入一大堆配置(下文)ide
一言不合就上個Demo:oop
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++post
一、建表:fetch
create table test_trancaction (user_id Int,name String) clustered by (user_id) into 3 buckets stored as orc TBLPROPERTIES ('transactional'='true');
二、導入數據:
create table test_insert_test(id int,name string) row format delimited fields TERMINATED BY ','; ---臨時表 hive> select * from test_insert_test; OK 1 jerrick 2 tom 3 jerry 4 lily 5 hanmei 6 limlei 7 lucky insert into test_trancaction select * from test_insert_test; hive> > select * from test_trancaction; OK 1 jerrick 2 tom 3 jerry 4 lily 5 hanmei 6 limlei 7 lucky
三、查看分桶狀況:
hive > dfs -ls /user/hive/warehouse/test_trancaction/; Found 1 items -rwxrwxr-x 3 hadoop hadoop 352 2016-07-13 16:24 /user/hive/warehouse/test_trancaction/000000_0
並無分桶,設置參數:set hive.enforce.bucketing=true
再次導入,查看日誌:Number of reduce tasks determined at compile time: 3,再次check hdf文件,發現已經分紅了3塊。
四、嘗試update:
修改配置hive-site
<!--start for trancaction --> <property> <name>hive.support.concurrency</name> <value>true</value> </property> <property> <name>hive.enforce.bucketing</name> <value>true</value> </property> <property> <name>hive.exec.dynamic.partition.mode</name> <value>nonstrict</value> </property> <property> <name>hive.txn.manager</name> <value>org.apache.hadoop.hive.ql.lockmgr.DbTxnManager</value> </property> <property> <name>hive.compactor.initiator.on</name> <value>true</value> </property> <property> <name>hive.compactor.worker.threads</name> <value>1</value> </property>
見證奇蹟的時刻: update test_trancaction set name='jerrick_up' where id=1;
哪有那麼一路順風,看看遇到的問題:
一、Load Data:
load data local inpath '/home/wangjiankui/hives/transaction.csv' into table test_trancaction;
查詢失敗:
select * from test_trancaction;
Failed with exception java.io.IOException:java.io.IOException: Malformed ORC file user/hive/warehouse/test_trancaction/transaction.csv. Invalid postscript.
緣由:load data是純拷貝,不會作其餘的處理,因此會報格式問題。
二、鏈接metastore失敗
select * from test_trancaction;
FAILED: LockException [Error 10280]: Error communicating with the metastore
看日誌:
Caused by: org.apache.thrift.TApplicationException: Internal error processing get_open_txns
重啓metastore
Error rolling back: Can't call rollback when autocommit=true
2016-07-13 18:11:13,056 ERROR metastore.HiveMetaStore (HiveMetaStore.java:run(6104)) - Failure when starting the compactor, compactions may not happen, java.lang.RuntimeException: Unable to set up transaction database for testing: Can't call rollback when autocommit=true
不要加這個配置!!!!
<property>
<name>hive.in.test</name>
<value>true</value>
</property>
ACID stands for four traits of database transactions: Atomicity (an operation either succeeds completely or fails, it does not leave partial data), Consistency (once an application performs an operation the results of that operation are visible to it in every subsequent operation), Isolation (operations by one user do not cause unexpected side effects for other users), and Durability (once an operation is complete it will be preserved even in the face of machine or system failure). These traits have long been expected of database systems as part of their transaction functionality.
Up until Hive 0.13, atomicity, consistency, and durability were provided at the partition level. Isolation could be provided by turning on one of the available locking mechanisms (ZooKeeper or in memory). With the addition of transactions in Hive 0.13 it is now possible to provide full ACID semantics at the row level, so that one application can add rows while another reads from the same partition without interfering with each other.
ACID是數據庫事務性的4個基本要素:原子性、一致性、隔離性和持久性。
Transactions with ACID semantics have been added to Hive to address the following use cases:
只支持ORCFile。
默認關閉,須要配置開啓。
表必須分桶。In general, distributing rows based on the hash will give you a even distribution(均勻分佈) in the buckets.
分桶介紹:http://www.aahyhaa.com/archives/316
只支持快照級別的隔離。
Hive offers APIs for streaming data ingest and streaming mutation:
A comparison of these two APIs is available in the Background section of the Streaming Mutation document.
INSERT...VALUES, UPDATE, and DELETE have been added to the SQL grammar, starting in Hive 0.14. See LanguageManual DML for details.
從0.14開始,支持INSET、UPDATE和DELETE.
Several new commands have been added to Hive's DDL in support of ACID and transactions, plus some existing DDL has been modified.
A new command SHOW TRANSACTIONS has been added, see Show Transactions for details.
A new command SHOW COMPACTIONS has been added, see Show Compactions for details.
The SHOW LOCKS command has been altered to provide information about the new locks associated with transactions. If you are using the ZooKeeper or in-memory lock managers you will notice no difference in the output of this command. See Show Locks for details.
A new option has been added to ALTER TABLE to request a compaction of a table or partition. In general users do not need to request compactions, as the system will detect the need for them and initiate the compaction. However, if compaction is turned off for a table or a user wants to compact the table at a time the system would not choose to, ALTER TABLE can be used to initiate the compaction. See Alter Table/Partition Compact for details. This will enqueue a request for compaction and return. To watch the progress of the compaction the user can use SHOW COMPACTIONS.
HDFS does not support in-place changes to files. It also does not offer read consistency in the face of writers appending to files being read by a user. In order to provide these features on top of HDFS we have followed the standard approach used in other data warehousing tools. Data for the table or partition is stored in a set of base files. New records, updates, and deletes are stored in delta files. A new set of delta files is created for each transaction (or in the case of streaming agents such as Flume or Storm, each batch of transactions) that alters a table or partition. At read time the reader merges the base and delta files, applying any updates and deletes as it reads.
Occasionally these changes need to be merged into the base files. To do this a set of threads has been added to the Hive metastore. They determine when this compaction needs to be done, execute the compaction, and then clean up afterwards. There are two types of compactions, minor and major.
All compactions are done in the background and do not prevent concurrent reads and writes of the data. After a compaction the system waits until all readers of the old files have finished and then removes the old files.
Compactions are MR jobs with name in the following form: <hostname>-compactor-<db>.<table>.<partition>
Previously all files for a partition (or a table if the table is not partitioned) lived in a single directory. With these changes, any partitions (or tables) written with an ACID aware writer will have a directory for the base files and a directory for each set of delta files.
A new lock manager has also been added to Hive, the DbLockManager. This lock manager stores all lock information in the metastore. In addition all transactions are stored in the metastore. This means that transactions and locks are durable in the face of server failure. To avoid clients dying and leaving transaction or locks dangling, a heartbeat is sent from lock holders and transaction initiators to the metastore on a regular basis. If a heartbeat is not received in the configured amount of time, the lock or transaction will be aborted.
As of Hive 1.3.0, the length of time that the DbLockManger will continue to try to acquire locks can be controlled via hive.lock.numretires and hive.lock.sleep.between.retries. When the DbLockManager cannot acquire a lock (due to existence of a competing lock), it will back off and try again after a certain time period. In order to support short running queries and not overwhelm the metastore at the same time, the DbLockManager will double the wait time after each retry. The initial back off time is 100ms and is capped by hive.lock.sleep.between.retries. hive.lock.numretries is the total number of times it will retry a given lock request. Thus the total time that the call to acquire locks will block (given default values of 10 retries and 60s sleep time) is (100ms + 200ms + 400ms + ... + 51200ms + 60s + 60s + ... + 60s) = 91m:42s:300ms.
More details on locks used by this Lock Manager.
These configuration parameters must be set appropriately to turn on transaction support in Hive:
The following sections list all of the configuration parameters that affect Hive transactions and compaction. Also see Limitations above and Table Properties below.
A number of new configuration parameters have been added to the system to support transactions.
Configuration key |
Values |
Location | Notes |
Default:org.apache.hadoop.hive.ql.lockmgr.DummyTxnManager Value required for transactions:org.apache.hadoop.hive.ql.lockmgr.DbTxnManager |
Client/ |
DummyTxnManager replicates pre Hive-0.13 behavior and provides no transactions. |
|
Default: 300 |
Client/ Metastore |
Time after which transactions are declared aborted if the client has not sent a heartbeat, in seconds. It's critical that this property has the same value for all components/services.5 |
|
hive.timedout.txn.reaper.start | Default: 100s | Metastore | Time delay of first reaper (the process which aborts timed-out transactions) run after the metastore starts (as of Hive 1.3.0). |
Default: 180s | Metastore | Time interval describing how often the reaper (the process which aborts timed-out transactions) runs (as of Hive 1.3.0). |
|
Default: 1000 |
Client | Maximum number of transactions that can be fetched in one call to open_txns().1 |
|
Default: false Value required for transactions: true (for exactly one instance of the Thrift metastore service) |
Metastore | Whether to run the initiator and cleaner threads on this metastore instance. It's critical that this is enabled on exactly one metastore service instance (not enforced yet). As of Hive 1.3.0 this property may be enabled on any number of metastore instances.
|
|
Default: 0 Value required for transactions: > 0 on at least one instance of the Thrift metastore service |
Metastore | How many compactor worker threads to run on this metastore instance.2 |
|
Default: 86400 |
Metastore | Time in seconds after which a compaction job will be declared failed and the compaction re-queued. |
|
hive.compactor.cleaner.run.interval | Default: 5000 | Metastore | Time in milliseconds between runs of the cleaner thread. (Hive 0.14.0 and later.) |
Default: 300 |
Metastore | Time in seconds between checks to see if any tables or partitions need to be compacted.3 |
|
Default: 10 |
Metastore | Number of delta directories in a table or partition that will trigger a minor compaction. |
|
Default: 0.1 |
Metastore | Percentage (fractional) size of the delta files relative to the base that will trigger a major compaction. 1 = 100%, so the default 0.1 = 10%. |
|
Default: 1000 |
Metastore | Number of aborted transactions involving a given table or partition that will trigger a major compaction. |
|
Default: 500 | Metastore | Maximum number of delta files that the compactor will attempt to handle in a single job (as of Hive 1.3.0).4 | |
Default: "" (empty string) | Metastore | Used to specify name of Hadoop queue to which Compaction jobs will be submitted. Set to empty string to let Hadoop choose the queue (as of Hive 1.3.0). |
1hive.txn.max.open.batch controls how many transactions streaming agents such as Flume or Storm open simultaneously. The streaming agent then writes that number of entries into a single file (per Flume agent or Storm bolt). Thus increasing this value decreases the number of delta files created by streaming agents. But it also increases the number of open transactions that Hive has to track at any given time, which may negatively affect read performance.
2Worker threads spawn MapReduce jobs to do compactions. They do not do the compactions themselves. Increasing the number of worker threads will decrease the time it takes tables or partitions to be compacted once they are determined to need compaction. It will also increase the background load on the Hadoop cluster as more MapReduce jobs will be running in the background.
3Decreasing this value will reduce the time it takes for compaction to be started for a table or partition that requires compaction. However, checking if compaction is needed requires several calls to the NameNode for each table or partition that has had a transaction done on it since the last major compaction. So decreasing this value will increase the load on the NameNode.
4If the compactor detects a very high number of delta files, it will first run several partial minor compactions (currently sequentially) and then perform the compaction actually requested.
5If the value is not the same active transactions may be determined to be "timed out" and consequently Aborted. This will result in errors like "No such transaction...", "No such lock ..."
In addition to the new parameters listed above, some existing parameters need to be set to support INSERT ... VALUES, UPDATE, and DELETE.
Configuration key |
Must be set to |
---|---|
hive.support.concurrency | true (default is false) |
hive.enforce.bucketing | true (default is false) (Not required as of Hive 2.0) |
hive.exec.dynamic.partition.mode | nonstrict (default is strict) |
If the data in your system is not owned by the Hive user (i.e., the user that the Hive metastore runs as), then Hive will need permission to run as the user who owns the data in order to perform compactions. If you have already set up HiveServer2 to impersonate users, then the only additional work to do is assure that Hive has the right to impersonate users from the host running the Hive metastore. This is done by adding the hostname to hadoop.proxyuser.hive.hosts
in Hadoop's core-site.xml
file. If you have not already done this, then you will need to configure Hive to act as a proxy user. This requires you to set up keytabs for the user running the Hive metastore and add hadoop.proxyuser.hive.hosts
and hadoop.proxyuser.hive.groups
to Hadoop's core-site.xml
file. See the Hadoop documentation on secure mode for your version of Hadoop (e.g., for Hadoop 2.5.1 it is at Hadoop in Secure Mode).
If a table is to be used in ACID writes (insert, update, delete) then the table property "transactional=true
" must be set on that table, starting with Hive 0.14.0. Also, hive.txn.manager must be set to org.apache.hadoop.hive.ql.lockmgr.DbTxnManager either in hive-site.xml or in the beginning of the session before any query is run. Without those, inserts will be done in the old style; updates and deletes will be prohibited. However, this does not apply to Hive 0.13.0.
If a table owner does not wish the system to automatically determine when to compact, then the table property "NO_AUTO_COMPACTION
" can be set. This will prevent all automatic compactions. Manual compactions can still be done with Alter Table/Partition Compact statements.
Table properties are set with the TBLPROPERTIES clause when a table is created or altered, as described in the Create Table and Alter Table Properties sections of Hive Data Definition Language. The "transactional
" and "NO_AUTO_COMPACTION
" table properties are case-sensitive in Hive releases 0.x and 1.0, but they are case-insensitive starting with release 1.1.0 (HIVE-8308).