/[Frey]/trunk/lib/Frey/Server.pm
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /trunk/lib/Frey/Server.pm

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 179 by dpavlin, Sun Aug 31 17:44:03 2008 UTC revision 533 by dpavlin, Wed Nov 26 07:58:05 2008 UTC
# Line 1  Line 1 
1  package Frey::Server;  package Frey::Server;
2    
3  use Moose;  use Moose;
4    extends 'Frey';
5  with 'Frey::Web';  with 'Frey::Web';
6    with 'Frey::Config';
7    
8  use Continuity;  use Continuity;
9  #use Continuity::REPL;  #use Continuity::REPL;
10  use Data::Dump qw/dump/;  use Data::Dump qw/dump/;
11    
12  use Carp::REPL;  #use Carp::REPL; ## XXX it would be nice, but it breaks error reporting too much
13  use Frey::ClassLoader;  use Frey::ClassLoader;
14    use Frey::Run;
15    use Frey::Editor;
16    
17  my @messages;    # Global (shared) list of messages  my @messages;    # Global (shared) list of messages
18  my $got_message; # Flag to indicate that there is a new message to display  my $got_message; # Flag to indicate that there is a new message to display
# Line 18  use vars qw( $repl $server ); Line 21  use vars qw( $repl $server );
21    
22  #$repl = Continuity::REPL->new;  #$repl = Continuity::REPL->new;
23    
24    =head1 NAME
25    
26    Frey::Server - Continuity based server for Frey
27    
28    =head2 DESCRIPTION
29    
30    This is one of pissible server implementations for Frey. In it's current stage, it's also most complete one.
31    
32    =head2 run
33    
34      $o->run( $optional_port );
35    
36    =cut
37    
38  sub run {  sub run {
39          my ( $self, $port ) = @_;          my ( $self, $port ) = @_;
40          $server = Continuity->new(          $server = Continuity->new(
41                  port => $port || 16001,                  port => $port || $self->config->{port} || 16001,
42                  path_session => 1,                  path_session => 1,
43                  cookie_session => 'sid',                  cookie_session => 'sid',
44                  callback => \&main,                  callback => \&main,
45                  debug_level => 1,                  debug_level => 2,
46                  staticp => sub { $_[0]->url =~ m/\.(jpg|jpeg|gif|png|css|ico|js|html?|xml|json|ya?ml)(\?.*)?$/ },                  staticp => sub {
47                            $_[0]->url =~ m{^/+(static|var).*\.(jpg|jpeg|gif|png|css|ico|js|html?|xml|json|ya?ml)(\?.*)?$}
48                    },
49          );          );
50          $Module::Reload::Debug = 1;          $Module::Reload::Debug = 1; # auto if debug_level > 1
51          Frey::ClassLoader->new->load_all_classes();          Frey::ClassLoader->new->load_all_classes();
52          $server->loop;          $server->loop;
53  }  }
54    
55  # This is the main entrypoint. We are looking for one of three things -- a  =head2 main
56  # pushstream, a sent message, or a request for the main HTML. We delegate each  
57  # of these cases, none of which will return (they all loop forever).  This is simple dispatcher for our server. Currently it's in flux and
58    documented only in source code.
59    
60    =cut
61    
62  sub main {  sub main {
63          my ($req) = @_;          my ($req) = @_;
     
         my $path = $req->request->url->path;  
         warn "REQUEST: $path ",dump( $req->params ),"\n";  
   
         Module::Reload->check if $path =~ m!reload! || $req->param('reload');  
64    
65  #       warn $req->request->header('User_Agent');          my $path = $req->request->url->path;
66    
67          sub run_markup {          eval {
                 my ( $req, $class ) = @_;  
68    
69                  my %params = $req->params;                  sub refresh {
70                            my ( $url, $time ) = @_;
71                            $url  ||= '/';
72                            $time ||= 1;
73                            warn "# refresh $url";
74                            qq|
75                                    <html>
76                                    <head>
77                                            <META HTTP-EQUIV="Refresh" CONTENT="$time; URL=$url"></META>
78                                    </head>
79                                    <body>
80                                            Refresh <a href="$url"><tt>$url</tt></a> in $time sec
81                                    </body>
82                                    </html>
83                                    \n\r\n\r
84                            |; # XXX newlines at end are important to flush content to browser
85                    }
86    
87                  my @required =                  if ( $path =~ m{/reload(.*)} ) {
88                          grep {                          Frey::Server->new->load_config;
89                                  defined $_ && !defined( $params{$_} )                          Module::Reload->check;
90                          }                          warn "# reload done";
91                          map {                          $req->print( refresh( $1, 1 ) );
92                                  my $attr = $class->meta->get_attribute($_);                          return;
93                                  $attr->is_required && $_                  } elsif ( $path =~ m{/exit(.*)} ) {
94                          } $class->meta->get_attribute_list;                          # FIXME do we need some kind of check here for production? :-)
95                            # ./bin/dev.sh will restart us during development
96                            $req->print( refresh( $1, 2 ) );
97                            exit;
98                    }
99    
100                  warn "## required = ",dump( @required );  #               warn $req->request->header('User_Agent');
                 warn "## params = ",dump( %params );  
101    
102                    my %params = $req->params;
103                  my $html;                  my $html;
104    
105                  if ( @required ) {                  sub rest2class {
106                          $html = qq|<form method="post">|;                          my $class = shift;
107                          $html .= qq|<input type="text" name="$_">| foreach @required;                          $class =~ s/-/::/; # sigh!
108                          $html .= qq|<input type="submit" value="Run $class"></form>|;                          return $class;
                 } else {  
                         my $o = $class->new( %params );  
                         $o->depends if $o->can('depends');  
                         $html = $o->markup;  
109                  }                  }
110    
                 warn ">>> markup $class ",length( $html ), " bytes\n";  
                 return $html;  
         }  
111    
112                    my $f;
113    
114  #       eval {                  my $editor = Frey::Editor->new;
         {  
115    
116                  my $f;                  # shared run params
117                    my $run = {
118                            request_url => $req->request->url,
119                            debug => 1,
120                    };
121    
122                    if (
123                            $path =~ m{/Frey[:-]+ObjectBrowser}
124                    ) {
125                            $f = Frey::ObjectBrowser->new( fey_class => $params{class} );
126                            $f->request( $req );
127                    } elsif (
128                            $path =~ m{/Frey[:-]+ObjectDesigner}
129                    ) {
130                            $f = Frey::ObjectDesigner->new( fey_class => $params{class} );
131                            $f->request( $req );
132                    } elsif ( $path =~ $editor->url_regex ) {
133                            $req->print( $editor->command( $path ) );
134                            system( $editor->command( $path ) );
135                            return;
136                    } elsif (
137                            $path =~ m{/([^/]+)/(as_\w+)/?([^/]+)?}
138                    ) {
139                            my $class = rest2class $1;
140                            warn "# run $path -> $class $2";
141                            $run->{format} = $3 if $3;
142                            $params{request_url} = $req->request->url;
143                            $f = Frey::Run->new( class => $class, params => \%params, run => $2, %$run );
144                    } elsif (
145                            $path =~ m{/([^/]+)/?$}
146                    ) {
147                            my $class = rest2class $1;
148                            warn "# introspect $class";
149                            $f = Frey::Run->new( class => 'Frey::Introspect', params => { class => $class }, %$run );
150                    } else {
151                            $f = Frey::Run->new( class => 'Frey::ClassBrowser', %$run );
152                    }
153    
154                  if ( $path =~ m!/~/([^/]+)(.*)! ) {                  if ( $f ) {
155                          $f = Frey::Introspect->new( package => $1 );                          $f->clean_status;
156                  } elsif ( $path =~ m!/ob/([^/]+)(.*)! ) {                          $f->add_status( { request => $req } );
157                          $f = Frey::ObjectBrowser->new( fey_class => $1 );                          warn "## status ", dump( map { keys %$_ } $f->status );
158                  } elsif ( $path =~ m!/od/([^/]+)(.*)! ) {                          my $html = $f->html;
159                          $f = Frey::ObjectDesigner->new( fey_class => $1 );                          die "no html output" unless $html;
160                  } elsif ( $path =~ m!/markup/([^/]+)(.*)! ) {                          warn "## html ",length($html)," bytes";
161                          $req->print( run_markup( $req, $1 )  );                          $req->print( "$html\n" );
162                  } else {                  } else {
163                          $f = Frey::ClassBrowser->new;                          warn "# can't call request on nothing!";
164                  }                  }
                 $f->html( $req ) if $f;  
165    
166          };          };
167    
168          if ( $@ ) {          if ( $@ ) {
169                  warn $@;                  warn $@;
170                  #$req->conn->send_error( 404 ); # FIXME this should probably be 500, but we can't ship page with it                  $req->conn->send_error( 404 );  # FIXME this should probably be 500, but we can't ship page with it
171                  $req->print( qq{<pre class="error">$@<pre>} );                  $req->print( qq{<pre class="frey-error">$@<pre>} );
172                  Carp::REPL::repl;       # FIXME if $self->debug  #               Carp::REPL::repl;
   
173          }          }
174    
175          # If this is a request for the pushtream, then give them that          # If this is a request for the pushtream, then give them that
# Line 113  sub main { Line 177  sub main {
177                  pushstream($req);                  pushstream($req);
178          }          }
179    
         if ( $path =~ m/die/ ) {  
                 Carp::REPL::repl;       # FIXME if $self->debug  
         }  
   
180          # If they are sending us a message, we give them a thread for that too          # If they are sending us a message, we give them a thread for that too
181          if($path =~ /sendmessage/) {          if($path =~ /sendmessage/) {
182                  send_message($req);                  send_message($req);
183          }          }
184    
185            if ($req->conn ) {
186                    $req->conn->close;
187                    warn "## close connection: $@";
188            }
189  }  }
190    
191  # Here we accept a connection to the browser, and keep it open. Meanwhile we  # Here we accept a connection to the browser, and keep it open. Meanwhile we

Legend:
Removed from v.179  
changed lines
  Added in v.533

  ViewVC Help
Powered by ViewVC 1.1.26