39#include <wx/filename.h>
49 if( aMessage.empty() )
52 size_t firstLineEnd = aMessage.find_first_of(
'\n' );
54 if( firstLineEnd != std::string::npos )
55 return aMessage.substr( 0, firstLineEnd );
64 time_t time =
static_cast<time_t
>( aTime.time );
68 localtime_s( &timeInfo, &time );
70 gmtime_r( &time, &timeInfo );
73 strftime( dateBuffer,
sizeof( dateBuffer ),
"%Y-%b-%d %H:%M:%S", &timeInfo );
93 constexpr auto kOrphanJoinTimeout = std::chrono::seconds( 5 );
99 "LIBGIT_BACKEND::Shutdown(): %zu orphan git thread(s) "
100 "did not finish within %lld ms; skipping libgit2 shutdown",
102 static_cast<long long>( kOrphanJoinTimeout.count() ) );
112 git_libgit2_shutdown();
118#if ( LIBGIT2_VER_MAJOR >= 1 ) || ( LIBGIT2_VER_MINOR >= 99 )
119 int major = 0, minor = 0, rev = 0;
120 return git_libgit2_version( &major, &minor, &rev ) == GIT_OK;
131 std::unique_lock<std::mutex> lock( common->
m_gitActionMutex, std::try_to_lock );
133 if( !lock.owns_lock() )
135 wxLogTrace(
traceGit,
"GIT_CLONE_HANDLER::PerformClone() could not lock" );
141 if( !clonePath.DirExists() )
143 if( !clonePath.Mkdir( wxS_DIR_DEFAULT, wxPATH_MKDIR_FULL ) )
145 aHandler->
AddErrorString( wxString::Format(
_(
"Could not create directory '%s'" ),
151 git_clone_options cloneOptions;
152 git_clone_init_options( &cloneOptions, GIT_CLONE_OPTIONS_VERSION );
153 cloneOptions.checkout_opts.checkout_strategy = GIT_CHECKOUT_SAFE;
155 cloneOptions.checkout_opts.progress_payload = aHandler;
158 cloneOptions.fetch_opts.callbacks.payload = aHandler;
159 cloneOptions.fetch_opts.proxy_opts.type = GIT_PROXY_AUTO;
163 git_repository* newRepo =
nullptr;
166 if( git_clone( &newRepo, remote.mbc_str(), aHandler->
GetClonePath().mbc_str(),
167 &cloneOptions ) != 0 )
180 const std::vector<wxString>& aFiles,
181 const wxString& aMessage,
182 const wxString& aAuthorName,
183 const wxString& aAuthorEmail )
185 git_repository* repo = aHandler->
GetRepo();
190 git_index*
index =
nullptr;
192 if( git_repository_index( &
index, repo ) != 0 )
194 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to get repository index: %s" ),
201 for(
const wxString& file : aFiles )
203 if( git_index_add_bypath(
index, file.mb_str() ) != 0 )
205 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to add file to index: %s" ),
211 if( git_index_write(
index ) != 0 )
213 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to write index: %s" ),
220 if( git_index_write_tree( &tree_id,
index ) != 0 )
222 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to write tree: %s" ),
227 git_tree* tree =
nullptr;
229 if( git_tree_lookup( &tree, repo, &tree_id ) != 0 )
231 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to lookup tree: %s" ),
237 git_commit* parent =
nullptr;
239 if( git_repository_head_unborn( repo ) == 0 )
241 git_reference* headRef =
nullptr;
243 if( git_repository_head( &headRef, repo ) != 0 )
245 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to get HEAD reference: %s" ),
252 if( git_reference_peel( (git_object**) &parent, headRef, GIT_OBJECT_COMMIT ) != 0 )
254 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to get commit: %s" ),
262 git_signature* author =
nullptr;
264 if( git_signature_now( &author, aAuthorName.mb_str(), aAuthorEmail.mb_str() ) != 0 )
266 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to create author signature: %s" ),
273 size_t parentsCount = parent ? 1 : 0;
275#if( LIBGIT2_VER_MAJOR == 1 && LIBGIT2_VER_MINOR == 8 \
276 && ( LIBGIT2_VER_REVISION < 2 || LIBGIT2_VER_REVISION == 3 ) )
277 git_commit*
const parents[1] = { parent };
278 git_commit**
const parentsPtr = parent ? parents :
nullptr;
280 const git_commit* parents[1] = { parent };
281 const git_commit** parentsPtr = parent ? parents :
nullptr;
284 if( git_commit_create( &oid, repo,
"HEAD", author, author,
nullptr,
285 aMessage.mb_str(), tree, parentsCount, parentsPtr ) != 0 )
287 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to create commit: %s" ),
297 const wxString& aMessage,
const wxString& aAuthorName,
298 const wxString& aAuthorEmail )
300 git_repository* repo = aHandler->
GetRepo();
305 if( git_repository_head_unborn( repo ) != 0 )
307 aHandler->
AddErrorString(
_(
"Cannot amend: the branch has no commits yet." ) );
311 git_reference* headRef =
nullptr;
313 if( git_repository_head( &headRef, repo ) != 0 )
321 git_commit* headCommit =
nullptr;
323 if( git_reference_peel( (git_object**) &headCommit, headRef, GIT_OBJECT_COMMIT ) != 0 )
331 git_index*
index =
nullptr;
333 if( git_repository_index( &
index, repo ) != 0 )
342 git_commit* parentCommit =
nullptr;
344 if( git_commit_parentcount( headCommit ) > 0 && git_commit_parent( &parentCommit, headCommit, 0 ) != 0 )
355 git_tree* headTree =
nullptr;
357 if( git_commit_tree( &headTree, headCommit ) != 0 )
366 if( git_index_read_tree(
index, headTree ) != 0 )
378 git_tree* parentTree =
nullptr;
380 if( git_commit_tree( &parentTree, parentCommit ) != 0 )
389 if( git_index_read_tree(
index, parentTree ) != 0 )
399 git_index_clear(
index );
402 const char* workdir = git_repository_workdir( repo );
404 for(
const wxString& file : aFiles )
406 bool onDisk = workdir && wxFileName::FileExists( wxString::FromUTF8( workdir ) + file );
408 int rc = onDisk ? git_index_add_bypath(
index, file.mb_str() )
409 : git_index_remove_bypath(
index, file.mb_str() );
420 if( git_index_write(
index ) != 0 )
429 if( git_index_write_tree( &tree_id,
index ) != 0 )
436 git_tree* tree =
nullptr;
438 if( git_tree_lookup( &tree, repo, &tree_id ) != 0 )
446 git_signature* author =
nullptr;
448 if( git_signature_now( &author, aAuthorName.mb_str(), aAuthorEmail.mb_str() ) != 0 )
458 if( git_commit_amend( &oid, headCommit,
"HEAD", author, author,
nullptr, aMessage.mb_str(), tree ) != 0 )
472 std::unique_lock<std::mutex> lock( common->
m_gitActionMutex, std::try_to_lock );
474 if( !lock.owns_lock() )
476 wxLogTrace(
traceGit,
"GIT_PUSH_HANDLER::PerformPush: Could not lock mutex" );
483 std::string remoteNameUtf8 = remoteName.utf8_string();
484 git_remote* remote =
nullptr;
486 if( git_remote_lookup( &remote, aHandler->
GetRepo(), remoteNameUtf8.c_str() ) != 0 )
488 aHandler->
AddErrorString( wxString::Format(
_(
"Could not lookup remote '%s'" ),
495 git_remote_callbacks remoteCallbacks;
496 git_remote_init_callbacks( &remoteCallbacks, GIT_REMOTE_CALLBACKS_VERSION );
502 remoteCallbacks.payload = aHandler;
504 git_proxy_options proxyOpts;
505 git_proxy_init_options( &proxyOpts, GIT_PROXY_OPTIONS_VERSION );
506 proxyOpts.type = GIT_PROXY_AUTO;
512 if( git_remote_connect( remote, GIT_DIRECTION_PUSH, &remoteCallbacks, &proxyOpts,
nullptr ) )
514 aHandler->
AddErrorString( wxString::Format(
_(
"Could not connect to remote: %s" ),
519 git_push_options pushOptions;
520 git_push_init_options( &pushOptions, GIT_PUSH_OPTIONS_VERSION );
521 pushOptions.callbacks = remoteCallbacks;
522 pushOptions.proxy_opts.type = GIT_PROXY_AUTO;
524 git_reference* head =
nullptr;
526 if( git_repository_head( &head, aHandler->
GetRepo() ) != 0 )
528 git_remote_disconnect( remote );
536 wxString refspec = ( aForce ? wxS(
"+" ) : wxS(
"" ) ) + wxString( git_reference_name( head ) );
537 std::string refspecUtf8 = refspec.utf8_string();
538 const char* refs[1] = { refspecUtf8.c_str() };
539 const git_strarray refspecs = { (
char**) refs, 1 };
541 if( git_remote_push( remote, &refspecs, &pushOptions ) )
544 aHandler->
AddErrorString( wxString::Format(
_(
"Could not push to remote: %s" ), errorMsg ) );
545 git_remote_disconnect( remote );
547 wxString lower = errorMsg.Lower();
549 if( lower.Contains( wxS(
"non-fast-forward" ) ) || lower.Contains( wxS(
"non-fastforwardable" ) )
550 || lower.Contains( wxS(
"would not be" ) ) )
559 if( git_reference_is_branch( head ) )
561 git_reference* upstreamRef =
nullptr;
562 int rc = git_branch_upstream( &upstreamRef, head );
564 if( rc == GIT_ENOTFOUND )
566 wxString upstreamName = wxString::Format(
"%s/%s", remoteName, git_reference_shorthand( head ) );
567 git_branch_set_upstream( head, upstreamName.utf8_string().c_str() );
569 else if( rc == GIT_OK )
575 git_remote_disconnect( remote );
583 git_repository* repo = aHandler->
GetRepo();
588 git_status_options opts;
589 git_status_init_options( &opts, GIT_STATUS_OPTIONS_VERSION );
591 opts.show = GIT_STATUS_SHOW_INDEX_AND_WORKDIR;
592 opts.flags = GIT_STATUS_OPT_INCLUDE_UNTRACKED | GIT_STATUS_OPT_RENAMES_HEAD_TO_INDEX
593 | GIT_STATUS_OPT_SORT_CASE_SENSITIVELY;
595 git_status_list* status_list =
nullptr;
597 if( git_status_list_new( &status_list, repo, &opts ) != GIT_OK )
604 bool hasChanges = ( git_status_list_entrycount( status_list ) > 0 );
611 const wxString& aPathspec )
613 std::map<wxString, FileStatus> fileStatusMap;
614 git_repository* repo = aHandler->
GetRepo();
617 return fileStatusMap;
619 git_status_options status_options;
620 git_status_init_options( &status_options, GIT_STATUS_OPTIONS_VERSION );
621 status_options.show = GIT_STATUS_SHOW_INDEX_AND_WORKDIR;
622 status_options.flags = GIT_STATUS_OPT_INCLUDE_UNTRACKED | GIT_STATUS_OPT_INCLUDE_UNMODIFIED
623 | GIT_STATUS_OPT_RECURSE_UNTRACKED_DIRS;
625 std::string pathspec_str;
626 std::vector<const char*> pathspec_ptrs;
628 if( !aPathspec.IsEmpty() )
630 pathspec_str = aPathspec.ToStdString();
631 pathspec_ptrs.push_back( pathspec_str.c_str() );
633 status_options.pathspec.strings =
const_cast<char**
>( pathspec_ptrs.data() );
634 status_options.pathspec.count = pathspec_ptrs.size();
637 git_status_list* status_list =
nullptr;
639 if( git_status_list_new( &status_list, repo, &status_options ) != GIT_OK )
642 return fileStatusMap;
647 size_t count = git_status_list_entrycount( status_list );
650 for(
size_t ii = 0; ii < count; ++ii )
652 const git_status_entry* entry = git_status_byindex( status_list, ii );
653 std::string
path( entry->head_to_index ? entry->head_to_index->old_file.path
654 : entry->index_to_workdir->old_file.path );
656 wxString absPath = repoWorkDir +
path;
659 static_cast<unsigned int>( entry->status ) };
662 return fileStatusMap;
668 git_repository* repo = aHandler->
GetRepo();
671 return wxEmptyString;
673 git_reference* currentBranchReference =
nullptr;
674 int rc = git_repository_head( ¤tBranchReference, repo );
677 if( currentBranchReference )
679 return git_reference_shorthand( currentBranchReference );
681 else if( rc == GIT_EUNBORNBRANCH )
683 return wxEmptyString;
688 return wxEmptyString;
694 const std::set<wxString>& aLocalChanges,
695 const std::set<wxString>& aRemoteChanges,
696 std::map<wxString, FileStatus>& aFileStatus )
698 git_repository* repo = aHandler->
GetRepo();
705 for(
auto& [absPath, fileStatus] : aFileStatus )
707 wxString relativePath = absPath;
708 if( relativePath.StartsWith( repoWorkDir ) )
710 relativePath = relativePath.Mid( repoWorkDir.length() );
713 relativePath.Replace( wxS(
"\\" ), wxS(
"/" ) );
717 std::string relativePathStd = relativePath.ToStdString();
721 if( aLocalChanges.count( relativePathStd ) )
725 else if( aRemoteChanges.count( relativePathStd ) )
748 git_repository* repo = aHandler->
GetRepo();
753 git_config*
config =
nullptr;
755 if( git_repository_config( &
config, repo ) != GIT_OK )
763 git_config_entry* entry =
nullptr;
764 int result = git_config_get_entry( &entry,
config, aKey.mb_str() );
767 if(
result != GIT_OK || entry ==
nullptr )
769 wxLogTrace(
traceGit,
"Config key '%s' not found", aKey );
773 aValue = wxString( entry->value );
780 git_repository* repo =
nullptr;
781 int error = git_repository_open( &repo, aPath.mb_str() );
785 git_repository_free( repo );
800 git_repository* repo =
nullptr;
802 if( git_repository_init( &repo, aPath.mb_str(), 0 ) != GIT_OK )
805 git_repository_free( repo );
807 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to initialize Git repository: %s" ),
814 wxLogTrace(
traceGit,
"Successfully initialized Git repository at %s", aPath );
821 if( aConfig.
url.IsEmpty() )
824 git_repository* repo = aHandler->
GetRepo();
828 aHandler->
AddErrorString(
_(
"No repository available to set up remote" ) );
836 git_remote* remote =
nullptr;
843 if( !aConfig.
url.Contains(
"@" ) && !aConfig.
username.IsEmpty() )
844 userPrefix = aConfig.
username +
"@";
846 if( aConfig.
url.StartsWith(
"ssh://" ) )
847 fullURL =
"ssh://" + userPrefix + aConfig.
url.Mid( 6 );
848 else if( aConfig.
url.Contains(
":" ) )
849 fullURL = userPrefix + aConfig.
url;
851 fullURL =
"ssh://" + userPrefix + aConfig.
url;
855 fullURL = aConfig.
url.StartsWith(
"https" ) ?
"https://" :
"http://";
863 fullURL.append( wxS(
":" ) );
867 fullURL.append( wxS(
"@" ) );
870 wxString bareURL = aConfig.
url;
872 if( bareURL.StartsWith(
"https://" ) )
873 bareURL = bareURL.Mid( 8 );
874 else if( bareURL.StartsWith(
"http://" ) )
875 bareURL = bareURL.Mid( 7 );
877 fullURL.append( bareURL );
881 fullURL = aConfig.
url;
886 if( git_remote_lookup( &remote, repo,
"origin" ) == GIT_OK )
889 error = git_remote_set_url( repo,
"origin", fullURL.ToStdString().c_str() );
893 error = git_remote_create_with_fetchspec( &remote, repo,
"origin", fullURL.ToStdString().c_str(),
894 "+refs/heads/*:refs/remotes/origin/*" );
898 if( error != GIT_OK )
909 wxLogTrace(
traceGit,
"Successfully set up remote origin" );
915 git_reference** aReference )
917 if( git_reference_lookup( aReference, repo, aBranchName.mb_str() ) == GIT_OK )
920 if( git_reference_dwim( aReference, repo, aBranchName.mb_str() ) == GIT_OK )
929 git_repository* repo = aHandler->
GetRepo();
937 git_reference* branchRef =
nullptr;
941 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to lookup branch '%s': %s" ),
947 const char* branchRefName = git_reference_name( branchRef );
948 git_object* branchObj =
nullptr;
950 if( git_revparse_single( &branchObj, repo, aBranchName.mb_str() ) != GIT_OK )
952 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to find branch head for '%s': %s" ),
960 git_checkout_options checkoutOpts;
961 git_checkout_init_options( &checkoutOpts, GIT_CHECKOUT_OPTIONS_VERSION );
962 checkoutOpts.checkout_strategy = GIT_CHECKOUT_SAFE;
964 if( git_checkout_tree( repo, branchObj, &checkoutOpts ) != GIT_OK )
966 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to switch to branch '%s': %s" ),
972 const char* headTarget = branchRefName;
974 if( git_reference_is_remote( branchRef ) )
976 wxString localName = wxString::FromUTF8( git_reference_shorthand( branchRef ) );
977 size_t slash = localName.find(
'/' );
979 if( slash != wxString::npos )
980 localName = localName.Mid( slash + 1 );
982 std::string localNameUtf8 = localName.utf8_string();
983 git_reference* localBranch =
nullptr;
985 if( git_branch_lookup( &localBranch, repo, localNameUtf8.c_str(), GIT_BRANCH_LOCAL ) != GIT_OK )
987 git_commit* target =
nullptr;
989 if( git_commit_lookup( &target, repo, git_object_id( branchObj ) ) != GIT_OK )
991 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to switch to branch '%s': %s" ), aBranchName,
998 if( git_branch_create( &localBranch, repo, localNameUtf8.c_str(), target, 0 ) != GIT_OK )
1000 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to create local branch '%s': %s" ), localName,
1005 git_branch_set_upstream( localBranch, git_reference_shorthand( branchRef ) );
1008 localBranchPtr.reset( localBranch );
1009 headTarget = git_reference_name( localBranch );
1012 if( git_repository_set_head( repo, headTarget ) != GIT_OK )
1014 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to update HEAD reference for branch '%s': %s" ),
1019 wxLogTrace(
traceGit,
"Successfully switched to branch '%s'", aBranchName );
1026 git_repository* repo = aHandler->
GetRepo();
1031 git_reference* branchRef =
nullptr;
1035 git_reference_free( branchRef );
1047 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformFetch() - No repository found" );
1053 if( !aSkipLock && !lock.owns_lock() )
1055 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformFetch() - Could not lock mutex" );
1060 std::string remoteNameUtf8 = remoteName.utf8_string();
1061 git_remote* remote =
nullptr;
1063 if( git_remote_lookup( &remote, aHandler->
GetRepo(), remoteNameUtf8.c_str() ) != 0 )
1065 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformFetch() - Failed to lookup remote '%s'",
1067 aHandler->
AddErrorString( wxString::Format(
_(
"Could not lookup remote '%s'" ),
1074 git_remote_callbacks remoteCallbacks;
1075 git_remote_init_callbacks( &remoteCallbacks, GIT_REMOTE_CALLBACKS_VERSION );
1079 remoteCallbacks.payload = aHandler;
1081 git_proxy_options proxyOpts;
1082 git_proxy_init_options( &proxyOpts, GIT_PROXY_OPTIONS_VERSION );
1083 proxyOpts.type = GIT_PROXY_AUTO;
1089 if( git_remote_connect( remote, GIT_DIRECTION_FETCH, &remoteCallbacks, &proxyOpts,
nullptr ) )
1092 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformFetch() - Failed to connect to remote: %s", errorMsg );
1093 aHandler->
AddErrorString( wxString::Format(
_(
"Could not connect to remote '%s': %s" ),
1094 remoteName, errorMsg ) );
1098 git_fetch_options fetchOptions;
1099 git_fetch_init_options( &fetchOptions, GIT_FETCH_OPTIONS_VERSION );
1100 fetchOptions.callbacks = remoteCallbacks;
1102 if( git_remote_fetch( remote,
nullptr, &fetchOptions,
nullptr ) )
1105 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformFetch() - Failed to fetch from remote: %s", errorMsg );
1106 aHandler->
AddErrorString( wxString::Format(
_(
"Could not fetch data from remote '%s': %s" ),
1107 remoteName, errorMsg ) );
1111 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformFetch() - Fetch completed successfully" );
1121 if( !lock.owns_lock() )
1123 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformPull() - Could not lock mutex" );
1130 git_oid pull_merge_oid = {};
1141#if ( LIBGIT2_VER_MAJOR >= 1 ) || ( LIBGIT2_VER_MINOR >= 99 )
1142 if( git_oid_is_zero( &pull_merge_oid ) )
1144 if( git_oid_iszero( &pull_merge_oid ) )
1147 git_reference* head_ref =
nullptr;
1149 if( git_repository_head( &head_ref, aHandler->
GetRepo() ) == GIT_OK )
1153 if( git_reference_is_branch( head_ref ) )
1155 const char* branch_shorthand = git_reference_shorthand( head_ref );
1157 wxString remoteRefName = wxString::Format(
"refs/remotes/%s/%s", remoteName, branch_shorthand );
1159 if( git_reference_name_to_id( &pull_merge_oid, aHandler->
GetRepo(),
1160 remoteRefName.utf8_string().c_str() )
1163 wxString upstream = wxString::Format(
"%s/%s", remoteName, branch_shorthand );
1164 git_branch_set_upstream( head_ref, upstream.utf8_string().c_str() );
1170#if ( LIBGIT2_VER_MAJOR >= 1 ) || ( LIBGIT2_VER_MINOR >= 99 )
1171 if( git_oid_is_zero( &pull_merge_oid ) )
1173 if( git_oid_iszero( &pull_merge_oid ) )
1176 aHandler->
AddErrorString(
_(
"Nothing to pull: the remote has no branch matching "
1177 "the current local branch." ) );
1181 git_annotated_commit* fetchhead_commit;
1183 if( git_annotated_commit_lookup( &fetchhead_commit, aHandler->
GetRepo(), &pull_merge_oid ) )
1190 const git_annotated_commit* merge_commits[] = { fetchhead_commit };
1191 git_merge_analysis_t merge_analysis;
1192 git_merge_preference_t merge_preference = GIT_MERGE_PREFERENCE_NONE;
1194 if( git_merge_analysis( &merge_analysis, &merge_preference, aHandler->
GetRepo(), merge_commits, 1 ) )
1200 if( merge_analysis & GIT_MERGE_ANALYSIS_UNBORN )
1206 if( merge_analysis & GIT_MERGE_ANALYSIS_UP_TO_DATE )
1208 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformPull() - Repository is up to date" );
1209 git_repository_state_cleanup( aHandler->
GetRepo() );
1213 if( merge_analysis & GIT_MERGE_ANALYSIS_FASTFORWARD )
1215 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformPull() - Fast-forward merge" );
1219 if( merge_analysis & GIT_MERGE_ANALYSIS_NORMAL )
1221 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformPull() - Normal merge" );
1223 git_config*
config =
nullptr;
1225 if( git_repository_config( &
config, aHandler->
GetRepo() ) != GIT_OK )
1227 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformPull() - Failed to get repository config" );
1228 aHandler->
AddErrorString(
_(
"Could not access repository configuration" ) );
1234 int rebase_value = 0;
1235 int ret = git_config_get_bool( &rebase_value,
config,
"pull.rebase" );
1237 if( ret == GIT_OK && rebase_value )
1239 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformPull() - Using rebase based on config" );
1243 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformPull() - Using merge based on config" );
1247 wxLogTrace(
traceGit,
"GIT_PULL_HANDLER::PerformPull() - Merge needs resolution" );
1254 git_reference* rawRef =
nullptr;
1256 if( git_repository_head( &rawRef, aHandler->
GetRepo() ) )
1264 git_oid updatedRefOid;
1265 const char* currentBranchName = git_reference_name( rawRef );
1266 const char* branch_shorthand = git_reference_shorthand( rawRef );
1268 wxString remoteBranchName = wxString::Format(
"refs/remotes/%s/%s", remote_name, branch_shorthand );
1270 if( git_reference_name_to_id( &updatedRefOid, aHandler->
GetRepo(), remoteBranchName.c_str() ) != GIT_OK )
1272 aHandler->
AddErrorString( wxString::Format(
_(
"Could not get reference OID for reference '%s'" ),
1273 remoteBranchName ) );
1277 git_commit* targetCommit =
nullptr;
1279 if( git_commit_lookup( &targetCommit, aHandler->
GetRepo(), &updatedRefOid ) != GIT_OK )
1287 git_tree* targetTree =
nullptr;
1289 if( git_commit_tree( &targetTree, targetCommit ) != GIT_OK )
1291 git_commit_free( targetCommit );
1292 aHandler->
AddErrorString(
_(
"Could not get tree from target commit" ) );
1298 git_checkout_options checkoutOptions;
1299 git_checkout_init_options( &checkoutOptions, GIT_CHECKOUT_OPTIONS_VERSION );
1300 auto notify_cb = []( git_checkout_notify_t why,
const char*
path,
const git_diff_file* baseline,
1301 const git_diff_file* target,
const git_diff_file* workdir,
void* payload ) ->
int
1305 case GIT_CHECKOUT_NOTIFY_CONFLICT:
1308 case GIT_CHECKOUT_NOTIFY_DIRTY:
1311 case GIT_CHECKOUT_NOTIFY_UPDATED:
1314 case GIT_CHECKOUT_NOTIFY_UNTRACKED:
1317 case GIT_CHECKOUT_NOTIFY_IGNORED:
1327 checkoutOptions.checkout_strategy = GIT_CHECKOUT_SAFE | GIT_CHECKOUT_ALLOW_CONFLICTS;
1328 checkoutOptions.notify_flags = GIT_CHECKOUT_NOTIFY_ALL;
1329 checkoutOptions.notify_cb = notify_cb;
1331 if( git_checkout_tree( aHandler->
GetRepo(),
reinterpret_cast<git_object*
>( targetTree ),
1332 &checkoutOptions ) != GIT_OK )
1334 aHandler->
AddErrorString(
_(
"Failed to perform checkout operation." ) );
1338 git_reference* updatedRef =
nullptr;
1340 if( git_reference_set_target( &updatedRef, rawRef, &updatedRefOid,
nullptr ) != GIT_OK )
1342 aHandler->
AddErrorString( wxString::Format(
_(
"Failed to update reference '%s' to point to '%s'" ),
1343 currentBranchName, git_oid_tostr_s( &updatedRefOid ) ) );
1349 if( git_repository_state_cleanup( aHandler->
GetRepo() ) != GIT_OK )
1351 aHandler->
AddErrorString(
_(
"Failed to clean up repository state after fast-forward." ) );
1355 git_revwalk* revWalker =
nullptr;
1357 if( git_revwalk_new( &revWalker, aHandler->
GetRepo() ) != GIT_OK )
1359 aHandler->
AddErrorString(
_(
"Failed to initialize revision walker." ) );
1364 git_revwalk_sorting( revWalker, GIT_SORT_TIME );
1366 if( git_revwalk_push_glob( revWalker, currentBranchName ) != GIT_OK )
1368 aHandler->
AddErrorString(
_(
"Failed to push reference to revision walker." ) );
1372 std::pair<std::string, std::vector<CommitDetails>>& branchCommits = aHandler->
m_fetchResults.emplace_back();
1373 branchCommits.first = currentBranchName;
1377 while( git_revwalk_next( &commitOid, revWalker ) == GIT_OK )
1379 git_commit* commit =
nullptr;
1381 if( git_commit_lookup( &commit, aHandler->
GetRepo(), &commitOid ) )
1383 aHandler->
AddErrorString( wxString::Format(
_(
"Could not lookup commit '%s'" ),
1384 git_oid_tostr_s( &commitOid ) ) );
1391 details.
m_sha = git_oid_tostr_s( &commitOid );
1393 details.
m_author = git_commit_author( commit )->name;
1396 branchCommits.second.push_back( details );
1408 git_status_options opts;
1409 git_status_init_options( &opts, GIT_STATUS_OPTIONS_VERSION );
1412 opts.show = GIT_STATUS_SHOW_WORKDIR_ONLY;
1413 opts.flags = GIT_STATUS_OPT_INCLUDE_UNTRACKED;
1415 git_status_list* status_list =
nullptr;
1417 if( git_status_list_new( &status_list, aRepo, &opts ) != GIT_OK )
1424 size_t count = git_status_list_entrycount( status_list );
1427 for(
size_t ii = 0; ii < count; ++ii )
1429 const git_status_entry* entry = git_status_byindex( status_list, ii );
1432 if( entry->status & ( GIT_STATUS_WT_MODIFIED | GIT_STATUS_WT_DELETED | GIT_STATUS_WT_TYPECHANGE ) )
1443 const git_annotated_commit** aMergeHeads,
1444 size_t aMergeHeadsCount )
1450 _(
"Cannot merge: you have unstaged changes. "
1451 "Please commit or stash them before pulling." ) );
1455 git_repository* repo = aHandler->
GetRepo();
1457 if( git_merge( repo, aMergeHeads, aMergeHeadsCount,
nullptr,
nullptr ) )
1460 aHandler->
AddErrorString( wxString::Format(
_(
"Merge failed: %s" ), errorMsg ) );
1464 git_index*
index =
nullptr;
1466 if( git_repository_index( &
index, repo ) != GIT_OK )
1468 aHandler->
AddErrorString(
_(
"Could not read repository index after merge." ) );
1474 if( git_index_has_conflicts(
index ) )
1477 git_object* head_obj =
nullptr;
1479 if( git_revparse_single( &head_obj, repo,
"HEAD" ) == GIT_OK )
1482 git_reset( repo, head_obj, GIT_RESET_HARD,
nullptr );
1485 git_repository_state_cleanup( repo );
1491 git_tree* tree =
nullptr;
1492 git_reference* head_ref =
nullptr;
1494 git_commit* head_commit =
nullptr;
1495 git_commit* merge_commit =
nullptr;
1496 git_signature* signature =
nullptr;
1498 if( git_index_write_tree( &tree_oid,
index ) != GIT_OK || git_tree_lookup( &tree, repo, &tree_oid ) != GIT_OK )
1506 if( git_repository_head( &head_ref, repo ) != GIT_OK )
1508 aHandler->
AddErrorString(
_(
"Could not get the repository head." ) );
1514 if( git_reference_name_to_id( &head_oid, repo,
"HEAD" ) != GIT_OK
1515 || git_commit_lookup( &head_commit, repo, &head_oid ) != GIT_OK
1516 || git_commit_lookup( &merge_commit, repo, git_annotated_commit_id( aMergeHeads[0] ) ) != GIT_OK )
1518 aHandler->
AddErrorString(
_(
"Could not look up the commits to merge." ) );
1525 if( git_signature_default( &signature, repo ) != GIT_OK )
1527 aHandler->
AddErrorString(
_(
"Could not create a commit signature. Set user.name "
1528 "and user.email in your git configuration." ) );
1534 const git_commit* parents[] = { head_commit, merge_commit };
1536 wxString::Format(
_(
"Merge remote-tracking branch into %s" ), git_reference_shorthand( head_ref ) );
1537 git_oid merge_commit_oid;
1539 if( git_commit_create( &merge_commit_oid, repo,
"HEAD", signature, signature,
nullptr,
1540 message.utf8_string().c_str(), tree, 2, parents )
1548 git_repository_state_cleanup( repo );
1554 const git_annotated_commit** aMergeHeads,
1555 size_t aMergeHeadsCount )
1561 _(
"Cannot pull with rebase: you have unstaged changes. "
1562 "Please commit or stash them before pulling." ) );
1566 git_rebase_options rebase_opts;
1567 git_rebase_init_options( &rebase_opts, GIT_REBASE_OPTIONS_VERSION );
1569 git_rebase* rebase =
nullptr;
1571 if( git_rebase_init( &rebase, aHandler->
GetRepo(),
nullptr, aMergeHeads[0],
nullptr, &rebase_opts ) )
1574 aHandler->
AddErrorString( wxString::Format(
_(
"Rebase failed to start: %s" ), errorMsg ) );
1582 git_rebase_operation* op =
nullptr;
1584 if( git_rebase_next( &op, rebase ) != 0 )
1587 if( git_rebase_commit(
nullptr, rebase,
nullptr,
nullptr,
nullptr,
nullptr ) )
1590 aHandler->
AddErrorString( wxString::Format(
_(
"Rebase commit failed: %s" ), errorMsg ) );
1595 if( git_rebase_finish( rebase,
nullptr ) )
1598 aHandler->
AddErrorString( wxString::Format(
_(
"Rebase finish failed: %s" ), errorMsg ) );
1608 git_repository* repo = aHandler->
GetRepo();
1609 git_reference* head_ref =
nullptr;
1611 if( git_repository_head( &head_ref, repo ) != GIT_OK )
1616 wxString remoteRef = wxString::Format(
"refs/remotes/%s/%s", remoteName, git_reference_shorthand( head_ref ) );
1618 return git_reference_lookup( aUpstreamRef, repo, remoteRef.utf8_string().c_str() ) == GIT_OK;
1624 git_repository* repo = aHandler->
GetRepo();
1631 if( !lock.owns_lock() )
1633 aHandler->
AddErrorString(
_(
"Another git operation is in progress." ) );
1637 git_reference* upstream_ref =
nullptr;
1641 aHandler->
AddErrorString(
_(
"Could not find the matching remote branch." ) );
1646 git_object* target =
nullptr;
1648 if( git_reference_peel( &target, upstream_ref, GIT_OBJECT_COMMIT ) != GIT_OK )
1656 if( git_reset( repo, target, GIT_RESET_HARD,
nullptr ) != GIT_OK )
1663 git_repository_state_cleanup( repo );
1670 git_repository* repo = aHandler->
GetRepo();
1677 if( !lock.owns_lock() )
1679 aHandler->
AddErrorString(
_(
"Another git operation is in progress." ) );
1685 aHandler->
AddErrorString(
_(
"Cannot rebase: you have unstaged changes. Please commit "
1686 "or stash them first." ) );
1690 git_reference* upstream_ref =
nullptr;
1694 aHandler->
AddErrorString(
_(
"Could not find the matching remote branch." ) );
1699 git_annotated_commit* onto =
nullptr;
1701 if( git_annotated_commit_from_ref( &onto, repo, upstream_ref ) != GIT_OK )
1708 git_signature* signature =
nullptr;
1710 if( git_signature_default( &signature, repo ) != GIT_OK )
1712 aHandler->
AddErrorString(
_(
"Could not create a commit signature. Set user.name and "
1713 "user.email in your git configuration." ) );
1718 git_rebase_options rebase_opts;
1719 git_rebase_init_options( &rebase_opts, GIT_REBASE_OPTIONS_VERSION );
1720 git_rebase* rebase =
nullptr;
1722 if( git_rebase_init( &rebase, repo,
nullptr, onto,
nullptr, &rebase_opts ) != GIT_OK )
1730 git_rebase_operation* op =
nullptr;
1732 while( git_rebase_next( &op, rebase ) == GIT_OK )
1734 git_index*
index =
nullptr;
1736 if( git_repository_index( &
index, repo ) == GIT_OK )
1740 if( git_index_has_conflicts(
index ) )
1742 git_rebase_abort( rebase );
1743 aHandler->
AddErrorString(
_(
"The rebase ran into conflicts. This is best "
1744 "resolved from a git command line." ) );
1751 if( git_rebase_commit( &commit_oid, rebase,
nullptr, signature,
nullptr,
nullptr ) != GIT_OK )
1753 git_rebase_abort( rebase );
1754 aHandler->
AddErrorString(
_(
"The rebase ran into conflicts. This is best "
1755 "resolved from a git command line." ) );
1760 if( git_rebase_finish( rebase, signature ) != GIT_OK )
1762 git_rebase_abort( rebase );
1774 git_object* head_commit = NULL;
1775 git_checkout_options opts;
1776 git_checkout_init_options( &opts, GIT_CHECKOUT_OPTIONS_VERSION );
1778 if( git_revparse_single( &head_commit, aHandler->
m_repository,
"HEAD" ) != 0 )
1783 opts.checkout_strategy = GIT_CHECKOUT_FORCE;
1794 opts.progress_cb =
nullptr;
1795 opts.notify_cb =
nullptr;
1796 opts.notify_payload =
static_cast<void*
>( aHandler );
1798 if( git_checkout_tree( aHandler->
m_repository, head_commit, &opts ) != 0 )
1800 const git_error* e = git_error_last();
1804 wxLogTrace(
traceGit, wxS(
"Checkout failed: %d: %s" ), e->klass, e->message );
1809 delete( paths[ii] );
1813 git_object_free( head_commit );
1819 git_repository* repo =
nullptr;
1822 if( git_repository_discover( &repo_path, aFilename, 0,
nullptr ) != GIT_OK )
1824 wxLogTrace(
traceGit,
"Can't repo discover %s: %s", aFilename,
1831 if( git_repository_open( &repo, repo_path.ptr ) != GIT_OK )
1833 wxLogTrace(
traceGit,
"Can't open repo for %s: %s", repo_path.ptr,
1846 if(
int error = git_reference_name_to_id( &head_oid, aRepo,
"HEAD" ); error != GIT_OK )
1848 wxLogTrace(
traceGit,
"Failed to lookup HEAD reference: %s",
1853 git_commit* commit =
nullptr;
1855 if(
int error = git_commit_lookup( &commit, aRepo, &head_oid ); error != GIT_OK )
1857 wxLogTrace(
traceGit,
"Failed to lookup commit: %s",
1863 git_reference* branchRef =
nullptr;
1865 if(
int error = git_branch_create( &branchRef, aRepo, aBranchName.mb_str(), commit, 0 ); error != GIT_OK )
1867 wxLogTrace(
traceGit,
"Failed to create branch: %s",
1872 git_reference_free( branchRef );
1878 bool aRemoveGitDir, wxString* aErrors )
1882 git_repository_free( aRepo );
1888 wxFileName gitDir( aProjectPath, wxEmptyString );
1889 gitDir.AppendDir(
".git" );
1891 if( gitDir.DirExists() )
1900 wxLogTrace(
traceGit,
"Failed to remove .git directory: %s", errors );
1906 wxLogTrace(
traceGit,
"Successfully removed VCS from project" );
1913 git_repository* repo = aHandler->
GetRepo();
1915 git_index*
index =
nullptr;
1918 if( git_repository_index( &
index, repo ) != 0 )
1920 wxLogError(
"Failed to get repository index" );
1926 if( git_index_find( &at_pos,
index, aFilePath.ToUTF8().data() ) == GIT_OK )
1928 wxLogError(
"%s already in index", aFilePath );
1939 git_repository* repo = aHandler->
GetRepo();
1940 git_index*
index =
nullptr;
1944 if( git_repository_index( &
index, repo ) != 0 )
1946 wxLogError(
"Failed to get repository index" );
1956 if( git_index_add_bypath(
index, file.ToUTF8().data() ) != 0 )
1958 wxLogError(
"Failed to add %s to index", file );
1964 if( git_index_write(
index ) != 0 )
1966 wxLogError(
"Failed to write index" );
1979 git_repository* repo = aHandler->
GetRepo();
1980 git_index*
index =
nullptr;
1983 if( git_repository_index( &
index, repo ) != 0 )
1985 wxLogError(
"Failed to get repository index" );
1991 if( git_index_find( &at_pos,
index, aFilePath.ToUTF8().data() ) != 0 )
1993 wxLogError(
"Failed to find index entry for %s", aFilePath );
2004 git_repository* repo = aHandler->
GetRepo();
2008 git_index*
index =
nullptr;
2011 if( git_repository_index( &
index, repo ) != 0 )
2013 wxLogError(
"Failed to get repository index" );
2019 if( git_index_remove_bypath(
index, file.ToUTF8().data() ) != 0 )
2021 wxLogError(
"Failed to remove index entry for %s", file );
2025 if( git_index_write(
index ) != 0 )
2027 wxLogError(
"Failed to write index" );
2031 if( git_index_write_tree( &oid,
index ) != 0 )
2033 wxLogError(
"Failed to write index tree" );
std::vector< wxString > m_filesToAdd
std::vector< wxString > m_filesFailedToAdd
KIGIT_ORPHAN_REGISTRY m_orphanRegistry
wxString GetClonePath() const
void AddErrorString(const wxString &aErrorString)
std::vector< std::pair< std::string, std::vector< CommitDetails > > > m_fetchResults
std::vector< wxString > m_filesToRemove
std::vector< wxString > m_filesToRevert
git_repository * m_repository
KIGIT_COMMON::GIT_STATUS ConvertStatus(unsigned int aGitStatus)
Convert git status flags to KIGIT_COMMON::GIT_STATUS.
std::mutex m_gitActionMutex
static wxString GetLastGitError()
void SetSSHKey(const wxString &aSSHKey)
void SetUsername(const wxString &aUsername)
wxString GetRemoteNameOrDefault() const
Returns GetRemotename() when non-empty, otherwise "origin".
git_repository * GetRepo() const
void SetCancelled(bool aCancel)
void SetPassword(const wxString &aPassword)
void SetRemote(const wxString &aRemote)
void SetRepo(git_repository *aRepo)
void AddErrorString(const wxString aErrorString)
wxString GetProjectDir() const
Get the project directory path, preserving symlinks if set.
git_repository * GetRepo() const
Get a pointer to the git repository.
unsigned & TestedTypes()
Return the connection types that have been tested for authentication.
KIGIT_COMMON * GetCommon() const
Get the common object.
wxString GetRemotename() const
Get the remote name.
void ResetNextKey()
Reset the next public key to test.
bool PerformAddToIndex(GIT_ADD_TO_INDEX_HANDLER *aHandler) override
void PerformRevert(GIT_REVERT_HANDLER *aHandler) override
PullResult handleMerge(GIT_PULL_HANDLER *aHandler, const git_annotated_commit **aMergeHeads, size_t aMergeHeadsCount)
bool PerformFetch(GIT_PULL_HANDLER *aHandler, bool aSkipLock) override
static bool hasUnstagedChanges(git_repository *aRepo)
bool Clone(GIT_CLONE_HANDLER *aHandler) override
PullResult handleRebase(GIT_PULL_HANDLER *aHandler, const git_annotated_commit **aMergeHeads, size_t aMergeHeadsCount)
void PerformRemoveFromIndex(GIT_REMOVE_FROM_INDEX_HANDLER *aHandler) override
bool RemoveVCS(git_repository *&aRepo, const wxString &aProjectPath, bool aRemoveGitDir, wxString *aErrors) override
bool RemoveFromIndex(GIT_REMOVE_FROM_INDEX_HANDLER *aHandler, const wxString &aFilePath) override
BranchResult SwitchToBranch(GIT_BRANCH_HANDLER *aHandler, const wxString &aBranchName) override
PullResult handleFastForward(GIT_PULL_HANDLER *aHandler)
CommitResult Commit(GIT_COMMIT_HANDLER *aHandler, const std::vector< wxString > &aFiles, const wxString &aMessage, const wxString &aAuthorName, const wxString &aAuthorEmail) override
std::map< wxString, FileStatus > GetFileStatus(GIT_STATUS_HANDLER *aHandler, const wxString &aPathspec) override
bool HasChangedFiles(GIT_STATUS_HANDLER *aHandler) override
PullResult RebaseOntoUpstream(GIT_PULL_HANDLER *aHandler) override
void UpdateRemoteStatus(GIT_STATUS_HANDLER *aHandler, const std::set< wxString > &aLocalChanges, const std::set< wxString > &aRemoteChanges, std::map< wxString, FileStatus > &aFileStatus) override
wxString GetCurrentBranchName(GIT_STATUS_HANDLER *aHandler) override
bool GetConfigString(GIT_CONFIG_HANDLER *aHandler, const wxString &aKey, wxString &aValue) override
git_repository * GetRepositoryForFile(const char *aFilename) override
bool AddToIndex(GIT_ADD_TO_INDEX_HANDLER *aHandler, const wxString &aFilePath) override
bool SetupRemote(GIT_INIT_HANDLER *aHandler, const RemoteConfig &aConfig) override
bool IsRepository(GIT_INIT_HANDLER *aHandler, const wxString &aPath) override
bool IsLibraryAvailable() override
bool ResetToUpstream(GIT_PULL_HANDLER *aHandler) override
PushResult Push(GIT_PUSH_HANDLER *aHandler, bool aForce=false) override
wxString GetWorkingDirectory(GIT_STATUS_HANDLER *aHandler) override
bool BranchExists(GIT_BRANCH_HANDLER *aHandler, const wxString &aBranchName) override
CommitResult Amend(GIT_COMMIT_HANDLER *aHandler, const std::vector< wxString > &aFiles, const wxString &aMessage, const wxString &aAuthorName, const wxString &aAuthorEmail) override
InitResult InitializeRepository(GIT_INIT_HANDLER *aHandler, const wxString &aPath) override
PullResult PerformPull(GIT_PULL_HANDLER *aHandler) override
int CreateBranch(git_repository *aRepo, const wxString &aBranchName) override
bool RmDirRecursive(const wxString &aFileName, wxString *aErrors)
Remove the directory aDirName and all its contents including subdirectories and their files.
const wxChar *const traceGit
Flag to enable Git debugging output.
int fetchhead_foreach_cb(const char *, const char *, const git_oid *aOID, unsigned int aIsMerge, void *aPayload)
int progress_cb(const char *str, int len, void *aPayload)
int update_cb(const char *aRefname, const git_oid *aFirst, const git_oid *aSecond, void *aPayload)
int transfer_progress_cb(const git_transfer_progress *aStats, void *aPayload)
int credentials_cb(git_cred **aOut, const char *aUrl, const char *aUsername, unsigned int aAllowedTypes, void *aPayload)
void clone_progress_cb(const char *aStr, size_t aLen, size_t aTotal, void *aPayload)
int push_transfer_progress_cb(unsigned int aCurrent, unsigned int aTotal, size_t aBytes, void *aPayload)
static bool lookup_branch_reference(git_repository *repo, const wxString &aBranchName, git_reference **aReference)
static bool lookup_upstream_ref(GIT_PULL_HANDLER *aHandler, git_reference **aUpstreamRef)
static std::string getFormattedCommitDate(const git_time &aTime)
static std::string getFirstLineFromCommitMessage(const std::string &aMessage)
std::unique_ptr< git_tree, decltype([](git_tree *aTree) { git_tree_free(aTree); })> GitTreePtr
A unique pointer for git_tree objects with automatic cleanup.
std::unique_ptr< git_revwalk, decltype([](git_revwalk *aWalker) { git_revwalk_free(aWalker); })> GitRevWalkPtr
A unique pointer for git_revwalk objects with automatic cleanup.
std::unique_ptr< git_commit, decltype([](git_commit *aCommit) { git_commit_free(aCommit); })> GitCommitPtr
A unique pointer for git_commit objects with automatic cleanup.
std::unique_ptr< git_buf, decltype([](git_buf *aBuf) { git_buf_free(aBuf); })> GitBufPtr
A unique pointer for git_buf objects with automatic cleanup.
std::unique_ptr< git_annotated_commit, decltype([](git_annotated_commit *aCommit) { git_annotated_commit_free(aCommit); })> GitAnnotatedCommitPtr
A unique pointer for git_annotated_commit objects with automatic cleanup.
std::unique_ptr< git_status_list, decltype([](git_status_list *aList) { git_status_list_free(aList); })> GitStatusListPtr
A unique pointer for git_status_list objects with automatic cleanup.
std::unique_ptr< git_config, decltype([](git_config *aConfig) { git_config_free(aConfig); })> GitConfigPtr
A unique pointer for git_config objects with automatic cleanup.
std::unique_ptr< git_reference, decltype([](git_reference *aRef) { git_reference_free(aRef); })> GitReferencePtr
A unique pointer for git_reference objects with automatic cleanup.
std::unique_ptr< git_config_entry, decltype([](git_config_entry *aEntry) { git_config_entry_free(aEntry); })> GitConfigEntryPtr
A unique pointer for git_config_entry objects with automatic cleanup.
std::unique_ptr< git_signature, decltype([](git_signature *aSignature) { git_signature_free(aSignature); })> GitSignaturePtr
A unique pointer for git_signature objects with automatic cleanup.
std::unique_ptr< git_rebase, decltype([](git_rebase *aRebase) { git_rebase_free(aRebase); })> GitRebasePtr
A unique pointer for git_rebase objects with automatic cleanup.
std::unique_ptr< git_index, decltype([](git_index *aIndex) { git_index_free(aIndex); })> GitIndexPtr
A unique pointer for git_index objects with automatic cleanup.
std::unique_ptr< git_object, decltype([](git_object *aObject) { git_object_free(aObject); })> GitObjectPtr
A unique pointer for git_object objects with automatic cleanup.
std::unique_ptr< git_remote, decltype([](git_remote *aRemote) { git_remote_free(aRemote); })> GitRemotePtr
A unique pointer for git_remote objects with automatic cleanup.
KIGIT_COMMON::GIT_CONN_TYPE connType
wxString result
Test unit parsing edge cases and error handling.
wxLogTrace helper definitions.