libgnomevfs2-mythtv/modules/mythtv-method.c
author rosfran
Fri Jan 19 18:52:47 2007 +0000 (2007-01-19)
branchtrunk
changeset 282 dde6d4e8b881
parent 277 890b023c759f
child 290 96b436a81e53
permissions -rwxr-xr-x
[svn r283] Revert back an old version of gmyth_livetv_setup; bug commited yesterday, and was caused by an invalid MythTV database select statement.
melunko@38
     1
/*
melunko@38
     2
 * @author Hallyson Melo <hallyson.melo@indt.org.br>
melunko@38
     3
 *
melunko@38
     4
 * This program is free software; you can redistribute it and/or modify
melunko@38
     5
 * it under the terms of the GNU Lesser General Public License as published by
melunko@38
     6
 * the Free Software Foundation; either version 2 of the License, or
melunko@38
     7
 * (at your option) any later version.
melunko@38
     8
 *
melunko@38
     9
 * This program is distributed in the hope that it will be useful,
melunko@38
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
melunko@38
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
melunko@38
    12
 * GNU General Public License for more details.
melunko@38
    13
 *
melunko@38
    14
 * You should have received a copy of the GNU Lesser General Public License
melunko@38
    15
 * along with this program; if not, write to the Free Software
melunko@38
    16
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
melunko@38
    17
 */
melunko@38
    18
melunko@38
    19
#ifdef HAVE_CONFIG_H
melunko@38
    20
#include <config.h>
melunko@38
    21
#endif
melunko@38
    22
melunko@38
    23
#include <string.h>
melunko@38
    24
#include <glib.h>
melunko@38
    25
#include <math.h>
melunko@38
    26
melunko@38
    27
#include <libgnomevfs/gnome-vfs-module.h>
melunko@38
    28
#include <libgnomevfs/gnome-vfs-utils.h>
melunko@38
    29
rosfran@277
    30
#include <gmyth/gmyth_file_transfer.h>
rosfran@277
    31
#include <gmyth/gmyth_livetv.h>
rosfran@277
    32
#include <gmyth/gmyth_uri.h>
rosfran@277
    33
#include <gmyth/gmyth_recorder.h>
rosfran@277
    34
#include <gmyth/gmyth_backendinfo.h>
rosfran@277
    35
#include <gmyth/gmyth_util.h>
melunko@38
    36
melunko@38
    37
#define GST_MYTHTV_ID_NUM               1
melunko@38
    38
#define MYTHTV_VERSION_DEFAULT          30
rosfran@277
    39
#define MYTHTV_TRANSFER_MAX_WAITS       100
melunko@38
    40
melunko@111
    41
#define MYTHTV_BUFFER_SIZE		1024*64
melunko@111
    42
melunko@38
    43
static GnomeVFSResult do_read (GnomeVFSMethod *method,
melunko@38
    44
                               GnomeVFSMethodHandle *method_handle,
melunko@38
    45
                               gpointer buffer,
melunko@38
    46
                               GnomeVFSFileSize num_bytes,
melunko@38
    47
                               GnomeVFSFileSize *bytes_read,
melunko@38
    48
                               GnomeVFSContext *context);
melunko@38
    49
melunko@38
    50
typedef struct {
melunko@48
    51
    GMythFileTransfer *file_transfer;
rosfran@277
    52
    GMythLiveTV 			*livetv;
rosfran@277
    53
    gint 							channel_num;
melunko@38
    54
    
melunko@38
    55
    gint mythtv_version;
melunko@38
    56
    guint64 content_size;
melunko@38
    57
    guint64 bytes_read;
melunko@111
    58
rosfran@116
    59
    GByteArray *buffer;
melunko@111
    60
    gsize buffer_remain;
melunko@38
    61
} MythtvHandle;
melunko@38
    62
melunko@38
    63
melunko@38
    64
melunko@38
    65
static GnomeVFSResult
melunko@38
    66
do_open (GnomeVFSMethod *method,
melunko@38
    67
         GnomeVFSMethodHandle **method_handle,
melunko@38
    68
         GnomeVFSURI *uri,
melunko@38
    69
         GnomeVFSOpenMode mode,
melunko@38
    70
         GnomeVFSContext *context)
