/[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 48 by dpavlin, Wed Jul 2 12:36:59 2008 UTC revision 581 by dpavlin, Fri Nov 28 13:16:47 2008 UTC
# Line 1  Line 1 
1  package Frey::Server;  package Frey::Server;
2    
3  use strict;  use Moose;
4  use warnings;  extends 'Frey';
5    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 base 'Frey';  #use Carp::REPL; ## XXX it would be nice, but it breaks error reporting too much
13  use Frey::HTML;  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 16  my $got_message; # Flag to indicate that Line 20  my $got_message; # Flag to indicate that
20  use vars qw( $repl $server );  use vars qw( $repl $server );
21    
22  #$repl = Continuity::REPL->new;  #$repl = Continuity::REPL->new;
23  $server = Continuity->new(  
24          port => 16001,  =head1 NAME
25          path_session => 1,  
26          cookie_session => 'sid',  Frey::Server - Continuity based server for Frey
27          callback => \&main,  
28          debug_level => 1,  =head2 DESCRIPTION
29          staticp => sub { $_[0]->url =~ m/\.(jpg|jpeg|gif|png|css|ico|js|html?)$/ },  
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 ) = @_;
40            $server = Continuity->new(
41                    port => $port || $self->config->{port} || 16001,
42                    path_session => 1,
43                    cookie_session => 'sid',
44                    callback => \&main,
45                    debug_level => 2,
46                    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; # auto if debug_level > 1
51            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) = @_;
64      
65    #       $req->send_headers("X-Frey-VERSION: $Frey::VERSION");
66    
67          my $path = $req->request->url->path;          my $path = $req->request->url->path;
         warn "REQUEST: $path\n";  
68    
69          warn $req->request->header('User_Agent');          eval {
70  #warn dump( $req );  
71                    if ( $path =~ m{/reload(.*)} ) {
72    
73                            $ENV{FREY_NO_LOG} = 1;
74                            my $cmd = "perl -c $0";
75                            warn "# check config with $cmd";
76                            if ( system($cmd) == 0 ) {
77                                    $req->print( "\r\n" );
78                                    my $server = Frey::Server->new;
79                                    $server->load_config;
80                                    $req->print( "\r\n" );
81                                    Module::Reload->check;
82                                    $req->print( "\r\n" );
83                                    $req->print( refresh( $1, 1 ) );
84                                    $req->print( "\r\n" );
85                                    warn "# reload done";
86                                    return;
87                            } else {
88                                    warn "ERROR: $?";
89                            }
90                            $ENV{FREY_NO_LOG} = 0;
91            
92                    } elsif ( $path =~ m{/exit(.*)} ) {
93                            # FIXME do we need some kind of check here for production? :-)
94                            # ./bin/dev.sh will restart us during development
95                            $req->print( refresh( $1, 2 ) );
96                            $req->print( "\r\n" );
97                            exit;
98                    }
99    
100    #               warn $req->request->header('User_Agent');
101    
102                    my %params = $req->params;
103                    my $html;
104    
105                    sub rest2class {
106                            my $class = shift;
107                            $class =~ s/-/::/; # sigh!
108                            return $class;
109                    }
110    
111    
112                    my $f;
113    
114                    my $editor = Frey::Editor->new;
115    
116                    # 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{/([^/]+)/(\w*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                            $req->print( "\r\n\r\n" ); # send something to browser so we don't time-out
144                            $f = Frey::Run->new( class => $class, params => \%params, run => $2, %$run );
145                    } elsif (
146                            $path =~ m{/([^/]+)/?$}
147                    ) {
148                            my $class = rest2class $1;
149                            warn "# introspect $class";
150                            $f = Frey::Run->new( class => 'Frey::Introspect', params => { class => $class }, %$run );
151                    } else {
152                            $f = Frey::Run->new( class => 'Frey::ClassBrowser', %$run );
153                    }
154    
155                    if ( $f ) {
156                            $f->clean_status;
157                            $f->add_status( { request => $req } );
158                            $f->status_parts;
159                            if ( my $html = $f->html ) {
160                                    warn "## html ",length($html)," bytes";
161                                    $req->print( $html );
162                            } else {
163                                    $req->print( qq|<pre class="frey-error">no output from $f</pre>\r\n\r\n| );
164                            }
165                    } else {
166                            warn "# can't call request on nothing!";
167                    }
168    
169            };
170    
171            if ( $@ ) {
172                    warn "SERVER ERROR: $@";
173    #               $req->conn->send_error( 404 );  # FIXME this should probably be 500, but we can't ship page with it
174                    $req->print( qq{<pre class="frey-error">$@<pre>\r\n\r\n} );
175    #               Carp::REPL::repl;
176            }
177    
178          # If this is a request for the pushtream, then give them that          # If this is a request for the pushtream, then give them that
179          if($path =~ /pushstream/) {          if($path =~ /pushstream/) {
180                  pushstream($req);                  pushstream($req);
181          }          }
182      
183          # 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
184          if($path =~ /sendmessage/) {          if($path =~ /sendmessage/) {
185                  send_message($req);                  send_message($req);
186          }          }
187    
188          # Otherwise, lets give them page          if ( $req->conn ) {
189          send_page($req);                  $req->conn->close;
190                    warn "## close connection: $@";
191            }
192  }  }
193    
194  # 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
# Line 92  sub send_message { Line 228  sub send_message {
228    }    }
229  }  }
230    
231  # This isn't a pushstream, nor a new message. It is just the main page. We loop  sub refresh {
232  # in case they ask for it multiple times :)          my ( $url, $time ) = @_;
233  sub send_page {          $url  ||= '/';
234          my ($req) = @_;          $time ||= 1;
235          my $templates = Template::Declare->templates;          warn "# refresh $url";
236          while(1) {          qq|
237                  warn "param = ",dump($req->param);                  <html>
238                  my $path = $req->request->url->path;                  <head>
239                            <META HTTP-EQUIV="Refresh" CONTENT="$time; URL=$url"></META>
240                  my $html;                  </head>
241                    <body>
242                  if ( $path =~ m/::/ ) {                          Refresh <a href="$url"><tt>$url</tt></a> in $time sec
243                          my ( undef, $module, $method ) = split(m!/!, $path, 3);                  </body>
244                    </html>
245                          if ( ! defined( $templates->{$module} ) ) {                  \n\r\n\r
246                                  $req->conn->send_status_line( 404, "$module" );          |; # XXX newlines at end are important to flush content to browser
                                 $html = "Package $module not found";  
                         } elsif ( ! $method ) {  
                                 $html = Frey::HTML->page( 'package-methods', $req, $module );  
                         } elsif ( grep(/^\Q$method\E$/, @{ $templates->{$module} }) ) {  
                                 $html = Frey::HTML->page( $method, $req );  
                         } else {  
                                 $req->conn->send_status_line( 404, "$module $method" );  
                                 $html = "Package $module doesn't have $method";  
                         }  
                 } else {  
                         warn "fallback to status page\n";  
                         $html = Frey::HTML->page( 'status' );  
                 }  
   
                 $req->print( $html );  
                 warn ">> ",length( $html ), " bytes\n";  
                 $req->next;  
         }  
247  }  }
248    
249  1;  1;

Legend:
Removed from v.48  
changed lines
  Added in v.581

  ViewVC Help
Powered by ViewVC 1.1.26