|
Packit |
90a5c9 |
/* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
Packit |
90a5c9 |
* contributor license agreements. See the NOTICE file distributed with
|
|
Packit |
90a5c9 |
* this work for additional information regarding copyright ownership.
|
|
Packit |
90a5c9 |
* The ASF licenses this file to You under the Apache License, Version 2.0
|
|
Packit |
90a5c9 |
* (the "License"); you may not use this file except in compliance with
|
|
Packit |
90a5c9 |
* the License. You may obtain a copy of the License at
|
|
Packit |
90a5c9 |
*
|
|
Packit |
90a5c9 |
* http://www.apache.org/licenses/LICENSE-2.0
|
|
Packit |
90a5c9 |
*
|
|
Packit |
90a5c9 |
* Unless required by applicable law or agreed to in writing, software
|
|
Packit |
90a5c9 |
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
Packit |
90a5c9 |
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
Packit |
90a5c9 |
* See the License for the specific language governing permissions and
|
|
Packit |
90a5c9 |
* limitations under the License.
|
|
Packit |
90a5c9 |
*/
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
#include <assert.h>
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
#include <apr_lib.h>
|
|
Packit |
90a5c9 |
#include <apr_strings.h>
|
|
Packit |
90a5c9 |
#include <apr_buckets.h>
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
#include "md_http.h"
|
|
Packit |
90a5c9 |
#include "md_log.h"
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
struct md_http_t {
|
|
Packit |
90a5c9 |
apr_pool_t *pool;
|
|
Packit |
90a5c9 |
apr_bucket_alloc_t *bucket_alloc;
|
|
Packit |
90a5c9 |
apr_off_t resp_limit;
|
|
Packit |
90a5c9 |
md_http_impl_t *impl;
|
|
Packit |
90a5c9 |
const char *user_agent;
|
|
Packit |
90a5c9 |
const char *proxy_url;
|
|
Packit |
90a5c9 |
};
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
static md_http_impl_t *cur_impl;
|
|
Packit |
90a5c9 |
static int cur_init_done;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
void md_http_use_implementation(md_http_impl_t *impl)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
if (cur_impl != impl) {
|
|
Packit |
90a5c9 |
cur_impl = impl;
|
|
Packit |
90a5c9 |
cur_init_done = 0;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
static long next_req_id;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
apr_status_t md_http_create(md_http_t **phttp, apr_pool_t *p, const char *user_agent,
|
|
Packit |
90a5c9 |
const char *proxy_url)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
md_http_t *http;
|
|
Packit |
90a5c9 |
apr_status_t rv = APR_SUCCESS;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
if (!cur_impl) {
|
|
Packit |
90a5c9 |
*phttp = NULL;
|
|
Packit |
90a5c9 |
return APR_ENOTIMPL;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
if (!cur_init_done) {
|
|
Packit |
90a5c9 |
if (APR_SUCCESS == (rv = cur_impl->init())) {
|
|
Packit |
90a5c9 |
cur_init_done = 1;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
else {
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
http = apr_pcalloc(p, sizeof(*http));
|
|
Packit |
90a5c9 |
http->pool = p;
|
|
Packit |
90a5c9 |
http->impl = cur_impl;
|
|
Packit |
90a5c9 |
http->user_agent = apr_pstrdup(p, user_agent);
|
|
Packit |
90a5c9 |
http->proxy_url = proxy_url? apr_pstrdup(p, proxy_url) : NULL;
|
|
Packit |
90a5c9 |
http->bucket_alloc = apr_bucket_alloc_create(p);
|
|
Packit |
90a5c9 |
if (!http->bucket_alloc) {
|
|
Packit |
90a5c9 |
return APR_EGENERAL;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
*phttp = http;
|
|
Packit |
90a5c9 |
return APR_SUCCESS;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
void md_http_set_response_limit(md_http_t *http, apr_off_t resp_limit)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
http->resp_limit = resp_limit;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
static apr_status_t req_create(md_http_request_t **preq, md_http_t *http,
|
|
Packit |
90a5c9 |
const char *method, const char *url, struct apr_table_t *headers,
|
|
Packit |
90a5c9 |
md_http_cb *cb, void *baton)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
md_http_request_t *req;
|
|
Packit |
90a5c9 |
apr_pool_t *pool;
|
|
Packit |
90a5c9 |
apr_status_t rv;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
rv = apr_pool_create(&pool, http->pool);
|
|
Packit |
90a5c9 |
if (rv != APR_SUCCESS) {
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
req = apr_pcalloc(pool, sizeof(*req));
|
|
Packit |
90a5c9 |
req->id = next_req_id++;
|
|
Packit |
90a5c9 |
req->pool = pool;
|
|
Packit |
90a5c9 |
req->bucket_alloc = http->bucket_alloc;
|
|
Packit |
90a5c9 |
req->http = http;
|
|
Packit |
90a5c9 |
req->method = method;
|
|
Packit |
90a5c9 |
req->url = url;
|
|
Packit |
90a5c9 |
req->headers = headers? apr_table_copy(req->pool, headers) : apr_table_make(req->pool, 5);
|
|
Packit |
90a5c9 |
req->resp_limit = http->resp_limit;
|
|
Packit |
90a5c9 |
req->cb = cb;
|
|
Packit |
90a5c9 |
req->baton = baton;
|
|
Packit |
90a5c9 |
req->user_agent = http->user_agent;
|
|
Packit |
90a5c9 |
req->proxy_url = http->proxy_url;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
*preq = req;
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
void md_http_req_destroy(md_http_request_t *req)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
if (req->internals) {
|
|
Packit |
90a5c9 |
req->http->impl->req_cleanup(req);
|
|
Packit |
90a5c9 |
req->internals = NULL;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
apr_pool_destroy(req->pool);
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
static apr_status_t schedule(md_http_request_t *req,
|
|
Packit |
90a5c9 |
apr_bucket_brigade *body, int detect_clen,
|
|
Packit |
90a5c9 |
long *preq_id)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
apr_status_t rv;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
req->body = body;
|
|
Packit |
90a5c9 |
req->body_len = body? -1 : 0;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
if (req->body && detect_clen) {
|
|
Packit |
90a5c9 |
rv = apr_brigade_length(req->body, 1, &req->body_len);
|
|
Packit |
90a5c9 |
if (rv != APR_SUCCESS) {
|
|
Packit |
90a5c9 |
md_http_req_destroy(req);
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
if (req->body_len == 0 && apr_strnatcasecmp("GET", req->method)) {
|
|
Packit |
90a5c9 |
apr_table_setn(req->headers, "Content-Length", "0");
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
else if (req->body_len > 0) {
|
|
Packit |
90a5c9 |
apr_table_setn(req->headers, "Content-Length", apr_off_t_toa(req->pool, req->body_len));
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
if (preq_id) {
|
|
Packit |
90a5c9 |
*preq_id = req->id;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
/* we send right away */
|
|
Packit |
90a5c9 |
rv = req->http->impl->perform(req);
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
apr_status_t md_http_GET(struct md_http_t *http,
|
|
Packit |
90a5c9 |
const char *url, struct apr_table_t *headers,
|
|
Packit |
90a5c9 |
md_http_cb *cb, void *baton, long *preq_id)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
md_http_request_t *req;
|
|
Packit |
90a5c9 |
apr_status_t rv;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
rv = req_create(&req, http, "GET", url, headers, cb, baton);
|
|
Packit |
90a5c9 |
if (rv != APR_SUCCESS) {
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
return schedule(req, NULL, 0, preq_id);
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
apr_status_t md_http_HEAD(struct md_http_t *http,
|
|
Packit |
90a5c9 |
const char *url, struct apr_table_t *headers,
|
|
Packit |
90a5c9 |
md_http_cb *cb, void *baton, long *preq_id)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
md_http_request_t *req;
|
|
Packit |
90a5c9 |
apr_status_t rv;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
rv = req_create(&req, http, "HEAD", url, headers, cb, baton);
|
|
Packit |
90a5c9 |
if (rv != APR_SUCCESS) {
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
return schedule(req, NULL, 0, preq_id);
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
apr_status_t md_http_POST(struct md_http_t *http, const char *url,
|
|
Packit |
90a5c9 |
struct apr_table_t *headers, const char *content_type,
|
|
Packit |
90a5c9 |
apr_bucket_brigade *body,
|
|
Packit |
90a5c9 |
md_http_cb *cb, void *baton, long *preq_id)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
md_http_request_t *req;
|
|
Packit |
90a5c9 |
apr_status_t rv;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
rv = req_create(&req, http, "POST", url, headers, cb, baton);
|
|
Packit |
90a5c9 |
if (rv != APR_SUCCESS) {
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
if (content_type) {
|
|
Packit |
90a5c9 |
apr_table_set(req->headers, "Content-Type", content_type);
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
return schedule(req, body, 1, preq_id);
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
apr_status_t md_http_POSTd(md_http_t *http, const char *url,
|
|
Packit |
90a5c9 |
struct apr_table_t *headers, const char *content_type,
|
|
Packit |
90a5c9 |
const char *data, size_t data_len,
|
|
Packit |
90a5c9 |
md_http_cb *cb, void *baton, long *preq_id)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
md_http_request_t *req;
|
|
Packit |
90a5c9 |
apr_status_t rv;
|
|
Packit |
90a5c9 |
apr_bucket_brigade *body = NULL;
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
rv = req_create(&req, http, "POST", url, headers, cb, baton);
|
|
Packit |
90a5c9 |
if (rv != APR_SUCCESS) {
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
if (data && data_len > 0) {
|
|
Packit |
90a5c9 |
body = apr_brigade_create(req->pool, req->http->bucket_alloc);
|
|
Packit |
90a5c9 |
rv = apr_brigade_write(body, NULL, NULL, data, data_len);
|
|
Packit |
90a5c9 |
if (rv != APR_SUCCESS) {
|
|
Packit |
90a5c9 |
md_http_req_destroy(req);
|
|
Packit |
90a5c9 |
return rv;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
if (content_type) {
|
|
Packit |
90a5c9 |
apr_table_set(req->headers, "Content-Type", content_type);
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
return schedule(req, body, 1, preq_id);
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|
|
Packit |
90a5c9 |
apr_status_t md_http_await(md_http_t *http, long req_id)
|
|
Packit |
90a5c9 |
{
|
|
Packit |
90a5c9 |
(void)http;
|
|
Packit |
90a5c9 |
(void)req_id;
|
|
Packit |
90a5c9 |
return APR_SUCCESS;
|
|
Packit |
90a5c9 |
}
|
|
Packit |
90a5c9 |
|