You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

255 lines
7.6 KiB

/*****************************************************************************
* file.c: HTTP read-only file
*****************************************************************************
* Copyright (C) 2015 Rémi Denis-Courmont
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation; either version 2.1 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include <assert.h>
#include <errno.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <vlc_common.h>
#include <vlc_block.h>
#include <vlc_strings.h>
#include "message.h"
#include "resource.h"
#include "file.h"
#pragma GCC visibility push(default)
struct vlc_http_file
{
struct vlc_http_resource resource;
uintmax_t offset;
};
static int vlc_http_file_req(const struct vlc_http_resource *res,
struct vlc_http_msg *req, void *opaque)
{
struct vlc_http_file *file = (struct vlc_http_file *)res;
const uintmax_t *offset = opaque;
if (file->resource.response != NULL)
{
const char *str = vlc_http_msg_get_header(file->resource.response,
"ETag");
if (str != NULL)
{
if (!memcmp(str, "W/", 2))
str += 2; /* skip weak mark */
vlc_http_msg_add_header(req, "If-Match", "%s", str);
}
else
{
time_t mtime = vlc_http_msg_get_mtime(file->resource.response);
if (mtime != -1)
vlc_http_msg_add_time(req, "If-Unmodified-Since", &mtime);
}
}
if (vlc_http_msg_add_header(req, "Range", "bytes=%" PRIuMAX "-", *offset)
&& *offset != 0)
return -1;
return 0;
}
static int vlc_http_file_resp(const struct vlc_http_resource *res,
const struct vlc_http_msg *resp, void *opaque)
{
const uintmax_t *offset = opaque;
if (vlc_http_msg_get_status(resp) == 206)
{
const char *str = vlc_http_msg_get_header(resp, "Content-Range");
if (str == NULL)
/* A multipart/byteranges response. This is not what we asked for
* and we do not support it. */
goto fail;
uintmax_t start, end;
if (sscanf(str, "bytes %" SCNuMAX "-%" SCNuMAX, &start, &end) != 2
|| start != *offset || start > end)
/* A single range response is what we asked for, but not at that
* start offset. */
goto fail;
}
(void) res;
return 0;
fail:
errno = EIO;
return -1;
}
static const struct vlc_http_resource_cbs vlc_http_file_callbacks =
{
vlc_http_file_req,
vlc_http_file_resp,
};
struct vlc_http_resource *vlc_http_file_create(struct vlc_http_mgr *mgr,
const char *uri, const char *ua,
const char *ref)
{
struct vlc_http_file *file = malloc(sizeof (*file));
if (unlikely(file == NULL))
return NULL;
if (vlc_http_res_init(&file->resource, &vlc_http_file_callbacks, mgr,
uri, ua, ref))
{
free(file);
return NULL;
}
file->offset = 0;
return &file->resource;
}
static uintmax_t vlc_http_msg_get_file_size(const struct vlc_http_msg *resp)
{
int status = vlc_http_msg_get_status(resp);
const char *range = vlc_http_msg_get_header(resp, "Content-Range");
if (status == 206 /* Partial Content */)
{ /* IETF RFC7233 §4.1 */
assert(range != NULL); /* checked by vlc_http_file_resp() */
uintmax_t end, total;
switch (sscanf(range, "bytes %*u-%" SCNuMAX "/%" SCNuMAX, &end, &total))
{
case 1:
if (unlikely(end == UINTMAX_MAX))
return -1; /* avoid wrapping to zero */
return end + 1;
case 2:
return total;
}
vlc_assert_unreachable(); /* checked by vlc_http_file_resp() */
}
if (status == 416 /* Range Not Satisfiable */)
{ /* IETF RFC7233 §4.4 */
uintmax_t total;
if (range == NULL)
return -1; /* valid but helpless response */
if (sscanf(range, "bytes */%" SCNuMAX, &total) == 1)
return total; /* this occurs when seeking beyond EOF */
}
return -1;
}
static bool vlc_http_msg_can_seek(const struct vlc_http_msg *resp)
{
int status = vlc_http_msg_get_status(resp);
if (status == 206 || status == 416)
return true; /* Partial Content */
return vlc_http_msg_get_token(resp, "Accept-Ranges", "bytes") != NULL;
}
uintmax_t vlc_http_file_get_size(struct vlc_http_resource *res)
{
int status = vlc_http_res_get_status(res);
if (status < 0)
return -1;
uintmax_t ret = vlc_http_msg_get_file_size(res->response);
if (ret != (uintmax_t)-1)
return ret;
if (status >= 300 || status == 201)
return -1; /* Error or redirection, size is unknown/irrelevant. */
/* Content-Range is meaningless here (see RFC7233 B), so check if the size
* of the response entity body is known. */
return vlc_http_msg_get_size(res->response);
}
bool vlc_http_file_can_seek(struct vlc_http_resource *res)
{ /* See IETF RFC7233 */
int status = vlc_http_res_get_status(res);
if (status < 0)
return false;
return vlc_http_msg_can_seek(res->response);
}
int vlc_http_file_seek(struct vlc_http_resource *res, uintmax_t offset)
{
struct vlc_http_msg *resp = vlc_http_res_open(res, &offset);
if (resp == NULL)
return -1;
struct vlc_http_file *file = (struct vlc_http_file *)res;
int status = vlc_http_msg_get_status(resp);
if (res->response != NULL)
{ /* Accept the new and ditch the old one if:
* - requested succeeded and range was accepted (206),
* - requested failed due to out-of-range (416),
* - request succeeded and seek offset is zero (2xx).
*/
if (status != 206 && status != 416 && (offset != 0 || status >= 300))
{
vlc_http_msg_destroy(resp);
return -1;
}
vlc_http_msg_destroy(res->response);
}
res->response = resp;
file->offset = offset;
return 0;
}
block_t *vlc_http_file_read(struct vlc_http_resource *res)
{
struct vlc_http_file *file = (struct vlc_http_file *)res;
block_t *block = vlc_http_res_read(res);
if (block == vlc_http_error)
block = NULL;
/* Automatically resume on short response or error if possible */
if (block == NULL && res->response != NULL
&& vlc_http_msg_can_seek(res->response)
&& file->offset < vlc_http_msg_get_file_size(res->response)
&& vlc_http_file_seek(res, file->offset) == 0)
{
block = vlc_http_res_read(res);
if (block == vlc_http_error)
block = NULL; /* Non-recovered error */
}
if (block != NULL)
file->offset += block->i_buffer;
return block;
}