HCatalog - Penulis Pembaca

HCatalog berisi API transfer data untuk input dan output paralel tanpa menggunakan MapReduce. API ini menggunakan abstraksi penyimpanan dasar tabel dan baris untuk membaca data dari cluster Hadoop dan menulis data ke dalamnya.

API Transfer Data terutama berisi tiga kelas; itu adalah -

  • HCatReader - Membaca data dari cluster Hadoop.

  • HCatWriter - Menulis data ke dalam cluster Hadoop.

  • DataTransferFactory - Menghasilkan contoh pembaca dan penulis.

API ini cocok untuk penyiapan node master-slave. Mari kita bahas lebih lanjutHCatReader dan HCatWriter.

HCatReader

HCatReader adalah kelas abstrak internal HCatalog dan memisahkan kerumitan sistem yang mendasari dari mana rekaman akan diambil.

Sr.No. Nama Metode & Deskripsi
1

Public abstract ReaderContext prepareRead() throws HCatException

Ini harus dipanggil pada node master untuk mendapatkan ReaderContext yang kemudian harus diserialkan dan dikirim node slave.

2

Public abstract Iterator <HCatRecorder> read() throws HCaException

Ini harus dipanggil di node budak untuk membaca HCatRecords.

3

Public Configuration getConf()

Ini akan mengembalikan objek kelas konfigurasi.

Kelas HCatReader digunakan untuk membaca data dari HDFS. Membaca adalah proses dua langkah di mana langkah pertama terjadi pada simpul master dari sistem eksternal. Langkah kedua dilakukan secara paralel pada beberapa node slave.

Pembacaan dilakukan pada a ReadEntity. Sebelum Anda mulai membaca, Anda perlu menentukan ReadEntity untuk membaca. Ini bisa dilakukan melaluiReadEntity.Builder. Anda dapat menentukan nama database, nama tabel, partisi, dan string filter. Misalnya -

ReadEntity.Builder builder = new ReadEntity.Builder();
ReadEntity entity = builder.withDatabase("mydb").withTable("mytbl").build(); 10.

Potongan kode di atas mendefinisikan objek ReadEntity ("entitas"), yang terdiri dari tabel bernama mytbl dalam database bernama mydb, yang dapat digunakan untuk membaca semua baris tabel ini. Perhatikan bahwa tabel ini harus ada dalam HCatalog sebelum memulai operasi ini.

Setelah menentukan ReadEntity, Anda mendapatkan instance HCatReader menggunakan ReadEntity dan konfigurasi cluster -

HCatReader reader = DataTransferFactory.getHCatReader(entity, config);

Langkah selanjutnya adalah mendapatkan ReaderContext dari reader sebagai berikut -

ReaderContext cntxt = reader.prepareRead();

HCatWriter

Abstraksi ini bersifat internal untuk HCatalog. Ini untuk memfasilitasi penulisan ke HCatalog dari sistem eksternal. Jangan mencoba membuat contoh ini secara langsung. Sebagai gantinya, gunakan DataTransferFactory.

Sr.No. Nama Metode & Deskripsi
1

Public abstract WriterContext prepareRead() throws HCatException

Sistem eksternal harus memanggil metode ini tepat sekali dari node master. Ini mengembalikan aWriterContext. Ini harus diserialkan dan dikirim ke node budak untuk dibangunHCatWriter sana.

2

Public abstract void write(Iterator<HCatRecord> recordItr) throws HCaException

Metode ini harus digunakan pada node budak untuk melakukan penulisan. RecordItr adalah objek iterator yang berisi kumpulan record untuk ditulis ke dalam HCatalog.

3

Public abstract void abort(WriterContext cntxt) throws HCatException

Metode ini harus dipanggil di node master. Tujuan utama dari metode ini adalah untuk melakukan pembersihan jika terjadi kegagalan.

4

public abstract void commit(WriterContext cntxt) throws HCatException

Metode ini harus dipanggil di node master. Tujuan dari metode ini adalah untuk melakukan metadata.

Mirip dengan membaca, menulis juga merupakan proses dua langkah di mana langkah pertama terjadi pada simpul master. Selanjutnya, langkah kedua terjadi secara paralel pada node slave.

Penulisan dilakukan pada a WriteEntity yang dapat dibangun dengan cara yang mirip dengan membaca -

WriteEntity.Builder builder = new WriteEntity.Builder();
WriteEntity entity = builder.withDatabase("mydb").withTable("mytbl").build();

Kode di atas membuat objek WriteEntity entityyang dapat digunakan untuk menulis ke dalam tabel bernamamytbl di database mydb.

Setelah membuat WriteEntity, langkah selanjutnya adalah mendapatkan WriterContext -

HCatWriter writer = DataTransferFactory.getHCatWriter(entity, config);
WriterContext info = writer.prepareWrite();

Semua langkah di atas terjadi pada node master. Node master kemudian membuat serial objek WriterContext dan membuatnya tersedia untuk semua budak.

Pada node slave, Anda perlu mendapatkan HCatWriter menggunakan WriterContext sebagai berikut -

HCatWriter writer = DataTransferFactory.getHCatWriter(context);

Kemudian, writermengambil iterator sebagai argumen untuk writemetode tersebut -

writer.write(hCatRecordItr);

Itu writer lalu menelepon getNext() pada iterator ini dalam satu loop dan menulis semua record yang dilampirkan ke iterator.

