lighttpd对flv视频文件的拖动一种变形

当前流行的视频网站,关于视频的在线拖动观看技术的小试,

1.flv文件
  flv视频文件,想大家都比较清楚,只要从关键帧的位置切开,并加上"FLV\x1\x1\0\0\0\x9\0\0\0\x9"公共头标志信息,后对任何支持flv视频文件dencoder的播放器都是可以正常播放的
  
  "FLV\x1\x1\0\0\0\x9\0\0\0\x9" + keyframe(body)

2.lighttpd中的mod_flv_streaming.c模块 (重点)

  http://www.lighttpd.net/
 
  http://blog.lighttpd.net/articles/2006/03/09/flv-streaming-with-lighttpd

  这里通过http的get方法,向服务器请求指定的视频片段

  形如: http://youserver.com/flv/abcdefg.flv?start=12345

  mod_flv_streaming.c默认的方式是,从start(文件的物理位置offset)的开始,默认到文件结束。
 
   相关服务器代码如下:

for (k = 0; k < p->conf.extensions->used; k++) {
		data_string *ds = (data_string *)p->conf.extensions->data[k];
		int ct_len = ds->value->used - 1;

		if (ct_len > s_len) continue;
		if (ds->value->used == 0) continue;

		if (0 == strncmp(con->physical.path->ptr + s_len - ct_len, ds->value->ptr, ct_len)) {
			data_string *get_param;
			stat_cache_entry *sce = NULL;
			buffer *b;
			int start;
			char *err = NULL;
			/* if there is a start=[0-9]+ in the header use it as start,
			 * otherwise send the full file */

			array_reset(p->get_params);
			buffer_copy_string_buffer(p->query_str, con->uri.query);
			split_get_params(p->get_params, p->query_str);
/*这个是重点,获得客户端的播放器通过get方法传来的start(文件拖动的开始位置offset)*/
			if (NULL == (get_param = (data_string *)array_get_element(p->get_params, "start"))) {
				return HANDLER_GO_ON;
			}

			/* too short */
			if (get_param->value->used < 2) return HANDLER_GO_ON;

			/* check if it is a number */
			start = strtol(get_param->value->ptr, &err, 10);
			if (*err != '\0') {
				return HANDLER_GO_ON;
			}

			if (start <= 0) return HANDLER_GO_ON;

			/* check if start is > filesize */
			if (HANDLER_GO_ON != stat_cache_get_entry(srv, con, con->physical.path, &sce)) {
				return HANDLER_GO_ON;
			}

			if (start > sce->st.st_size) {
				return HANDLER_GO_ON;
			}

			/* we are safe now, let's build a flv header */
			b = chunkqueue_get_append_buffer(con->write_queue);
                        
                        /*准备发送给客户端播放器的flv公共头信息13字节*/
			buffer_copy_string_len(b, CONST_STR_LEN("FLV\x1\x1\0\0\0\x9\0\0\0\x9"));
                        
                        /*发送seek到的拖动关键桢的位置keyframe, 并计算出从拖动点到该视频文件结束的长度*/
			http_chunk_append_file(srv, con, con->physical.path, start, sce->st.st_size - start);
                       /* http相关信息的填充*/
			response_header_overwrite(srv, con, CONST_STR_LEN("Content-Type"), CONST_STR_LEN("video/x-flv"));

			con->file_finished = 1;
                       /*done, 处理拖动完成*/
			return HANDLER_FINISHED


3 对客户端的播放器而言,服务器对flv文件的拖动支持是透明的,
  只要播放器安照合法的get方法向服务器请求,即可得到可以播放的视频片段,
  但是这里的start=xxxx,一定是flv视频文件的关键桢位置,否则不能播放
  一般而言,对flv视频文件可以使用
  yamdi
  http://yamdi.sourceforge.net/
  加上metadata信息,即可完美支持拖动,
  播放器可以再第一次请求视频文件的时候获得该视频的metadata信息(即包含关键桢,时间,等等与视频相关的信息)

4.以上讨论的,都是lighttpd-mod_flv_streaming.c默认的对flv的拖动支持,
  其实可以,做一些简单的扩展,
  可以增加end参数,即是增加拖动的结束参数,
  假设,很多用户并不会观看视频到结束,
  每次请求N字节
  http://youserver.com/flv/abcdefg.flv?start=23456&end=23456+N
  播放器可以增加这样的功能,
  a.先请求某一段视频
  b.当当前视频片段没有播放完毕之前,不会去请求下一个片段,
    只有当当前视频片段在即将播放结束的时候,再去请求下一个片段,
    这样,当用户,看了前面的某一片段后,突然关闭该播放页,以至于不会白白浪费掉那已经download到本地,但是并没有观看的视频,节余带宽

  粗略代码实现如下:


if (NULL != (get_param = (data_string *)array_get_element(p->get_params, "start"))) 
{

    /* too short */
    if (get_param->value->used < 2) 
					return HANDLER_GO_ON;

			    /* check if it is a number */
			    start = strtol(get_param->value->ptr, &err, 10);
			    if (*err != '\0') 
				{
				    return HANDLER_GO_ON;
			    }
			    
			    /* check if tflvbegin is >= 0 */
			    if (start < 0) 
					return HANDLER_GO_ON;

			    /* check if start is > filesize */
			    if (start > sce->st.st_size) 
				{
				    return HANDLER_GO_ON;
			    }
			}
			else 
			{
				return HANDLER_GO_ON;
			}
                        
			/* if there is a start=[0-9]+ in the header use it as end,
			 * otherwise send the full file */
/*这里重点,仿照lighttpd如何获得get方法的中的start,同理获得end参数, 并做一些必要的合法性检查*/
			if (NULL != (get_param = (data_string *)array_get_element(p->get_params, "end"))) 
			{
			    /* too short */
			    if (get_param->value->used < 2) 
					return HANDLER_GO_ON;

			    /* check if it is a number */
			    end = strtol(get_param->value->ptr, &err, 10);
			    if (*err != '\0') 
				{
				    return HANDLER_GO_ON;
			    }

			    /*参数检查,必须*/
			    /* check if end is > 0 
			     * check if start < end 
			     * make sure star > 0 
			     * */
			    if (end <= 0 || start >= end) 
					return HANDLER_GO_ON;

			    /* check if end is > filesize */
			    if (end > sce->st.st_size) 
				{
					//return HANDLER_GO_ON;
					tflvend = sce->st.st_size; /* path tflvend is not right */
			    }
			}
			else 
			{
				return HANDLER_GO_ON;
			
    
   }

/* we are safe now, let's build a flv header */
			b = chunkqueue_get_append_buffer(con->write_queue);
                        
                        /*准备发送给客户端播放器的flv公共头信息13字节*/
			buffer_copy_string_len(b, CONST_STR_LEN("FLV\x1\x1\0\0\0\x9\0\0\0\x9"));

                      /*这个是重点,以前默认的是发送的数据长度是用 fileseize-start, 现在要用end替代filesize, 即是 end-start*/
			http_chunk_append_file(srv, con, con->physical.path, start, end - start);
			response_header_overwrite(srv, con, CONST_STR_LEN("Content-Type"), CONST_STR_LEN("video/x-flv"));
			con->file_finished = 1;
			return HANDLER_FINISHED



-------------------------------------
gamil:[email protected]

你可能感兴趣的:(C++,c,lighttpd,Gmail,Go)