/*******************************************************************************/ /* Copyright (C) 2008 Jonathan Moore Liles */ /* */ /* This program is free software; you can redistribute it and/or modify it */ /* under the terms of the GNU General Public License as published by the */ /* Free Software Foundation; either version 2 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 General Public License for */ /* more details. */ /* */ /* You should have received a copy of the GNU General Public License along */ /* with This program; see the file COPYING. If not,write to the Free Software */ /* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /*******************************************************************************/ /* This class handles all journaling. All journaled objects must inherit from Loggable as well as define a few special methods (via macros), get and set methods, and have contructors and destructors that call log_create() and log_destroy() in the appropriate order. Any action that might affect multiple loggable objects *must* be braced by calls to Loggable::block_start() and Loggable::block_end() in order for Undo to work properly. */ #include "Loggable.H" #include #include #include #include "util/file.h" #include using std::min; using std::max; FILE *Loggable::_fp; int Loggable::_log_id = 0; int Loggable::_level = 0; off_t Loggable::_undo_offset = 0; size_t Loggable::_loggables_size = 0; Loggable ** Loggable::_loggables; std::map Loggable::_class_map; std::queue Loggable::_transaction; progress_func *Loggable::_progress_callback = NULL; void *Loggable::_progress_callback_arg = NULL; /** ensure that _loggables array is big enough for /n/ elements */ void Loggable::ensure_size ( size_t n ) { if ( n > _loggables_size ) { size_t p = 0; while ( ( (unsigned)1 << p ) < n ) ++p; size_t os = _loggables_size; _loggables_size = 1 << p ; _loggables = (Loggable**) realloc( _loggables, sizeof( Loggable ** ) * _loggables_size ); memset( _loggables + os, 0, _loggables_size - os ); } } /** Open the journal /filename/ and replay it, bringing the end state back into RAM */ bool Loggable::open ( const char *filename ) { FILE *fp; Loggable::_fp = NULL; if ( ! ( fp = fopen( filename, "a+" ) ) ) { WARNING( "Could not open log file for writing!" ); return false; } if ( newer( "snapshot", filename ) ) { DMESSAGE( "Loading snapshot" ); FILE *fp = fopen( "snapshot", "r" ); replay( fp ); fclose( fp ); } else { DMESSAGE( "Replaying journal" ); replay( fp ); } fseek( fp, 0, SEEK_END ); _undo_offset = ftell( fp ); Loggable::_fp = fp; return true; } #include #include /** replay journal or snapshot */ bool Loggable::replay ( FILE *fp ) { /* FIXME: bogus */ char buf[BUFSIZ]; struct stat st; fstat( fileno( fp ), &st ); off_t total = st.st_size; off_t current = 0; if ( _progress_callback ) _progress_callback( 0, _progress_callback_arg ); while ( fscanf( fp, "%[^\n]\n", buf ) == 1 ) { if ( ! ( ! strcmp( buf, "{" ) || ! strcmp( buf, "}" ) ) ) { if ( *buf == '\t' ) do_this( buf + 1, false ); else do_this( buf, false ); } current = ftell( fp ); if ( _progress_callback ) _progress_callback( current * 100 / total, _progress_callback_arg ); } if ( _progress_callback ) _progress_callback( 0, _progress_callback_arg ); return true; } /** close journal and delete all loggable objects, returing the systemt to a blank slate */ bool Loggable::close ( void ) { DMESSAGE( "closing journal and destroying all journaled objects" ); if ( ! _fp ) return true; fclose( _fp ); _fp = NULL; snapshot( "snapshot" ); for ( int i = 0; i < _log_id - 1; ++i ) { Loggable ** l = &_loggables[ i ]; if ( *l ) { delete *l; *l = NULL; } } _log_id = 0; return true; } /** must be called after construction in create() methods */ void Loggable::update_id ( int id ) { /* make sure we're the last one */ assert( _id == _log_id ); assert( _loggables[ _id - 1 ] == this ); _loggables[ _id - 1 ] = NULL; _log_id = max( _log_id, id ); /* return this id number to the system */ // --_log_id; _id = id; /* make sure it'll fit */ ensure_size( _id ); ASSERT( ! _loggables[ _id - 1 ], "Attempt to create object with an ID (0x%X) that already exists. The existing object is of type \"%s\", the new one is \"%s\". Corrupt journal?", _id, _loggables[ _id - 1 ]->class_name(), class_name() ); _loggables[ _id - 1 ] = this; } /** return a pointer to a static copy of /s/ with all special characters escaped */ const char * Loggable::escape ( const char *s ) { static char r[512]; size_t i = 0; for ( ; *s && i < sizeof( r ); ++i, ++s ) { if ( '\n' == *s ) { r[ i++ ] = '\\'; r[ i ] = 'n'; } else if ( '"' == *s ) { r[ i++ ] = '\\'; r[ i ] = '"'; } else r[ i ] = *s; } r[ i ] = '\0'; return r; } /** 'do' a message like "Audio_Region 0xF1 set :r 123" */ bool Loggable::do_this ( const char *s, bool reverse ) { int id; if ( ! ( sscanf( s, "%*s %X ", &id ) > 0 ) ) return false; Loggable *l = find( id ); // assert( l ); char classname[40]; char command[40]; char *arguments = NULL; const char *create, *destroy; if ( reverse ) { // sscanf( s, "%s %*X %s %*[^\n<]<< %a[^\n]", classname, command, &arguments ); sscanf( s, "%s %*X %s%*[^\n<]<< %a[^\n]", classname, command, &arguments ); create = "destroy"; destroy = "create"; DMESSAGE( "undoing \"%s\"", s ); } else { sscanf( s, "%s %*X %s %a[^\n<]", classname, command, &arguments ); create = "create"; destroy = "destroy"; } if ( ! strcmp( command, destroy ) ) { /* deleting eg. a track, which contains a list of other widgets, causes destroy messages to be emitted for all those widgets, but when replaying the journal the destroy message causes the children to be deleted also... This is a temporary hack. Would it be better to queue up objects for deletion (when?) */ if ( l ) delete l; } else if ( ! strcmp( command, "set" ) ) { // printf( "got set command (%s).\n", arguments ); Log_Entry e( arguments ); l->log_start(); l->set( e ); l->log_end(); } else if ( ! strcmp( command, create ) ) { Log_Entry e( arguments ); ASSERT( _class_map[ std::string( classname ) ], "Journal contains an object of class \"%s\", but I don't know how to create such objects.", classname ); { /* create */ Loggable *l = _class_map[ std::string( classname ) ]( e ); l->update_id( id ); l->log_create(); } } if ( arguments ) free( arguments ); return true; } /** Reverse the last journal transaction */ void Loggable::undo ( void ) { const int bufsiz = 1024; char buf[bufsiz]; block_start(); long here = ftell( _fp ); fseek( _fp, _undo_offset, SEEK_SET ); backwards_fgets( buf, bufsiz, _fp ); if ( ! strcmp( buf, "}\n" ) ) { DMESSAGE( "undoing block" ); for ( ;; ) { backwards_fgets( buf, bufsiz, _fp ); char *s = buf; if ( *s != '\t' ) break; else ++s; do_this( s, true ); } } else do_this( buf, true ); off_t uo = ftell( _fp ); ASSERT( _undo_offset <= here, "WTF?" ); block_end(); _undo_offset = uo; } /** Make all loggable ids consecutive. This invalidates any existing * journal or snapshot, so you *must* write out a new one after * performing this operation*/ void Loggable::compact_ids ( void ) { int id = 0; for ( int i = 0; i < _log_id; ++i ) if ( _loggables[ i ] ) { ++id; if ( _loggables[ id - 1 ] ) continue; _loggables[ id - 1 ] = _loggables[ i ]; _loggables[ i ] = NULL; _loggables[ id - 1 ]->_id = id; } _log_id = id; } /** write a snapshot of the current state of all loggable objects to * file handle /fp/ */ bool Loggable::snapshot ( FILE *fp ) { FILE *ofp = _fp; if ( ! ( _fp = fp ) ) { _fp = ofp; return false; } block_start(); for ( int i = 0; i < _log_id; ++i ) { const Loggable * l = _loggables[ i ]; if ( l && _class_map[ std::string( l->class_name() ) ] ) l->log_create(); } block_end(); _fp = ofp; return true; } /** write a snapshot of the current state of all loggable objects to * file /name/ */ bool Loggable::snapshot ( const char *name ) { FILE *fp; if ( ! ( fp = fopen( name, "w" ) )) return false; snapshot( fp ); fclose( fp ); return true; } /** Replace the journal with a snapshot of the current state */ void Loggable::compact ( void ) { fseek( _fp, 0, SEEK_SET ); ftruncate( fileno( _fp ), 0 ); compact_ids(); if ( ! snapshot( _fp ) ) FATAL( "Could not write snapshot!" ); fseek( _fp, 0, SEEK_END ); } #include /** Buffered sprintf wrapper */ void Loggable::log ( const char *fmt, ... ) { if ( ! _fp ) return; /* FIXME: bogus limit */ static char buf[1024]; static int i = 0; va_list args; if ( fmt ) { va_start( args, fmt ); i += vsprintf( buf + i, fmt, args ); va_end( args ); } if ( rindex( buf, '\n' ) ) { _transaction.push( strdup( buf ) ); i = 0; } } /** End the current transaction and commit it to the journal */ void Loggable::flush ( void ) { if ( ! _fp ) { // printf( "error: no log file open!\n" ); while ( ! _transaction.empty() ) { free( _transaction.front() ); _transaction.pop(); } return; } int n = _transaction.size(); if ( n > 1 ) fprintf( _fp, "{\n" ); while ( ! _transaction.empty() ) { char *s = _transaction.front(); _transaction.pop(); if ( n > 1 ) fprintf( _fp, "\t" ); fprintf( _fp, "%s", s ); free( s ); } if ( n > 1 ) fprintf( _fp, "}\n" ); if ( n ) /* something done, reset undo index */ _undo_offset = ftell( _fp ); fflush( _fp ); } /** Print bidirectional journal entry */ void Loggable::log_print( const Log_Entry *o, const Log_Entry *n ) const { if ( ! _fp ) return; if ( n ) for ( int i = 0; i < n->size(); ++i ) { const char *s, *v; n->get( i, &s, &v ); log( "%s %s%s", s, v, n->size() == i + 1 ? "" : " " ); } if ( o && o->size() ) { if ( n ) log( " << " ); for ( int i = 0; i < o->size(); ++i ) { const char *s, *v; o->get( i, &s, &v ); log( "%s %s%s", s, v, o->size() == i + 1 ? "" : " " ); } } log( "\n" ); } /** Remember current object state for later comparison. *Must* be * called before any user action that might change one of the object's * journaled properties. */ void Loggable::log_start ( void ) { if ( ! _old_state ) { _old_state = new Log_Entry; get( *_old_state ); } ++_nest; } /** Log any change to the object's state since log_start(). */ void Loggable::log_end ( void ) { if ( --_nest > 0 ) return; Log_Entry *new_state; new_state = new Log_Entry; get( *new_state ); if ( Log_Entry::diff( _old_state, new_state ) ) { log( "%s 0x%X set ", class_name(), _id ); log_print( _old_state, new_state ); } if ( new_state ) delete new_state; if ( _old_state ) delete _old_state; _old_state = NULL; if ( Loggable::_level == 0 ) Loggable::flush(); } /** Log object creation. *Must* be called at the end of all public * constructors for leaf classes */ void Loggable::log_create ( void ) const { if ( ! _fp ) /* replaying, don't bother */ return; log( "%s 0x%X create ", class_name(), _id ); Log_Entry e; get( e ); if ( e.size() ) log_print( NULL, &e ); else log( "\n" ); if ( Loggable::_level == 0 ) Loggable::flush(); } /** Log object destruction. *Must* be called at the beginning of the * destructors of leaf classes */ void Loggable::log_destroy ( void ) const { if ( ! _fp ) /* tearing down... don't bother */ return; log( "%s 0x%X destroy << ", class_name(), _id ); Log_Entry e; get( e ); log_print( NULL, &e ); if ( Loggable::_level == 0 ) Loggable::flush(); }