melunko@38
    71
{
melunko@111
    72
    MythtvHandle *myth_handle;
melunko@160
    73
    GMythBackendInfo *backend_info;
rosfran@277
    74
    GMythURI *gmyth_uri = NULL;
rosfran@277
    75
    gboolean ret = TRUE;
rosfran@277
    76
    gboolean is_livetv = FALSE;
rosfran@277
    77
    gint wait_to_transfer = 0;
melunko@38
    78
melunko@38
    79
    _GNOME_VFS_METHOD_PARAM_CHECK (method_handle != NULL);
melunko@38
    80
    _GNOME_VFS_METHOD_PARAM_CHECK (uri != NULL);
melunko@38
    81
melunko@111
    82
    myth_handle = g_new0 (MythtvHandle, 1);
melunko@111
    83
melunko@38
    84
    if (mode & GNOME_VFS_OPEN_WRITE) {
melunko@38
    85
        return GNOME_VFS_ERROR_NOT_PERMITTED;
melunko@38
    86
    }
melunko@38
    87
renatofilho@149
    88
    if (gnome_vfs_uri_get_host_name (uri) == NULL) {
renatofilho@149
    89
    	return GNOME_VFS_ERROR_INVALID_HOST_NAME;
melunko@38
    90
    }
melunko@38
    91
melunko@38
    92
    /* Initialize mythtv handler*/
melunko@38
    93
    myth_handle->file_transfer = NULL;
rosfran@277
    94
    myth_handle->livetv = NULL;
melunko@38
    95
    myth_handle->mythtv_version = MYTHTV_VERSION_DEFAULT;
melunko@38
    96
    myth_handle->bytes_read = 0;
melunko@38
    97
    myth_handle->content_size = -1;
melunko@38
    98
rosfran@277
    99
    /* Creates and fills out the backend info structure */    
rosfran@281
   100
	  backend_info = gmyth_backend_info_new_with_uri ( 
rosfran@281
   101
	  			gnome_vfs_unescape_string( gnome_vfs_uri_to_string( uri, GNOME_VFS_URI_HIDE_NONE ), "" ) );
rosfran@277
   102
	  
rosfran@277
   103
	  /* creates an instance of  */  
rosfran@281
   104
	  gmyth_uri = gmyth_uri_new_with_value( 
rosfran@281
   105
	  		gnome_vfs_unescape_string( gnome_vfs_uri_to_string( uri, GNOME_VFS_URI_HIDE_NONE ), "" ) );
rosfran@277
   106
	  
rosfran@277
   107
	  /* Connect to the backend */	  
rosfran@277
   108
	  if ( gmyth_uri != NULL && ( is_livetv = gmyth_uri_is_livetv( gmyth_uri ) ) == TRUE ) {
rosfran@277
   109
	    myth_handle->livetv = gmyth_livetv_new ();
rosfran@180
   110
	    
rosfran@277
   111
	    myth_handle->channel_num = gmyth_uri_get_channel_num( gmyth_uri );
rosfran@277
   112
	
rosfran@277
   113
	    if ( myth_handle->channel_num != -1 ) {
rosfran@277
   114
	      if (gmyth_livetv_channel_setup (myth_handle->livetv, myth_handle->channel_num,
rosfran@277
   115
	              backend_info) == FALSE) {
rosfran@277
   116
	        g_object_unref( gmyth_uri );
rosfran@277
   117
	        ret = FALSE;
rosfran@277
   118
	      }
rosfran@277
   119
	    } else {
rosfran@277
   120
	      if ( gmyth_livetv_setup (myth_handle->livetv, backend_info) == FALSE ) {
rosfran@277
   121
	      	g_object_unref( gmyth_uri );
rosfran@277
   122
	        ret = FALSE;
rosfran@277
   123
	      }
rosfran@277
   124
	    }
rosfran@277
   125
	
rosfran@277
   126
	    myth_handle->file_transfer = gmyth_livetv_create_file_transfer (myth_handle->livetv);
rosfran@277
   127
	
rosfran@277
   128
	    if (NULL == myth_handle->file_transfer) {
rosfran@277
   129
	      ret = FALSE;
rosfran@277
   130
	    }
rosfran@277
   131
	    
rosfran@277
   132
	    if ( gmyth_uri != NULL )
rosfran@277
   133
	    	g_object_unref( gmyth_uri );
rosfran@277
   134
	    	
rosfran@277
   135
	  } else {
rosfran@277
   136
	
rosfran@277
   137
	    myth_handle->file_transfer = gmyth_file_transfer_new (backend_info);
rosfran@277
   138
	    
rosfran@277
   139
	    /* Verifies if the file exists */
rosfran@277
   140
	    if (!gmyth_util_file_exists (backend_info, gnome_vfs_uri_get_path (uri))) {
rosfran@277
   141
	        g_object_unref (backend_info);
rosfran@277
   142
					return GNOME_VFS_ERROR_NOT_FOUND;
rosfran@277
   143
	    }
rosfran@277
   144
	    
rosfran@277
   145
	    /* sets the Playback monitor connection */
rosfran@277
   146
	    ret = gmyth_file_transfer_open ( myth_handle->file_transfer, gnome_vfs_uri_get_path (uri) );
rosfran@277
   147
		
rosfran@277
   148
	  } /* if - LiveTV or not? */
rosfran@277
   149
	  
rosfran@277
   150
    if (ret == FALSE) {
rosfran@277
   151
	    g_warning ("MythTV FileTransfer open error\n");
rosfran@277
   152
	    return GNOME_VFS_ERROR_NOT_OPEN;
rosfran@277
   153
	  } 
rosfran@277
   154
	
rosfran@277
   155
	  if ( is_livetv == TRUE && ret == TRUE ) {
rosfran@277
   156
	    /* loop finished, set the max tries variable to zero again... */
rosfran@277
   157
	    wait_to_transfer = 0;
rosfran@277
   158
	
rosfran@277
   159
	    while ( wait_to_transfer++ < MYTHTV_TRANSFER_MAX_WAITS &&
rosfran@277
   160
	        (gmyth_livetv_is_recording (myth_handle->livetv) == FALSE))
rosfran@277
   161
	      g_usleep (500);
rosfran@277
   162
	
rosfran@277
   163
	    /* IS_RECORDING again, just like the MythTV backend does... */
rosfran@277
   164
	    gmyth_livetv_is_recording (myth_handle->livetv);
rosfran@277
   165
	
rosfran@277
   166
	    sleep (4);                  /* FIXME: this is evil (tpm) */
rosfran@277
   167
	  }
melunko@251
   168
renatofilho@188
   169
    g_object_unref (backend_info);
melunko@38
   170
melunko@111
   171
    g_return_val_if_fail (myth_handle->file_transfer != NULL, GNOME_VFS_ERROR_NOT_OPEN);
melunko@38
   172
melunko@38
   173
    myth_handle->content_size = myth_handle->file_transfer->filesize;
melunko@111
   174
    
rosfran@116
   175
    myth_handle->buffer = g_byte_array_sized_new (MYTHTV_BUFFER_SIZE);
melunko@111
   176
    myth_handle->buffer_remain = 0;
melunko@38
   177
melunko@38
   178
    *method_handle = (GnomeVFSMethodHandle *) myth_handle;
melunko@38
   179
melunko@38
   180
    return GNOME_VFS_OK;
melunko@38
   181
}
melunko@38
   182
