34#include <wx/filename.h>
36#include <wx/textfile.h>
42 m_repo( aRepo ), m_connType(
GIT_CONN_TYPE::GIT_CONN_LOCAL ), m_testedTypes( 0 ),
48 m_repo( aOther.m_repo ),
49 m_connType( aOther.m_connType ),
50 m_remote( aOther.m_remote ),
51 m_hostname( aOther.m_hostname ),
52 m_username( aOther.m_username ),
53 m_password( aOther.m_password ),
54 m_testedTypes( aOther.m_testedTypes ),
57 m_publicKeys( aOther.m_publicKeys ),
58 m_nextPublicKey( aOther.m_nextPublicKey )
73 wxCHECK(
m_repo, wxEmptyString );
74 git_reference* head =
nullptr;
76 int retval = git_repository_head( &head,
m_repo );
78 if( retval && retval != GIT_EUNBORNBRANCH && retval != GIT_ENOTFOUND )
82 git_reference* branch;
84 if( git_reference_resolve( &branch, head ) )
86 wxLogTrace(
traceGit,
"Failed to resolve branch" );
91 const char* branchName =
"";
93 if( git_branch_name( &branchName, branch ) )
95 wxLogTrace(
traceGit,
"Failed to get branch name" );
99 return wxString( branchName );
108 std::vector<wxString> branchNames;
109 std::map<git_time_t, wxString> branchNamesMap;
112 git_branch_iterator* branchIterator =
nullptr;
114 if( git_branch_iterator_new( &branchIterator,
m_repo, GIT_BRANCH_LOCAL ) )
116 wxLogTrace(
traceGit,
"Failed to get branch iterator" );
121 git_reference* branchReference =
nullptr;
122 git_branch_t branchType;
124 while( git_branch_next( &branchReference, &branchType, branchIterator ) != GIT_ITEROVER )
126 const char* branchName =
"";
129 if( git_branch_name( &branchName, branchReference ) )
131 wxLogTrace(
traceGit,
"Failed to get branch name in iter loop" );
135 const git_oid* commitId = git_reference_target( branchReference );
137 git_commit* commit =
nullptr;
139 if( git_commit_lookup( &commit,
m_repo, commitId ) )
141 wxLogTrace(
traceGit,
"Failed to get commit in iter loop" );
146 git_time_t commitTime = git_commit_time( commit );
148 if( git_branch_is_head( branchReference ) )
149 firstName = branchName;
151 branchNamesMap.emplace( commitTime, branchName );
155 if( !firstName.IsEmpty() )
156 branchNames.push_back( firstName );
159 for(
auto rit = branchNamesMap.rbegin(); rit != branchNamesMap.rend(); ++rit )
160 branchNames.push_back( rit->second );
169 std::vector<wxString> projDirs;
175 if( git_reference_name_to_id( &oid,
m_repo,
"HEAD" ) != GIT_OK )
181 if( git_commit_lookup( &commit,
m_repo, &oid ) != GIT_OK )
189 if( git_commit_tree( &tree, commit ) != GIT_OK )
199 tree, GIT_TREEWALK_PRE,
200 [](
const char* root,
const git_tree_entry* entry,
void* payload )
202 std::vector<wxString>* prjs =
static_cast<std::vector<wxString>*
>( payload );
203 wxFileName root_fn( git_tree_entry_name( entry ) );
205 root_fn.SetPath( root );
207 if( git_tree_entry_type( entry ) == GIT_OBJECT_BLOB
208 && ( ( root_fn.GetExt() ==
"kicad_pro" ) || ( root_fn.GetExt() ==
"pro" ) ) )
210 prjs->push_back( root_fn.GetFullPath() );
217 std::sort( projDirs.begin(), projDirs.end(),
218 [](
const wxString& a,
const wxString& b )
220 int a_freq = a.Freq( wxFileName::GetPathSeparator() );
221 int b_freq = b.Freq( wxFileName::GetPathSeparator() );
223 if( a_freq == b_freq )
226 return a_freq < b_freq;
236 auto get_modified_files = [&](
const git_oid* from_oid,
const git_oid* to_oid ) -> std::set<wxString>
238 std::set<wxString> modified_set;
239 git_revwalk* walker =
nullptr;
241 if( !
m_repo || !from_oid )
244 if( git_revwalk_new( &walker,
m_repo ) != GIT_OK )
252 if( git_revwalk_push( walker, from_oid ) != GIT_OK )
258 if( to_oid && git_revwalk_hide( walker, to_oid ) != GIT_OK )
267 while( git_revwalk_next( &oid, walker ) == GIT_OK )
269 if( git_commit_lookup( &commit,
m_repo, &oid ) != GIT_OK )
276 git_tree *tree, *parent_tree =
nullptr;
278 if( git_commit_tree( &tree, commit ) != GIT_OK )
287 if( !git_commit_parentcount( commit ) )
290 tree, GIT_TREEWALK_PRE,
291 [](
const char* root,
const git_tree_entry* entry,
void* payload )
293 std::set<wxString>* modified_set_internal =
static_cast<std::set<wxString>*
>( payload );
294 wxString filePath = wxString::Format(
"%s%s", root, git_tree_entry_name( entry ) );
295 modified_set_internal->insert( filePath );
304 if( git_commit_parent( &parent, commit, 0 ) != GIT_OK )
313 if( git_commit_tree( &parent_tree, parent ) != GIT_OK )
323 git_diff_options diff_opts;
324 git_diff_init_options( &diff_opts, GIT_DIFF_OPTIONS_VERSION );
326 if( !
m_repo || !parent_tree || !tree )
329 if( git_diff_tree_to_tree( &diff,
m_repo, parent_tree, tree, &diff_opts ) == GIT_OK )
331 size_t num_deltas = git_diff_num_deltas( diff );
333 for(
size_t i = 0; i < num_deltas; ++i )
335 const git_diff_delta*
delta = git_diff_get_delta( diff, i );
336 modified_set.insert(
delta->new_file.path );
339 git_diff_free( diff );
352 std::pair<std::set<wxString>,std::set<wxString>> modified_files;
355 return modified_files;
357 git_reference* head =
nullptr;
358 git_reference* remote_head =
nullptr;
360 if( git_repository_head( &head,
m_repo ) != GIT_OK )
362 wxLogTrace(
traceGit,
"Failed to get modified HEAD" );
363 return modified_files;
368 if( git_branch_upstream( &remote_head, head ) != GIT_OK )
370 wxLogTrace(
traceGit,
"Failed to get modified remote HEAD" );
375 if( remote_head !=
nullptr && head !=
nullptr )
377 const git_oid* head_oid = git_reference_target( head );
378 const git_oid* remote_oid = git_reference_target( remote_head );
380 modified_files.first = get_modified_files( head_oid, remote_oid );
381 modified_files.second = get_modified_files( remote_oid, head_oid );
384 return modified_files;
393 git_reference* head =
nullptr;
394 git_reference* remote_head =
nullptr;
396 if( git_repository_head( &head,
m_repo ) != GIT_OK )
404 if( git_branch_upstream( &remote_head, head ) != GIT_OK )
412 const git_oid* head_oid = git_reference_target( head );
413 const git_oid* remote_oid = git_reference_target( remote_head );
414 git_revwalk* walker =
nullptr;
416 if( git_revwalk_new( &walker,
m_repo ) != GIT_OK )
424 if( !head_oid || git_revwalk_push( walker, head_oid ) != GIT_OK )
430 if( remote_oid && git_revwalk_hide( walker, remote_oid ) != GIT_OK )
439 if( git_revwalk_next( &oid, walker ) != GIT_OK )
452 git_remote* remote =
nullptr;
454 if( git_remote_lookup( &remote,
m_repo,
"origin" ) != GIT_OK )
456 wxLogTrace(
traceGit,
"Failed to get remote for haspushpull" );
463 const char* fetch_url = git_remote_url( remote );
464 const char* push_url = git_remote_pushurl( remote );
469 wxLogTrace(
traceGit,
"No push URL set, using fetch URL" );
470 push_url = fetch_url;
473 return fetch_url && push_url;
479 wxCHECK(
m_repo, wxEmptyString );
482 git_reference* head =
nullptr;
483 git_reference* upstream =
nullptr;
485 if( git_repository_head( &head,
m_repo ) != GIT_OK )
493 if( git_branch_upstream( &upstream, head ) != GIT_OK )
496 git_strarray remotes = {
nullptr, 0 };
498 if( git_remote_list( &remotes,
m_repo ) == GIT_OK )
500 if( remotes.count == 1 )
501 retval = remotes.strings[0];
503 git_strarray_dispose( &remotes );
512 git_remote* remote =
nullptr;
514 if( git_remote_lookup( &remote,
m_repo,
"origin" ) == GIT_OK )
516 retval = git_remote_name( remote );
517 git_remote_free( remote );
521 wxLogTrace(
traceGit,
"Failed to get remote name from default remote: %s",
530 git_buf remote_name = GIT_BUF_INIT_CONST(
nullptr, 0 );
532 if( git_branch_remote_name( &remote_name,
m_repo, git_reference_name( upstream ) ) == GIT_OK )
534 retval = remote_name.ptr;
535 git_buf_dispose( &remote_name );
540 "Failed to get remote name from upstream branch: %s",
561 return wxEmptyString;
563 const char *
path = git_repository_path(
m_repo );
564 wxString retval =
path;
573 wxFileName keyFile( wxGetHomeDir(), wxEmptyString );
574 keyFile.AppendDir(
".ssh" );
575 keyFile.SetFullName(
"id_rsa" );
577 if( keyFile.FileExists() )
580 keyFile.SetFullName(
"id_dsa" );
582 if( keyFile.FileExists() )
585 keyFile.SetFullName(
"id_ecdsa" );
587 if( keyFile.FileExists() )
590 keyFile.SetFullName(
"id_ed25519" );
592 if( keyFile.FileExists() )
596 wxFileName sshConfig( wxGetHomeDir(), wxEmptyString );
597 sshConfig.AppendDir(
".ssh" );
598 sshConfig.SetFullName(
"config" );
600 if( sshConfig.FileExists() )
602 wxTextFile configFile( sshConfig.GetFullPath() );
607 for( wxString line = configFile.GetFirstLine(); !configFile.Eof(); line = configFile.GetNextLine() )
609 line.Trim(
false ).Trim(
true );
611 if( line.StartsWith(
"Host " ) )
616 if( line.StartsWith(
"Host" ) && line.Contains(
m_hostname ) )
619 if( match && line.StartsWith(
"IdentityFile" ) )
621 wxString keyPath = line.AfterFirst(
' ' ).Trim(
false ).Trim(
true );
624 if( keyPath.StartsWith(
"~" ) )
625 keyPath.Replace(
"~", wxGetHomeDir(),
false );
628 if( wxFileName::FileExists( keyPath ) )
646 git_remote* remote =
nullptr;
648 if( git_remote_lookup( &remote,
m_repo, remote_name.ToStdString().c_str() ) == GIT_OK )
650 const char* url = git_remote_url( remote );
655 git_remote_free( remote );
669 if( remote.IsEmpty() )
672 if( remote.StartsWith(
"https://" ) || remote.StartsWith(
"http://" ) )
676 else if( remote.StartsWith(
"ssh://" ) || remote.StartsWith(
"git@" ) || remote.StartsWith(
"git+ssh://" )
677 || remote.EndsWith(
".git" ) )
697 size_t atPos = uri.find(
'@' );
699 if( atPos != wxString::npos )
701 size_t protoEnd = uri.find(
"//" );
703 if( protoEnd != wxString::npos )
705 wxString credentials = uri.Mid( protoEnd + 2, atPos - protoEnd - 2 );
706 size_t colonPos = credentials.find(
':' );
708 if( colonPos != wxString::npos )
711 m_password = credentials.Mid( colonPos + 1, credentials.Length() - colonPos - 1 );
727 size_t colonPos =
m_remote.find(
':' );
728 if( colonPos != wxString::npos )
734 size_t hostStart =
m_remote.find(
"://" ) + 2;
735 size_t hostEnd =
m_remote.find(
'/', hostStart );
738 if( hostEnd != wxString::npos )
739 host =
m_remote.Mid( hostStart, hostEnd - hostStart );
743 atPos = host.find(
'@' );
745 if( atPos != wxString::npos )
762 if( sshKey.IsEmpty() )
764 wxLogTrace(
traceGit,
"Finished testing all possible ssh keys" );
766 return GIT_PASSTHROUGH;
769 wxString sshPubKey = sshKey +
".pub";
772 wxLogTrace(
traceGit,
"Testing %s\n", sshKey );
774 if( git_credential_ssh_key_new( aOut, aUsername.mbc_str(), sshPubKey.mbc_str(), sshKey.mbc_str(),
775 password.mbc_str() ) != GIT_OK )
777 wxLogTrace(
traceGit,
"Failed to create SSH key credential for %s: %s",
779 return GIT_PASSTHROUGH;
790 git_credential_userpass_plaintext_new( aOut, aUsername.mbc_str(), password.mbc_str() );
798 if( git_credential_ssh_key_from_agent( aOut, aUsername.mbc_str() ) != GIT_OK )
800 wxLogTrace(
traceGit,
"Failed to create SSH agent credential for %s: %s",
802 return GIT_PASSTHROUGH;
811 const git_oid* aOID,
unsigned int aIsMerge,
void* aPayload )
814 git_oid_cpy( (git_oid*) aPayload, aOID );
820extern "C" void clone_progress_cb(
const char* aStr,
size_t aLen,
size_t aTotal,
void* aPayload )
824 wxString progressMessage( aStr );
829extern "C" int progress_cb(
const char* str,
int len,
void* aPayload )
833 wxString progressMessage( str, len );
844 wxString progressMessage = wxString::Format(
_(
"Received %u of %u objects" ),
845 aStats->received_objects,
846 aStats->total_objects );
848 parent->
UpdateProgress( aStats->received_objects, aStats->total_objects, progressMessage );
854extern "C" int update_cb(
const char* aRefname,
const git_oid* aFirst,
const git_oid* aSecond,
857 constexpr int cstring_len = 8;
858 char a_str[cstring_len + 1];
859 char b_str[cstring_len + 1];
864 git_oid_tostr( b_str, cstring_len, aSecond );
866#if ( LIBGIT2_VER_MAJOR >= 1 ) || ( LIBGIT2_VER_MINOR >= 99 )
867 if( !git_oid_is_zero( aFirst ) )
869 if( !git_oid_iszero( aFirst ) )
872 git_oid_tostr( a_str, cstring_len, aFirst );
873 status = wxString::Format(
_(
"* [updated] %s..%s %s" ), a_str, b_str, aRefname );
877 status = wxString::Format(
_(
"* [new] %s %s" ), b_str, aRefname );
889 long long progress = 100;
894 progress = ( aCurrent * 100ll ) / aTotal;
897 wxString progressMessage = wxString::Format(
_(
"Writing objects: %lld%% (%u/%u), %zu bytes" ),
898 progress, aCurrent, aTotal, aBytes );
908 wxString status( aStatus );
910 if( !status.IsEmpty() )
912 wxString statusMessage = wxString::Format(
_(
"* [rejected] %s (%s)" ), aRefname, aStatus );
917 wxString statusMessage = wxString::Format(
_(
"[updated] %s" ), aRefname );
925extern "C" int credentials_cb( git_cred** aOut,
const char* aUrl,
const char* aUsername,
926 unsigned int aAllowedTypes,
void* aPayload )
931 wxLogTrace(
traceGit,
"Credentials callback for %s, testing %d", aUrl, aAllowedTypes );
935 wxLogTrace(
traceGit,
"Local repository, no credentials needed" );
936 return GIT_PASSTHROUGH;
939 if( aAllowedTypes & GIT_CREDENTIAL_USERNAME
940 && !( parent->
TestedTypes() & GIT_CREDENTIAL_USERNAME ) )
942 wxString username = parent->
GetUsername().Trim().Trim(
false );
943 wxLogTrace(
traceGit,
"Username credential for %s at %s with allowed type %d",
944 username, aUrl, aAllowedTypes );
945 if( git_credential_username_new( aOut, username.ToStdString().c_str() ) != GIT_OK )
947 wxLogTrace(
traceGit,
"Failed to create username credential for %s: %s",
952 wxLogTrace(
traceGit,
"Created username credential for %s", username );
958 && ( aAllowedTypes & GIT_CREDENTIAL_USERPASS_PLAINTEXT )
959 && !( parent->
TestedTypes() & GIT_CREDENTIAL_USERPASS_PLAINTEXT ) )
962 wxLogTrace(
traceGit,
"Plaintext authentication for %s at %s with allowed type %d",
967 && ( aAllowedTypes & GIT_CREDENTIAL_SSH_KEY )
968 && !( parent->
TestedTypes() & GIT_CREDENTIAL_SSH_KEY ) )
978 return GIT_PASSTHROUGH;
981 git_error_set_str( GIT_ERROR_NET,
_(
"Unable to authenticate" ).mbc_str() );
virtual void UpdateProgress(int aCurrent, int aTotal, const wxString &aMessage)
std::vector< wxString > GetBranchNames() const
void updateConnectionType()
GIT_CONN_TYPE GetConnType() const
static wxString GetLastGitError()
wxString GetCurrentBranchName() const
wxString GetGitRootDirectory() const
void SetSSHKey(const wxString &aSSHKey)
static const unsigned KIGIT_CREDENTIAL_SSH_AGENT
int HandlePlaintextAuthentication(git_cred **aOut, const wxString &aUsername)
std::vector< wxString > GetProjectDirs()
Return a vector of project files in the repository.
wxString GetPassword() const
git_repository * GetRepo() const
wxString GetNextPublicKey()
KIGIT_COMMON(git_repository *aRepo)
std::pair< std::set< wxString >, std::set< wxString > > GetDifferentFiles() const
Return a pair of sets of files that differ locally from the remote repository The first set is files ...
std::vector< wxString > m_publicKeys
bool HasPushAndPullRemote() const
void UpdateCurrentBranchInfo()
bool HasLocalCommits() const
int HandleSSHAgentAuthentication(git_cred **aOut, const wxString &aUsername)
wxString GetRemotename() const
int HandleSSHKeyAuthentication(git_cred **aOut, const wxString &aUsername)
wxString GetUsername() const
Get the username.
unsigned & TestedTypes()
Return the connection types that have been tested for authentication.
KIGIT_COMMON * GetCommon() const
Get the common object.
KIGIT_COMMON::GIT_CONN_TYPE GetConnType() const
Get the connection type.
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 push_update_reference_cb(const char *aRefname, const char *aStatus, 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)
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_branch_iterator, decltype([](git_branch_iterator *aIter) { git_branch_iterator_free(aIter) GitBranchIteratorPtr
A unique pointer for git_branch_iterator objects with automatic cleanup.
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_reference, decltype([](git_reference *aRef) { git_reference_free(aRef) GitReferencePtr
A unique pointer for git_reference 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_remote, decltype([](git_remote *aRemote) { git_remote_free(aRemote) GitRemotePtr
A unique pointer for git_remote objects with automatic cleanup.
wxLogTrace helper definitions.