前文:

textfile格式在遇到文本数据就会有分隔符及换行符问题,所以采用parquet作为存储格式,但也会引进数据类型转换的问题。

正则匹配将数据发送到不同的hdfs文件夹。

模糊查询将会产生慢查询,一般我们用可以存放在es中。

一、Mysql导数据导Hive

1.1 建表

create external table if not exists ods.ods_stu( 
`id` int comment '主键Id', 
`name` string comment '名称',
`addtime` string commint '添加时间'
)comment '姓名表'
partitioned by (dt string)
stored as parquet
location '/user/hive/warehouse/ods.db/ods_stu';

备注:parquet存储格式查询效率高

1.2 使用sqoop导数据

sqoop import \
--connect jdbc:mysql://localhost:3306/ods_stu\
--username root\
--password 123456\
--null-string 'NULL'  \
--null-non-string 'NULL' \
--query '
select * 
from stu
where $CONDITIONS' \
--target-dir /user/hive/warehouse/ods.db/ods_stu/dt=2020-06-04 \
--map-column-java addtime=String \
--map-column-hive addtime=String 
--append \
--split-by id \
--num-mappers 2 \
--as-parquetfile \

存在问题:sqoop直接从mysql导入parquet格式的hive表有一个数据类型异常

Failed with exception java.io.IOException:org.apache.hadoop.hive.ql.metadata.HiveException: java.lang.ClassCastException: org.apache.hadoop.io.LongWritable cannot be cast to org.apache.hadoop.hive.serde2.io.TimestampWritabl

解决方式:采用 --map-column-java 和 --map-column-hive 把mysql中时间类型之间转换为string类型存储hive中

load data inpath '/user/hive/warehouse/ods.db/ods_stu/dt=2020-06-04' into table ods.ods_minisns_systemtag partition (dt = '2020-06-04');

1.3 使用spark导数据

/**
     * 通过拉取mysql数据,创建临时表写入hive中
     */
    val df: DataFrame = spark.read.format("jdbc")
      .options(mysqlParam)
      //增量:单线程
      .option("query", query_mysql)
      .load()
    df.createOrReplaceTempView(tmp_tablename)
    spark.sql(insert_hive);

备注:spark可以直接写入parquet格式的hive表,并且可以直接数据清洗。但spark连接查询mysql会有一个散列id排序的问题,并且使用query自定义sql查询的话只有1个线程拉数据。而sqoop可以完美的解决这两个问题,使用方式简单。

1.4sqoop与spark导数据的比较

 

sqoop

spark

textfile格式外部表

不支持复杂分隔符,需要编译

无需关心分隔符

paquet格式

需处理数据类型转换

无需关心数据类型问题

散列id同步

存在倾斜问题

存在倾斜问题

并行度

完美处理

query自定义查询只有单线程查询mysql

直接清洗

仅在mysql查询时清洗,性能较差

拉取数据后直接清洗

使用难度

简单配置

写个工具类传参也可,但较复杂

二、Flume采集Kafka数据

 

2.1kafka数据到hive外部表

a1.sources = r1
a1.channels = c1
a1.sinks = k1

#————————————————————  sources  ————————————————————#

a1.sources.r1.type = org.apache.flume.source.kafka.KafkaSource
a1.sources.r1.batchSize = 5000
a1.sources.r1.batchDurationMillis = 2000
a1.sources.r1.kafka.bootstrap.servers = bigdata-test:9092
a1.sources.r1.kafka.topics = test_0603
a1.sources.r1.kafka.consumer.group.id = custom.g.id
a1.sources.r1.kafka.consumer.auto.offset.reset=earliest



#————————————————————  interceptors  ————————————————————#

a1.sources.r1.interceptors = i1
a1.sources.r1.interceptors.i1.type = regex_extractor

# 正则获取捕获组并命名,用于sink文件夹路径设置

a1.sources.r1.interceptors.i1.regex = (\\d{4}\\-\\d{2}\\-\\d{2})
a1.sources.r1.interceptors.i1.serializers = s1
a1.sources.r1.interceptors.i1.serializers.s1.name = eventTime



#————————————————————  channels  ————————————————————#

a1.channels.c1.type = memory
a1.channels.c1.capacity = 10000
a1.channels.c1.transactionCapacity = 10000
a1.channels.c1.byteCapacityBufferPercentage = 20
a1.channels.c1.byteCapacity = 800000



#————————————————————  sinks  ————————————————————#

a1.sinks.k1.type = hdfs
a1.sinks.k1.hdfs.path = /tmp/flume_test_0603/%{eventTime}

# 使用当地时间(而不是从事件的时间戳头)而取代转义序列。
a1.sinks.k1.hdfs.useLocalTimeStamp = true

# DataStream不会压缩输出文件
a1.sinks.k1.hdfs.fileType = DataStream	

# 默认为1024,触发滚动的文件大小,以字节为单位(0:从不基于文件大小滚动)
a1.sinks.k1.hdfs.rollSize = 104857600  

# 默认为30,滚动当前文件之前要等待的秒数(0 =根据时间间隔从不滚动)
#a1.sinks.k1.hdfs.rollInterval = 7200   

# 默认为10,滚动之前写入文件的事件数(0 =基于事件数从不滚动)
a1.sinks.k1.hdfs.rollCount = 0     

a1.sinks.k1.hdfs.filePrefix = %Y%m%d_
a1.sinks.k1.hdfs.fileSuffix = .log