melunko@38
   183
static GnomeVFSResult
melunko@38
   184
do_read (GnomeVFSMethod *method,
melunko@38
   185
         GnomeVFSMethodHandle *method_handle,
melunko@38
   186
         gpointer buffer,
melunko@38
   187
         GnomeVFSFileSize num_bytes,
melunko@38
   188
         GnomeVFSFileSize *bytes_read,
melunko@38
   189
         GnomeVFSContext *context)
melunko@38
   190
{
melunko@38
   191
    MythtvHandle *myth_handle = (MythtvHandle *) method_handle;
melunko@38
   192
    GnomeVFSFileSize bytes_to_read;
melunko@38
   193
melunko@38
   194
    *bytes_read = 0;
melunko@38
   195
melunko@38
   196
    if (myth_handle->bytes_read >= myth_handle->content_size)
melunko@38
   197
        return GNOME_VFS_ERROR_EOF;
melunko@38
   198
melunko@38
   199
    // fixme: change this to min math function
melunko@38
   200
    if (num_bytes > myth_handle->content_size - myth_handle->bytes_read)
melunko@38
   201
	    bytes_to_read = myth_handle->content_size - myth_handle->bytes_read;
melunko@38
   202
    else
melunko@38
   203
	    bytes_to_read = num_bytes;
melunko@38
   204
melunko@38
   205
    /* Loop sending the Myth File Transfer request:
melunko@38
   206
    * Retry whilst authentication fails and we supply it. */
rosfran@116
   207
  
renatofilho@149
   208
    if ( bytes_to_read > myth_handle->buffer_remain ) {
renatofilho@149
   209
        GByteArray *tmp_buffer = g_byte_array_new();
melunko@38
   210
renatofilho@149
   211
        gint len = gmyth_file_transfer_read (myth_handle->file_transfer,
renatofilho@149
   212
              tmp_buffer, MYTHTV_BUFFER_SIZE - myth_handle->buffer_remain, TRUE);
melunko@38
   213
melunko@251
   214
	if (len < 0) {
renatofilho@149
   215
            g_byte_array_free (tmp_buffer, TRUE);
renatofilho@164
   216
            g_warning ("Fail to read bytes");
melunko@251
   217
	    return GNOME_VFS_ERROR_IO;
melunko@251
   218
        } else if (len == 0) {
melunko@251
   219
	    g_byte_array_free (tmp_buffer, TRUE);
melunko@251
   220
	    g_warning ("End of file probably achieved");
melunko@251
   221
	    return GNOME_VFS_ERROR_EOF;
melunko@251
   222
	}
melunko@111
   223
renatofilho@149
   224
        myth_handle->buffer = g_byte_array_append (myth_handle->buffer,
renatofilho@149
   225
    		tmp_buffer->data, len);
rosfran@116
   226
rosfran@116
   227
		myth_handle->buffer_remain += len;
rosfran@116
   228
renatofilho@169
   229
        g_byte_array_free (tmp_buffer, TRUE);
rosfran@116
   230
    	tmp_buffer = NULL;    	
rosfran@116
   231
    }
melunko@38
   232
    
melunko@111
   233
    bytes_to_read = (bytes_to_read > myth_handle->buffer_remain) ? myth_handle->buffer_remain : bytes_to_read;
renatofilho@164
   234
    /* gets the first buffer_size bytes from the byte array buffer variable */ 
rosfran@116
   235
renatofilho@149
   236
    g_memmove (buffer, myth_handle->buffer->data, bytes_to_read);
rosfran@116
   237
melunko@111
   238
    myth_handle->bytes_read += bytes_to_read;
rosfran@116
   239
    myth_handle->buffer_remain -= bytes_to_read;    
rosfran@116
   240
rosfran@116
   241
  	/* flushs the newly buffer got from byte array */
renatofilho@149
   242
  	myth_handle->buffer = g_byte_array_remove_range (myth_handle->buffer, 0, bytes_to_read);
melunko@111
   243
    *bytes_read = bytes_to_read;
melunko@38
   244
  
melunko@38
   245
    return GNOME_VFS_OK;
melunko@38
   246
}
melunko@38
   247
