gitprep / lib / Gitprep.pm /
a300cc8 10 years ago
1 contributor
344 lines | 9.388kb
use 5.008007;
package Gitprep;
use Mojo::Base 'Mojolicious';

use Carp 'croak';
use DBIx::Custom;
use Encode qw/encode decode/;
use Gitprep::API;
use Gitprep::Git;
use Gitprep::Manager;
use Gitprep::SmartHTTP;
use Scalar::Util 'weaken';
use Validator::Custom;

# Digest::SHA loading to Mojo::Util if not loaded
{
  package Mojo::Util;
  eval {require Digest::SHA; import Digest::SHA qw(sha1 sha1_hex)};
}

our $VERSION = '1.02';

has 'dbi';
has 'git';
has 'manager';
has 'validator';

use constant BUFFER_SIZE => 8192;

sub startup {
  my $self = shift;
  
  # Config file
  $self->plugin('INIConfig', {ext => 'conf'});
  
  # Config file for developper
  unless ($ENV{GITPREP_NO_MYCONFIG}) {
    my $my_conf_file = $self->home->rel_file('gitprep.my.conf');
    $self->plugin('INIConfig', {file => $my_conf_file}) if -f $my_conf_file;
  }
  
  # Listen
  my $conf = $self->config;
  my $listen = $conf->{hypnotoad}{listen} ||= ['http://*:10020'];
  $listen = [split /,/, $listen] unless ref $listen eq 'ARRAY';
  $conf->{hypnotoad}{listen} = $listen;
  
  # Git
  my $git = Gitprep::Git->new;
  my $git_bin
    = $conf->{basic}{git_bin} ? $conf->{basic}{git_bin} : $git->search_bin;
  if (!$git_bin || ! -e $git_bin) {
    $git_bin ||= '';
    my $error = "Can't detect or found git command ($git_bin)."
      . " set git_bin in gitprep.conf";
    $self->log->error($error);
    croak $error;
  }
  $git->bin($git_bin);
  
  # Repository Manager
  my $manager = Gitprep::Manager->new(app => $self);
  weaken $manager->{app};
  $self->manager($manager);
  
  # Repository home
  my $rep_home = $ENV{GITPREP_REP_HOME} || $self->home->rel_file('data/rep');
  $git->rep_home($rep_home);
  unless (-d $rep_home) {
    mkdir $rep_home
      or croak "Can't create directory $rep_home: $!";
  }
  $self->git($git);
  
  # Added public path
  # push @{$self->static->paths}, $rep_home;
  
  # DBI
  my $db_file = $ENV{GITPREP_DB_FILE}
    || $self->home->rel_file('data/gitprep.db');
  my $dbi = DBIx::Custom->connect(
    dsn => "dbi:SQLite:database=$db_file",
    connector => 1,
    option => {sqlite_unicode => 1, sqlite_use_immediate_transaction => 1}
  );
  $self->dbi($dbi);
  
  # Database file permision
  if (my $user = $self->config->{hypnotoad}{user}) {
    my $uid = (getpwnam $user)[2];
    chown $uid, -1, $db_file;
  }
  if (my $group = $self->config->{hypnotoad}{group}) {
    my $gid = (getgrnam $group)[2];
    chown -1, $gid, $db_file;
  }
  
  # Setup database
  $self->manager->setup_database;
  
  # Model
  my $models = [
    {table => 'user', primary_key => 'id'},
    {table => 'project', primary_key => ['user_id', 'name']},
    {table => 'number', primary_key => 'key'}
  ];
  $dbi->create_model($_) for @$models;

  # Validator
  my $validator = Validator::Custom->new;
  $self->validator($validator);
  $validator->register_constraint(
    user_name => sub {
      my $value = shift;
      
      return ($value || '') =~ /^[a-zA-Z0-9_\-]+$/
    },
    project_name => sub {
      my $value = shift;
      
      return ($value || '') =~ /^[a-zA-Z0-9_\-]+$/
    }
  );
  

  # Routes
  sub template {
    my $template = shift;
    
    return sub { shift->render($template, , 'mojo.maybe' => 1) };
  }
  
  {
    my $r = $self->routes;

    # DBViewer(only development)
    if ($self->mode eq 'development') {
      eval {
        $self->plugin(
          'DBViewer',
          dsn => "dbi:SQLite:database=$db_file"
        );
      };
    }
    
    # Auto route
    {
      my $r = $r->under(sub {
        my $self = shift;
        
        my $api = $self->gitprep_api;
        
        # Authentication
        {
          my $path = $self->req->url->path->parts->[0] || '';
          
          # Admin
          if ($path eq '_admin' && !$api->logined_admin) {
            $self->redirect_to('/');
            return;
          }
        }
        
        return 1; 
      });
      
      # Auto routes
      $self->plugin('AutoRoute', route => $r);

      # Custom routes
      {
        my $id_re = qr/[a-zA-Z0-9_-]+/;
        
        # User
        my $r = $r->route('/:user', user => $id_re);
        {
          # Home
          $r->get('/' => template '/user');
          
          # Settings
          $r->get('/_settings' => template '/user-settings');
        }

        # Smart HTTP
        {
          my $r = $r->route('/(:project).git', project => $id_re);
          
          my $sh = Gitprep::SmartHTTP->new;
          
          # Fetch
          $r->get('/info/refs')->to(cb => sub {
            my $self = shift;
            
            my $service = $self->param('service') || '';
            
            my $user = $self->param('user');
            my $project = $self->param('project');
            
            if ($service eq 'git-upload-pack') {
              
              my $rep = $git->rep($user, $project);
              my @cmd = $git->cmd($user, $project, 'upload-pack', '--stateless-rpc', '--advertise-refs', $rep);
              
              warn "@cmd";
              
              use IPC::Open3 'open3';
              use Symbol 'gensym';
              my ($cout, $cerr) = (gensym, gensym );
              my $pid = open3(my $cin, $cout, $cerr, @cmd );
              close $cin;
              my ( $refs, $err, $buf ) = ( '', '', '' );
              my $s = IO::Select->new( $cout, $cerr );
              while (my @ready = $s->can_read) {
                for my $handle (@ready) {
                  while ( sysread( $handle, $buf, BUFFER_SIZE ) ) {
                    if ( $handle == $cerr ) {
                      $err .= $buf;
                    }
                    else {
                      $refs .= $buf;
                    }
                  }
                  $s->remove($handle) if eof($handle);
                }
              }
              close $cout;
              close $cerr;
              waitpid($pid, 0);

              if ($err) {
                app->log->error($err);
                $self->render_exception($err);
              }
              
              $self->res->headers->content_type('application/x-git-upload-pack-advertisement');
              
              my $data =
                $sh->pkt_write("# service=git-upload-pack\n") . $sh->pkt_flush() . $refs;
              
              $self->render(data => $data);
            }
            else {
              $sh->dumb_info_refs;
            }
          });

          # $r->post('/git-upload-pack');
          # $r->post('/git-receive-pack');
          # $r->get('/HEAD');
          # $r->get('/objects/info/alternates');
          # $r->get('/objects/info/http-alternates');
          # $r->get('/objects/info/packs');
          # $r->get('/objects/[0-9a-f]{2}/[0-9a-f]{38}');
          # $r->get('/objects/pack/pack-[0-9a-f]{40}\.pack');
          # $r->get('/objects/pack/pack-[0-9a-f]{40}\.idx');
        }
                
        # Project
        {
          my $r = $r->route('/:project', project => $id_re);
          
          # Home
          $r->get('/' => template '/project');
          
          # Commit
          $r->get('/commit/*diff' => template '/commit');
          
          # Commits
          $r->get('/commits/*rev_file' => template '/commits');
          
          # Branches
          $r->any('/branches/*base_branch' => {base_branch => undef} => template '/branches');

          # Tags
          $r->get('/tags' => template '/tags');

          # Tree
          $r->get('/tree/*rev_dir' => template '/tree');
          
          # Blob
          $r->get('/blob/*rev_file' => template '/blob');
          
          # Sub module
          $r->get('/submodule/*rev_file' => template '/submodule');

          # Raw
          $r->get('/raw/*rev_file' => template '/raw');

          # Blame
          $r->get('/blame/*rev_file' => template '/blame');
          
          # Archive
          $r->get('/archive/(*rev).tar.gz' => template '/archive')->to(archive_type => 'tar');
          $r->get('/archive/(*rev).zip' => template '/archive')->to(archive_type => 'zip' );
          
          # Compare
          $r->get('/compare/(*rev1)...(*rev2)' => template '/compare');
          
          # Settings
          $r->any('/settings' => template '/settings');
          
          # Fork
          $r->any('/fork' => template '/fork');

          # Network
          $r->get('/network' => template '/network');

          # Network Graph
          $r->get('/network/graph/(*rev1)...(*rev2_abs)' => template '/network/graph');

          # Import branch
          $r->any('/import-branch/:remote_user/:remote_project' => template '/import-branch');
          
          # Get branches and tags
          $r->get('/api/revs' => template '/api/revs');
        }
      }
    }
  }

  # Helper
  {
    # API
    $self->helper(gitprep_api => sub { Gitprep::API->new(shift) });
  }
  
  # Reverse proxy support
  my $reverse_proxy_on = $self->config->{reverse_proxy}{on};
  my $path_depth = $self->config->{reverse_proxy}{path_depth};
  if ($reverse_proxy_on) {
    $ENV{MOJO_REVERSE_PROXY} = 1;
    if ($path_depth) {
      $self->hook('before_dispatch' => sub {
        my $self = shift;
        for (1 .. $path_depth) {
          my $prefix = shift @{$self->req->url->path->parts};
          push @{$self->req->url->base->path->parts}, $prefix;
        }
      });
    }
  }
}

1;