#————————————————————  绑定  ————————————————————#

a1.sources.r1.channels = c1
a1.sinks.k1.channel = c1

备注:通过正则匹配kafka日志中的数据,将数据发送到不同的文件夹下,但是正在写入时文件格式为tmp类型,hive不能识别出文件,所以可以按2小时形成一个正式文件。当没有数据写入时,就不会生成tmp文件。

2.2flume切割文件相关参数

#————————————————————  滚动切换:控制写文件的切换规则  ————————————————————#
# 按文件体积(字节)来切
ag1.sinks.sink1.hdfs.rollSize = 512000
# 按event条数切
ag1.sinks.sink1.hdfs.rollCount = 1000000 
# 按时间间隔切换文件,创建文件多久后
ag1.sinks.sink1.hdfs.rollInterval = 60 



#————————————————————  时间切换:控制生成目录的规则  ————————————————————#

ag1.sinks.sink1.hdfs.path =hdfs://hadoop04:9000/flume/%y/%m/%d/%H

ag1.sinks.sink1.hdfs.round = true
ag1.sinks.sink1.hdfs.roundValue = 10
ag1.sinks.sink1.hdfs.roundUnit = minute
ag1.sinks.sink1.hdfs.useLocalTimeStamp = true

三、Es数据采集

3.1采集mysql数据到Es

stu.conf文件

input {
  jdbc{
    # mysql 数据库链接
    jdbc_connection_string => "jdbc:mysql://localhost:3306/minisns?useUnicode=true&characterEncoding=utf-8&useSSL=false"
    # 用户名和密码
    jdbc_user => "123456"
    jdbc_password => "123456"
    #驱动
    jdbc_driver_library => "/opt/mysql/mysql-connector-java.jar"
    # 驱动类名
    jdbc_driver_class => "com.mysql.jdbc.Driver"
    jdbc_paging_enabled => "true"
    #jdbc_page_size => "5000"
    jdbc_page_size => "50000"
    jdbc_default_timezone =>"Asia/Shanghai"
    # mysql文件, 也可以直接写SQL语句在此处,如下:
    statement_filepath => "/opt/logstash/config/jdbc_sql/stu.sql"
    # 这里类似crontab,可以定制定时操作,比如每分钟执行一次同步(分 时 天 月 年)
    schedule => "*/5 * * * *"
    #type => "jdbc"
    # 是否记录上次执行结果, 如果为真,将会把上次执行到的 tracking_column 字段的值记录下来,保存到 last_run_metadata_path 指定的文件中
    record_last_run => true

    # 是否需要记录某个column 的值,如果record_last_run为真,可以自定义我们需要 track 的 column 名称,此时该参数就要为 true. 否则默认 track 的是 timestamp 的值.
    use_column_value => true
    # 如果 use_column_value 为真,需配置此参数. track 的数据库 column 名,该 column 必须是递增的. 一般是mysql主键
    tracking_column => "addtime"
    tracking_column_type => "datatime"
    last_run_metadata_path => "/opt/logstash/config/track_value/stu"
    # 是否清除 last_run_metadata_path 的记录,如果为真那么每次都相当于从头开始查询所有的数据库记录
    clean_run => false
    # 是否将 字段(column) 名称转小写
    lowercase_column_names => false
  }
}
          
output { 
  elasticsearch {
    hosts => ["http://es-01:9200","http://es-02:9200","http://es-03:9200"]
    index => "stu_v1"
    document_id => "%{Id}"
    user => "elastic"
    password => "123456"
  } 
}

stu.sql文件

select
Id
,name
,date_format(addtime, '%Y-%m-%d %H:%i:%s') as addtime
from student
where addtime >= date_add(:sql_last_value, interval -8 hour)

3.2采集Kafka数据到Es

#=========================== Filebeat inputs =============================

filebeat.inputs:

# Each - is an input. Most options can be set at the input level, so
# you can use different inputs for various configurations.
# Below are the input specific configurations.

- type: kafka
  hosts:
    - kafka-01:9092
    - kafka-02:9092
    - kafka-03:9092
  topics: ["stu"]
  group_id: "test_group"
  # Change to true to enable this input configuration.
  #enabled: false
  
  
#==================== Elasticsearch template setting ==========================

setup.template.settings:
  index.number_of_shards: 3
  index.number_of_replicas: 1
  #index.codec: best_compression
  #_source.enabled: false
  
  
#================================ Outputs =====================================

# Configure what output to use when sending the data collected by the beat.

#-------------------------- Elasticsearch output ------------------------------
output.elasticsearch:
  hosts: ["es-01:9200","es-02:9200","es-03:9200"]
  username: "elastic"
  password: "123456"
  index: "student_%{+yyyyMM}"

setup.template.name: "student_"
setup.template.pattern: "student_*"
setup.ilm.enabled: false


#================================ Processors =====================================

# Configure processors to enhance or manipulate events generated by the beat.

# This allows to enable 6.7 migration aliases
#migration.6_to_7.enabled: true
processors:
  - drop_fields:
        fields: ['host','agent','ecs','kafka']
  - decode_json_fields:
        fields: ["message"]
        process_array: true
        max_depth: 1
        target: ""
        overwrite_keys: false
        add_error_key: true

3.3Logstash与FileBeat的对比

 

logstash

filebeat

语言

Jruby

golang

应用场景

采集kafka数据后过滤分析发送到es

采集日志,然后发送到消息队列

优势

丰富的input|filter|output插件

轻量级