melunko@38
   248
static GnomeVFSResult
melunko@38
   249
do_close (GnomeVFSMethod *method,
melunko@38
   250
          GnomeVFSMethodHandle *method_handle,
melunko@38
   251
          GnomeVFSContext *context)
melunko@38
   252
{
melunko@111
   253
melunko@38
   254
    MythtvHandle *myth_handle = (MythtvHandle *) method_handle;
melunko@38
   255
melunko@111
   256
    if (myth_handle->file_transfer) {
renatofilho@164
   257
	    gmyth_file_transfer_close (myth_handle->file_transfer);
melunko@38
   258
        g_object_unref (myth_handle->file_transfer);
renatofilho@164
   259
    	myth_handle->file_transfer = NULL;
melunko@111
   260
    }
rosfran@127
   261
    
rosfran@277
   262
    if (myth_handle->livetv) {
rosfran@277
   263
      g_object_unref (myth_handle->livetv);
rosfran@277
   264
    	myth_handle->livetv = NULL;
rosfran@277
   265
    }
rosfran@277
   266
    
rosfran@127
   267
    if (myth_handle->buffer) {
renatofilho@164
   268
		g_byte_array_free (myth_handle->buffer, TRUE);
renatofilho@164
   269
		myth_handle->buffer = NULL;
rosfran@127
   270
    }
melunko@38
   271
melunko@38
   272
    g_free (myth_handle);
melunko@38
   273
melunko@38
   274
    return GNOME_VFS_OK;
melunko@38
   275
}
melunko@38
   276
melunko@38
   277
static GnomeVFSResult
melunko@38
   278
do_get_file_info (GnomeVFSMethod *method,
melunko@38
   279
                  GnomeVFSURI *uri,
melunko@38
   280
                  GnomeVFSFileInfo *file_info,
melunko@38
   281
                  GnomeVFSFileInfoOptions options,
melunko@38
   282
                  GnomeVFSContext *context)