Itu TestReaderWriter.javafile digunakan untuk menguji kelas HCatreader dan HCatWriter. Program berikut menunjukkan cara menggunakan HCatReader dan HCatWriter API untuk membaca data dari file sumber dan kemudian menulisnya ke file tujuan.

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hive.metastore.api.MetaException;
import org.apache.hadoop.hive.ql.CommandNeedRetryException;
import org.apache.hadoop.mapreduce.InputSplit;

import org.apache.hive.HCatalog.common.HCatException;
import org.apache.hive.HCatalog.data.transfer.DataTransferFactory;
import org.apache.hive.HCatalog.data.transfer.HCatReader;
import org.apache.hive.HCatalog.data.transfer.HCatWriter;
import org.apache.hive.HCatalog.data.transfer.ReadEntity;
import org.apache.hive.HCatalog.data.transfer.ReaderContext;
import org.apache.hive.HCatalog.data.transfer.WriteEntity;
import org.apache.hive.HCatalog.data.transfer.WriterContext;
import org.apache.hive.HCatalog.mapreduce.HCatBaseTest;

import org.junit.Assert;
import org.junit.Test;

public class TestReaderWriter extends HCatBaseTest {
   @Test
   public void test() throws MetaException, CommandNeedRetryException,
      IOException, ClassNotFoundException {
		
      driver.run("drop table mytbl");
      driver.run("create table mytbl (a string, b int)");
		
      Iterator<Entry<String, String>> itr = hiveConf.iterator();
      Map<String, String> map = new HashMap<String, String>();
		
      while (itr.hasNext()) {
         Entry<String, String> kv = itr.next();
         map.put(kv.getKey(), kv.getValue());
      }
		
      WriterContext cntxt = runsInMaster(map);
      File writeCntxtFile = File.createTempFile("hcat-write", "temp");
      writeCntxtFile.deleteOnExit();
		
      // Serialize context.
      ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(writeCntxtFile));
      oos.writeObject(cntxt);
      oos.flush();
      oos.close();
		
      // Now, deserialize it.
      ObjectInputStream ois = new ObjectInputStream(new FileInputStream(writeCntxtFile));
      cntxt = (WriterContext) ois.readObject();
      ois.close();
      runsInSlave(cntxt);
      commit(map, true, cntxt);
		
      ReaderContext readCntxt = runsInMaster(map, false);
      File readCntxtFile = File.createTempFile("hcat-read", "temp");
      readCntxtFile.deleteOnExit();
      oos = new ObjectOutputStream(new FileOutputStream(readCntxtFile));
      oos.writeObject(readCntxt);
      oos.flush();
      oos.close();
		
      ois = new ObjectInputStream(new FileInputStream(readCntxtFile));
      readCntxt = (ReaderContext) ois.readObject();
      ois.close();
		
      for (int i = 0; i < readCntxt.numSplits(); i++) {
         runsInSlave(readCntxt, i);
      }
   }
	
   private WriterContext runsInMaster(Map<String, String> config) throws HCatException {
      WriteEntity.Builder builder = new WriteEntity.Builder();
      WriteEntity entity = builder.withTable("mytbl").build();
		
      HCatWriter writer = DataTransferFactory.getHCatWriter(entity, config);
      WriterContext info = writer.prepareWrite();
      return info;
   }
	
   private ReaderContext runsInMaster(Map<String, String> config, 
      boolean bogus) throws HCatException {
      ReadEntity entity = new ReadEntity.Builder().withTable("mytbl").build();
      HCatReader reader = DataTransferFactory.getHCatReader(entity, config);
      ReaderContext cntxt = reader.prepareRead();
      return cntxt;
   }
	
   private void runsInSlave(ReaderContext cntxt, int slaveNum) throws HCatException {
      HCatReader reader = DataTransferFactory.getHCatReader(cntxt, slaveNum);
      Iterator<HCatRecord> itr = reader.read();
      int i = 1;
		
      while (itr.hasNext()) {
         HCatRecord read = itr.next();
         HCatRecord written = getRecord(i++);
			
         // Argh, HCatRecord doesnt implement equals()
         Assert.assertTrue("Read: " + read.get(0) + "Written: " + written.get(0),
         written.get(0).equals(read.get(0)));
			
         Assert.assertTrue("Read: " + read.get(1) + "Written: " + written.get(1),
         written.get(1).equals(read.get(1)));
			
         Assert.assertEquals(2, read.size());
      }
		
      //Assert.assertFalse(itr.hasNext());
   }
	
   private void runsInSlave(WriterContext context) throws HCatException {
      HCatWriter writer = DataTransferFactory.getHCatWriter(context);
      writer.write(new HCatRecordItr());
   }
	
   private void commit(Map<String, String> config, boolean status,
      WriterContext context) throws IOException {
      WriteEntity.Builder builder = new WriteEntity.Builder();
      WriteEntity entity = builder.withTable("mytbl").build();
      HCatWriter writer = DataTransferFactory.getHCatWriter(entity, config);
		
      if (status) {
         writer.commit(context);
      } else {
         writer.abort(context);
      }
   }
	
   private static HCatRecord getRecord(int i) {
      List<Object> list = new ArrayList<Object>(2);
      list.add("Row #: " + i);
      list.add(i);
      return new DefaultHCatRecord(list);
   }
	
   private static class HCatRecordItr implements Iterator<HCatRecord> {
      int i = 0;
		
      @Override
      public boolean hasNext() {
         return i++ < 100 ? true : false;
      }
		
      @Override
      public HCatRecord next() {
         return getRecord(i);
      }
		
      @Override
      public void remove() {
         throw new RuntimeException();
      }
   }
}

Program di atas membaca data dari HDFS dalam bentuk record dan menulis data record ke mytable