ZLMediaKit/src/Common/Parser.h

146 lines
4.0 KiB
C++
Raw Normal View History

2020-04-04 20:30:09 +08:00
/*
* Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
*
* This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit).
2020-04-04 20:30:09 +08:00
*
* Use of this source code is governed by MIT license that can be found in the
* LICENSE file in the root of the source tree. All contributing project authors
* may be found in the AUTHORS file in the root of the source tree.
*/
2019-06-28 16:48:02 +08:00
#ifndef ZLMEDIAKIT_PARSER_H
#define ZLMEDIAKIT_PARSER_H
#include <map>
#include <string>
#include "Util/util.h"
2021-09-30 16:10:09 +08:00
namespace mediakit {
2019-06-28 16:48:02 +08:00
2023-06-10 11:04:52 +08:00
// 从字符串中提取子字符串
std::string findSubString(const char *buf, const char *start, const char *end, size_t buf_size = 0);
2023-06-10 11:04:52 +08:00
// 把url解析为主机地址和端口号,兼容ipv4/ipv6/dns
void splitUrl(const std::string &url, std::string &host, uint16_t &port);
// 解析proxy url,仅支持http
void parseProxyUrl(const std::string &proxy_url, std::string &proxy_host, uint16_t &proxy_port, std::string &proxy_auth);
2019-06-28 16:48:02 +08:00
struct StrCaseCompare {
2023-06-10 11:04:52 +08:00
bool operator()(const std::string &__x, const std::string &__y) const { return strcasecmp(__x.data(), __y.data()) < 0; }
2019-06-28 16:48:02 +08:00
};
class StrCaseMap : public std::multimap<std::string, std::string, StrCaseCompare> {
2021-09-30 16:10:09 +08:00
public:
using Super = multimap<std::string, std::string, StrCaseCompare>;
2019-06-28 16:48:02 +08:00
StrCaseMap() = default;
~StrCaseMap() = default;
2019-07-17 14:50:24 +08:00
std::string &operator[](const std::string &k) {
2019-12-28 13:39:25 +08:00
auto it = find(k);
2021-09-30 16:10:09 +08:00
if (it == end()) {
it = Super::emplace(k, "");
2019-06-28 16:48:02 +08:00
}
return it->second;
}
2023-06-10 12:22:28 +08:00
template <typename K, typename V>
void emplace(K &&k, V &&v) {
2019-12-28 13:39:25 +08:00
auto it = find(k);
2021-09-30 16:10:09 +08:00
if (it != end()) {
2019-06-28 16:48:02 +08:00
return;
}
2023-06-10 12:22:28 +08:00
Super::emplace(std::forward<K>(k), std::forward<V>(v));
2019-06-28 16:48:02 +08:00
}
2023-06-10 12:22:28 +08:00
template <typename K, typename V>
void emplace_force(K &&k, V &&v) {
Super::emplace(std::forward<K>(k), std::forward<V>(v));
2019-06-28 16:48:02 +08:00
}
};
2023-06-10 11:04:52 +08:00
// rtsp/http/sip解析类
2019-06-28 16:48:02 +08:00
class Parser {
2020-04-20 18:13:45 +08:00
public:
2022-05-08 16:33:33 +08:00
Parser() = default;
~Parser() = default;
2021-09-30 16:10:09 +08:00
2023-06-10 12:22:28 +08:00
// 解析http/rtsp/sip请求需要确保buf以\0结尾
void parse(const char *buf, size_t size);
2021-09-30 16:10:09 +08:00
2023-06-10 12:22:28 +08:00
// 获取命令字如GET/POST
2023-06-10 11:04:52 +08:00
const std::string &method() const;
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 请求时获取中间url不包含?后面的参数
const std::string &url() const;
2023-06-10 12:22:28 +08:00
// 回复时获取状态码如200/404
2023-06-10 11:04:52 +08:00
const std::string &status() const;
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 获取中间url包含?后面的参数
std::string fullUrl() const;
2021-09-30 16:10:09 +08:00
2023-06-10 12:22:28 +08:00
// 请求时获取协议名如HTTP/1.1
2023-06-10 11:04:52 +08:00
const std::string &protocol() const;
2023-06-10 12:22:28 +08:00
// 回复时,获取状态字符串,如 OK/Not Found
2023-06-10 11:04:52 +08:00
const std::string &statusStr() const;
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 根据header key名获取请求header value值
const std::string &operator[](const char *name) const;
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 获取http body或sdp
const std::string &content() const;
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 清空,为了重用
void clear();
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 获取?后面的参数
const std::string &params() const;
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 重新设置url
void setUrl(std::string url);
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 重新设置content
void setContent(std::string content);
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 获取header列表
2020-04-20 18:13:45 +08:00
StrCaseMap &getHeader() const;
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 获取url参数列表
2020-04-20 18:13:45 +08:00
StrCaseMap &getUrlArgs() const;
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
// 解析?后面的参数
static StrCaseMap parseArgs(const std::string &str, const char *pair_delim = "&", const char *key_delim = "=");
2021-09-30 16:10:09 +08:00
2023-06-10 11:04:52 +08:00
static std::string mergeUrl(const std::string &base_url, const std::string &path);
2019-06-28 16:55:28 +08:00
private:
2023-06-10 12:22:28 +08:00
std::string _method;
std::string _url;
std::string _protocol;
std::string _content;
std::string _params;
2023-06-10 12:22:28 +08:00
mutable StrCaseMap _headers;
mutable StrCaseMap _url_args;
2019-06-28 16:48:02 +08:00
};
2023-06-10 11:04:52 +08:00
// 解析rtsp url的工具类
class RtspUrl {
2022-05-08 16:33:33 +08:00
public:
bool _is_ssl;
uint16_t _port;
std::string _url;
std::string _user;
std::string _passwd;
std::string _host;
public:
RtspUrl() = default;
~RtspUrl() = default;
void parse(const std::string &url);
private:
2023-06-10 11:04:52 +08:00
void setup(bool, const std::string &, const std::string &, const std::string &);
2022-05-08 16:33:33 +08:00
};
2023-06-10 11:04:52 +08:00
} // namespace mediakit
2019-06-28 16:48:02 +08:00
2023-06-10 11:04:52 +08:00
#endif // ZLMEDIAKIT_PARSER_H