<?php # Copyright (c) 2014 Johannes Goehr # Copyright (c) 2014 Bob Clough # Licensed under the MIT license if ( false === include_once( config_get( 'plugin_path' ) . 'Source/MantisSourcePlugin.class.php' ) ) { return; } require_once( config_get( 'core_path' ) . 'url_api.php' ); require_once( config_get( 'core_path' ) . 'json_api.php' ); class SourceGitlabPlugin extends MantisSourcePlugin { const ERROR_INVALID_PRIMARY_BRANCH = 'invalid_branch'; public function register() { $this->name = plugin_lang_get( 'title' ); $this->description = plugin_lang_get( 'description' ); $this->version = '1.0.1'; $this->requires = array( 'MantisCore' => '1.2.0', 'Source' => '0.16', ); $this->author = 'Johannes Goehr'; $this->contact = 'johannes.goehr@mobilexag.de'; $this->url = 'http://www.mobilexag.de'; } public function errors() { $t_errors_list = array( self::ERROR_INVALID_PRIMARY_BRANCH, ); foreach( $t_errors_list as $t_error ) { $t_errors[$t_error] = plugin_lang_get( 'error_' . $t_error ); } return $t_errors; } public $type = 'gitlab'; public function show_type() { return plugin_lang_get( 'gitlab' ); } public function show_changeset( $p_repo, $p_changeset ) { $t_ref = substr( $p_changeset->revision, 0, 8 ); $t_branch = $p_changeset->branch; return "$t_branch $t_ref"; } public function show_file( $p_repo, $p_changeset, $p_file ) { return "$p_file->action - $p_file->filename"; } public function url_repo( $p_repo, $p_changeset=null ) { $t_root = $p_repo->info['hub_root']; $t_reponame = $p_repo->info['hub_reponame']; $t_ref = ""; if ( !is_null( $p_changeset ) ) { $t_ref = "/$p_changeset->revision"; } if ( !is_null( $t_ref ) ) { return "$t_root/$t_reponame/"; } return "$t_root/$t_reponame/tree/$t_ref"; } public function url_changeset( $p_repo, $p_changeset ) { $t_root = $p_repo->info['hub_root']; $t_reponame = $p_repo->info['hub_reponame']; $t_ref = $p_changeset->revision; return "$t_root/$t_reponame/commit/$t_ref"; } public function url_file( $p_repo, $p_changeset, $p_file ) { $t_root = $p_repo->info['hub_root']; $t_reponame = $p_repo->info['hub_reponame']; $t_ref = $p_changeset->revision; $t_filename = $p_file->filename; return "$t_root/$t_reponame/blob/$t_ref/$t_filename"; } public function url_diff( $p_repo, $p_changeset, $p_file ) { $t_root = $p_repo->info['hub_root']; $t_reponame = $p_repo->info['hub_reponame']; $t_ref = $p_changeset->revision; $t_filename = $p_file->filename; return "$t_root/$t_reponame/commit/$t_ref?view=parallel"; } public function update_repo_form( $p_repo ) { $t_hub_root = null; $t_hub_repoid = null; $t_hub_app_secret = null; if ( isset( $p_repo->info['hub_root'] ) ) { $t_hub_root = $p_repo->info['hub_root']; } if ( isset( $p_repo->info['hub_repoid'] ) ) { $t_hub_repoid = $p_repo->info['hub_repoid']; } if ( isset( $p_repo->info['hub_reponame'] ) ) { $t_hub_reponame = $p_repo->info['hub_reponame']; } if ( isset( $p_repo->info['hub_app_secret'] ) ) { $t_hub_app_secret = $p_repo->info['hub_app_secret']; } if ( isset( $p_repo->info['master_branch'] ) ) { $t_master_branch = $p_repo->info['master_branch']; } else { $t_master_branch = 'master'; } ?> <tr <?php echo helper_alternate_class() ?>> <td class="category"><?php echo plugin_lang_get( 'hub_root' ) ?></td> <td><input name="hub_root" maxlength="250" size="40" value="<?php echo string_attribute( $t_hub_root ) ?>"/></td> </tr> <tr <?php echo helper_alternate_class() ?>> <td class="category"><?php echo plugin_lang_get( 'hub_repoid' ) ?></td> <td><input name="hub_repoid" maxlength="250" size="40" value="<?php echo string_attribute( $t_hub_repoid ) ?>"/></td> </tr> <tr <?php echo helper_alternate_class() ?>> <td class="category"><?php echo plugin_lang_get( 'hub_reponame' ) ?></td> <td><input name="hub_reponame" maxlength="250" size="40" value="<?php echo string_attribute( $t_hub_reponame ) ?>"/></td> </tr> <tr <?php echo helper_alternate_class() ?>> <td class="category"><?php echo plugin_lang_get( 'hub_app_secret' ) ?></td> <td><input name="hub_app_secret" maxlength="250" size="40" value="<?php echo string_attribute( $t_hub_app_secret ) ?>"/></td> </tr> <tr <?php echo helper_alternate_class() ?>> <td class="category"><?php echo plugin_lang_get( 'master_branch' ) ?></td> <td><input name="master_branch" maxlength="250" size="40" value="<?php echo string_attribute( $t_master_branch ) ?>"/></td> </tr> <?php } public function update_repo( $p_repo ) { $f_hub_root = gpc_get_string( 'hub_root' ); $f_hub_repoid = gpc_get_string( 'hub_repoid' ); $f_hub_reponame = gpc_get_string( 'hub_reponame' ); # Getting the repoid doesnt seem to work with the latest gitlab - but we can get all the # repos the current user can access and go through them to find the correct id if( empty( $f_hub_repoid ) ) { $t_hub_reponame_enc = urlencode( $f_hub_reponame ); $t_uri = $this->api_uri( $p_repo, "projects" ); $t_member = null; $t_json = json_url( $t_uri, $t_member ); $f_hub_repoid = 'RepoName is invalid'; if( !is_null( $t_json ) ) { foreach( $t_json as $project ) { if ( property_exists( $project, 'path_with_namespace' ) && $project->path_with_namespace == $f_hub_reponame && property_exists( $project, 'id' ) ) { $f_hub_repoid = (string)$project ->id; } } } } $f_hub_app_secret = gpc_get_string( 'hub_app_secret' ); $f_master_branch = gpc_get_string( 'master_branch' ); if ( !preg_match( '/\*|^[a-zA-Z0-9_\., -]*$/', $f_master_branch ) ) { plugin_error( self::ERROR_INVALID_PRIMARY_BRANCH ); } $p_repo->info['hub_root'] = $f_hub_root; $p_repo->info['hub_repoid'] = $f_hub_repoid; $p_repo->info['hub_reponame'] = $f_hub_reponame; $p_repo->info['hub_app_secret'] = $f_hub_app_secret; $p_repo->info['master_branch'] = $f_master_branch; return $p_repo; } private function api_uri( $p_repo, $p_path ) { $t_root = $p_repo->info['hub_root']; $t_uri = $t_root . '/api/v3/' . $p_path; if( isset( $p_repo->info['hub_app_secret'] ) ) { $t_access_token = $p_repo->info['hub_app_secret']; if ( !is_blank( $t_access_token ) ) { $t_uri .= '?private_token=' . $t_access_token; } } return $t_uri; } public function precommit() { $f_payload = file_get_contents( "php://input" ); if ( is_null( $f_payload ) ) { return; } $t_data = json_decode( $f_payload, true ); $t_repoid = $t_data['project_id']; $t_repo_table = plugin_table( 'repository', 'Source' ); $t_query = "SELECT * FROM $t_repo_table WHERE info LIKE " . db_param(); $t_result = db_query_bound( $t_query, array( '%' . $t_repoid . '%' ) ); if ( db_num_rows( $t_result ) < 1 ) { return; } while ( $t_row = db_fetch_array( $t_result ) ) { $t_repo = new SourceRepo( $t_row['type'], $t_row['name'], $t_row['url'], $t_row['info'] ); $t_repo->id = $t_row['id']; if ( $t_repo->info['hub_repoid'] == $t_repoid ) { return array( 'repo' => $t_repo, 'data' => $t_data ); } } return; } public function commit( $p_repo, $p_data ) { $t_commits = array(); foreach( $p_data['commits'] as $t_commit ) { $t_commits[] = $t_commit['id']; } $t_refData = split( '/', $p_data['ref'] ); $t_branch = $t_refData[2]; return $this->import_commits( $p_repo, $t_commits, $t_branch ); } public function import_full( $p_repo ) { echo '<pre>'; $t_branch = $p_repo->info['master_branch']; if ( is_blank( $t_branch ) ) { $t_branch = 'master'; } # if we're not allowed everything, populate an array of what we are allowed if( $t_branch != '*' ) { $t_branches_allowed = array_map( 'trim', explode( ',', $t_branch ) ); } # Always pull back full list of repos $t_repoid = $p_repo->info['hub_repoid']; $t_uri = $this->api_uri( $p_repo, "projects/$t_repoid/repository/branches" ); $t_member = null; $t_json = json_url( $t_uri, $t_member ); $t_branches = array(); foreach( $t_json as $t_branch ) { if( empty( $t_branches_allowed ) || in_array( $t_branch->name, $t_branches_allowed ) ) { $t_branches[] = $t_branch; } } $t_changesets = array(); $t_changeset_table = plugin_table( 'changeset', 'Source' ); foreach( $t_branches as $t_branch ) { $t_query = "SELECT parent FROM $t_changeset_table WHERE repo_id=" . db_param() . ' AND branch=' . db_param() . ' ORDER BY timestamp ASC'; $t_result = db_query_bound( $t_query, array( $p_repo->id, $t_branch->name ), 1 ); $t_commits = array( $t_branch->commit->id ); if ( db_num_rows( $t_result ) > 0 ) { $t_parent = db_result( $t_result ); echo "Oldest '$t_branch->name' branch parent: '$t_parent'\n"; if ( !empty( $t_parent ) ) { $t_commits[] = $t_parent; echo "Parents not empty"; } echo "Parents empty"; } $t_changesets = array_merge( $t_changesets, $this->import_commits( $p_repo, $t_commits, $t_branch->name ) ); } echo '</pre>'; return $t_changesets; } public function import_latest( $p_repo ) { return $this->import_full( $p_repo ); } public function import_commits( $p_repo, $p_commit_ids, $p_branch='' ) { static $s_parents = array(); static $s_counter = 0; $t_repoid = $p_repo->info['hub_repoid']; if ( is_array( $p_commit_ids ) ) { $s_parents = array_merge( $s_parents, $p_commit_ids ); } else { $s_parents[] = $p_commit_ids; } $t_changesets = array(); while( count( $s_parents ) > 0 && $s_counter < 200 ) { $t_commit_id = array_shift( $s_parents ); echo "Retrieving $t_commit_id ... <br/>"; $t_uri = $this->api_uri( $p_repo, "projects/$t_repoid/repository/commits/$t_commit_id" ); $t_member = null; $t_json = json_url( $t_uri, $t_member ); if ( false === $t_json || is_null( $t_json ) ) { # Some error occured retrieving the commit echo "failed.\n"; continue; } else if ( !property_exists( $t_json, 'id' ) ) { echo "failed ($t_json->message).\n"; continue; } list( $t_changeset, $t_commit_parents ) = $this->json_commit_changeset( $p_repo, $t_json, $p_branch ); if ( $t_changeset ) { $t_changesets[] = $t_changeset; } $s_parents = array_merge( $s_parents, $t_commit_parents ); } $s_counter = 0; return $t_changesets; } private function json_commit_changeset( $p_repo, $p_json, $p_branch='' ) { echo "processing $p_json->id ... "; if ( !SourceChangeset::exists( $p_repo->id, $p_json->id ) ) { $t_parents = array(); foreach( $p_json->parent_ids as $t_parent ) { $t_parents[] = $t_parent; } $t_changeset = new SourceChangeset( $p_repo->id, $p_json->id, $p_branch, date( 'Y-m-d H:i:s', strtotime( $p_json->authored_date ) ), $p_json->author_name, $p_json->message ); if ( count( $p_json->parents ) > 0 ) { $t_parent = $p_json->parents[0]; $t_changeset->parent = $t_parent->id; } $t_changeset->author_email = $p_json->author_email; $t_changeset->save(); echo "saved.\n"; return array( $t_changeset, $t_parents ); } else { echo "already exists.\n"; return array( null, array() ); } } public static function url_post( $p_url, $p_post_data ) { $t_post_data = http_build_query( $p_post_data ); # Use the PHP cURL extension if( function_exists( 'curl_init' ) ) { $t_curl = curl_init( $p_url ); curl_setopt( $t_curl, CURLOPT_RETURNTRANSFER, true ); curl_setopt( $t_curl, CURLOPT_POST, true ); curl_setopt( $t_curl, CURLOPT_POSTFIELDS, $t_post_data ); $t_data = curl_exec( $t_curl ); curl_close( $t_curl ); return $t_data; } else { # Last resort system call $t_url = escapeshellarg( $p_url ); $t_post_data = escapeshellarg( $t_post_data ); return shell_exec( 'curl ' . $t_url . ' -d ' . $t_post_data ); } } }