melunko@38
   283
{
renatofilho@188
   284
    GMythFileTransfer *file_transfer = NULL;
renatofilho@188
   285
    GMythBackendInfo *backend_info = NULL;
renatofilho@188
   286
renatofilho@188
   287
    file_info->name = g_strdup (gnome_vfs_uri_get_path (uri));
melunko@38
   288
    file_info->valid_fields = file_info->valid_fields
melunko@38
   289
        | GNOME_VFS_FILE_INFO_FIELDS_TYPE
melunko@38
   290
        | GNOME_VFS_FILE_INFO_FIELDS_MIME_TYPE
melunko@38
   291
        | GNOME_VFS_FILE_INFO_FIELDS_PERMISSIONS;
melunko@38
   292
    file_info->type = GNOME_VFS_FILE_TYPE_REGULAR;
melunko@111
   293
    // fixme: get from file extension?
melunko@111
   294
    file_info->mime_type = g_strdup ("video/x-nuv");
melunko@38
   295
    file_info->permissions =
melunko@38
   296
        GNOME_VFS_PERM_USER_READ |
melunko@38
   297
        GNOME_VFS_PERM_OTHER_READ |
melunko@38
   298
        GNOME_VFS_PERM_GROUP_READ;
melunko@38
   299
renatofilho@188
   300
    backend_info = gmyth_backend_info_new_full (gnome_vfs_uri_get_host_name (uri),
renatofilho@188
   301
            gnome_vfs_uri_get_user_name (uri),
renatofilho@188
   302
            gnome_vfs_uri_get_password (uri),
renatofilho@188
   303
            NULL,
renatofilho@188
   304
            gnome_vfs_uri_get_host_port (uri));
renatofilho@188
   305
renatofilho@188
   306
    file_transfer = gmyth_file_transfer_new (backend_info);
renatofilho@188
   307
    if (gmyth_file_transfer_open (file_transfer, gnome_vfs_uri_get_path (uri)) == TRUE) {
renatofilho@188
   308
        file_info->size = gmyth_file_transfer_get_filesize (file_transfer);
renatofilho@188
   309
        file_info->block_count = GNOME_VFS_FILE_INFO_FIELDS_BLOCK_COUNT;
renatofilho@188
   310
        file_info->io_block_size = GNOME_VFS_FILE_INFO_FIELDS_IO_BLOCK_SIZE;
renatofilho@188
   311
    }
renatofilho@188
   312
renatofilho@188
   313
    g_object_unref (file_transfer);
renatofilho@188
   314
    g_object_unref (backend_info);
melunko@38
   315
    return GNOME_VFS_OK;
melunko@38
   316
}
melunko@38
   317
melunko@38
   318
static gboolean
melunko@38
   319
do_is_local (GnomeVFSMethod *method,
melunko@38
   320
             const GnomeVFSURI *uri)
melunko@38
   321
{
melunko@38
   322
	return FALSE;
melunko@38
   323
}
melunko@38
   324
melunko@38
   325
static GnomeVFSMethod method = {
melunko@38
   326
	sizeof (GnomeVFSMethod),
melunko@38
   327
	do_open,
melunko@38
   328
	NULL,
melunko@38
   329
	do_close,
melunko@38
   330
	do_read,
melunko@38
   331
	NULL,
melunko@38
   332
	NULL,
melunko@38
   333
	NULL,
melunko@38
   334
	NULL,
melunko@38
   335
	NULL,
melunko@38
   336
	NULL,
melunko@38
   337
	NULL,
melunko@38
   338
	do_get_file_info,
melunko@38
   339
	NULL,
melunko@38
   340
	do_is_local,
melunko@38
   341
	NULL,
melunko@38
   342
	NULL,
melunko@38
   343
	NULL,
melunko@38
   344
	NULL,
melunko@38
   345
	NULL,
melunko@38
   346
	NULL,
melunko@38
   347
	NULL,
melunko@38
   348
	NULL,
melunko@38
   349
	NULL,
melunko@38
   350
	NULL,
melunko@38
   351
	NULL,
melunko@38
   352
	NULL,
melunko@38
   353
};
melunko@38
   354
melunko@38
   355
melunko@38
   356
GnomeVFSMethod *
melunko@38
   357
vfs_module_init (const char *method_name, const char *args)
melunko@38
   358
{
melunko@38
   359
	return &method;
melunko@38
   360
}
melunko@38
   361
melunko@38
   362
void
melunko@38
   363
vfs_module_shutdown (GnomeVFSMethod *method)
melunko@38
   364
{
melunko@38
   365
}