加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
mysql_decoder.h 6.75 KB
一键复制 编辑 原始数据 按行查看 历史
xyz347 提交于 2023-06-18 08:22 . support yaml sqlite (#39)
/*
* Copyright (C) 2021 Duowan Inc. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef __X_PACK_MYSQL_DECODER_H
#define __X_PACK_MYSQL_DECODER_H
#include <stdexcept>
#include <map>
#include <vector>
#include <cstdlib>
#include <mysql/mysql.h>
#include "extend.h"
#include "traits.h"
#include "json.h"
#include "string.h"
#include "time.h"
namespace xpack {
// for mysql types not list here
// if true we will call xpack_mysql_decode(MYSQL_FIELD *field, char *column, T&val, const Extend *ext)
template <class T>
struct is_xpack_mysql_xtype {static bool const value = false;};
class MySQLDecoder {
friend class mysql;
public:
template <typename T>
bool decode(const char*key, T&val, const Extend *ext) {
int idx = find(key);
if (idx >= 0) {
return this->decode_type(idx, val, ext);
}
return false;
}
const char *Name() const {
return "db";
}
private:
MYSQL_RES *_res;
MYSQL_ROW _row;
std::map<const char*, int, cmp_str> _index;
MySQLDecoder(MYSQL_RES *result):_res(result), _row(NULL) {
mysql_data_seek(_res, 0); // reset cursor to head
for (int i=0; i<int(result->field_count); ++i) {
_index[result->fields[i].name] = i;
}
}
// decode to struct
template <class T>
inline XPACK_IS_XPACK(T) decode_top(T& val, const Extend *ext) {
if (NULL != (_row = mysql_fetch_row(_res))) {
val.__x_pack_decode(*this, val, ext);
return true;
}
return false;
}
template <class T>
inline XPACK_IS_XPACK(T) decode_top(std::vector<T>& val, const Extend *ext) {
while (NULL != (_row = mysql_fetch_row(_res))) {
val.push_back(T());
T& tmp = val.back();
tmp.__x_pack_decode(*this, tmp, ext);
}
return true;
}
template <class T>
inline XPACK_IS_XOUT(T) decode_top(T& val, const Extend *ext) {
if (NULL != (_row = mysql_fetch_row(_res))) {
__x_pack_decode_out(*this, val, ext);
}
return true;
}
template <class T>
inline XPACK_IS_XOUT(T) decode_top(std::vector<T>& val, const Extend *ext) {
while (NULL != (_row = mysql_fetch_row(_res))) {
val.push_back(T());
__x_pack_decode_out(*this, val.back(), ext);
}
return true;
}
// decode special column of first row
template <class T>
bool decode_column(const char*field, T&val, const Extend *ext) {
int idx;
if (NULL==_row && NULL == (_row = mysql_fetch_row(_res)) && (0 > (idx = find(field)))) {
return false;
}
return decode_type(idx, val, ext);
}
// decode special column of rows
template <class T>
bool decode_column(const char*field, std::vector<T>&val, const Extend *ext) {
int idx;
if (0 > (idx = find(field))) {
return false;
}
while (NULL != (_row = mysql_fetch_row(_res))) {
T t;
decode_type(idx, t, ext);
val.push_back(t);
}
return true;
}
int find(const char*field) const {
std::map<const char*, int, cmp_str>::const_iterator it = _index.find(field);
if (it != _index.end()) {
return it->second;
}
return -1;
}
// std::string
bool decode_type(const int idx, std::string &val, const Extend *ext) {
val = _row[idx];
return true;
}
// bool
bool decode_type(const int idx, bool &val, const Extend *ext) {
int tmp;
if (decode_type(idx, tmp, ext)) {
val = tmp != 0;
return true;
}
return false;
}
// unsigned integer. DATE and TIME use signed plz
template <class T>
typename x_enable_if<numeric<T>::is_integer && !numeric<T>::is_signed, bool>::type decode_type(const int idx, T &val, const Extend *ext) {
val = (T)std::strtoul(_row[idx], NULL, 10);
return true;
}
// signed integer
template <class T>
typename x_enable_if<numeric<T>::is_integer && numeric<T>::is_signed, bool>::type decode_type(const int idx, T &val, const Extend *ext) {
const char *str = _row[idx];
if (str==NULL || str[0]=='\0') {
val = 0;
return true;
}
switch (_res->fields[idx].type) {
case MYSQL_TYPE_TIME: {
std::vector<std::string> ss;
if (3 == Util::split(ss, str, ':')) {
long int s0 = std::strtol(ss[0].c_str(), NULL, 10);
long int s1 = std::strtol(ss[1].c_str(), NULL, 10);
long int s2 = std::strtol(ss[2].c_str(), NULL, 10);
if (s0 >= 0) {
val = 1;
} else {
val = -1;
s0 *= -1;
}
val *= (T)(s0*3600 + s1*60 +s2);
}
}
break;
case MYSQL_TYPE_DATETIME:
case MYSQL_TYPE_TIMESTAMP:{ // parse to unix timestamp, seconds, discard fractional part
tm t;
strptime(str, "%F %T", &t);
val = (T)mktime(&t);
}
break;
default:
val = (T)std::strtol(str, NULL, 10);
}
return true;
}
// float
template <class T>
typename x_enable_if<numeric<T>::is_float, bool>::type decode_type(const int idx, T &val, const Extend *ext) {
val = (T)std::strtod(_row[idx], NULL);
return true;
}
// mysql covert type
template <class T>
inline typename x_enable_if<is_xpack_mysql_xtype<T>::value, bool>::type decode_type(const int idx, T &val, const Extend *ext) {
return xpack_mysql_decode(&_res->fields[idx], _row[idx], val, ext);
}
// class/struct defined XPACK/XPACK_OUT, default use json to parse, if use other, use xpack_mysql_decode
template <class T>
inline XPACK_IS_XOUT(T) decode_type(const int idx, T &val, const Extend *ext) {
(void)ext;
return xpack::json::decode(_row[idx], val);
}
template <class T>
inline XPACK_IS_XPACK(T) decode_type(const int idx, T &val, const Extend *ext) {
(void)ext;
return xpack::json::decode(_row[idx], val);
}
};
}
#endif
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化