KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
kicad_git_common.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.TXT for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 3
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/gpl-3.0.html
19 * or you may search the http://www.gnu.org website for the version 3 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24#include "kicad_git_common.h"
25#include "kicad_git_memory.h"
26#include "git_repo_mixin.h"
27
28#include <git/git_progress.h>
30#include <kiplatform/secrets.h>
31#include <trace_helpers.h>
32
33#include <git2.h>
34#include <wx/filename.h>
35#include <wx/log.h>
36#include <wx/textfile.h>
37#include <wx/utils.h>
38#include <map>
39#include <vector>
40
41KIGIT_COMMON::KIGIT_COMMON( git_repository* aRepo ) :
42 m_repo( aRepo ), m_connType( GIT_CONN_TYPE::GIT_CONN_LOCAL ), m_testedTypes( 0 ),
43 m_nextPublicKey( 0 )
44{}
45
47 // Initialize base class and member variables
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 ),
55 // The mutex is default-initialized, not copied
56 m_gitActionMutex(),
57 m_publicKeys( aOther.m_publicKeys ),
58 m_nextPublicKey( aOther.m_nextPublicKey )
59{
60}
61
63{}
64
65
66git_repository* KIGIT_COMMON::GetRepo() const
67{
68 return m_repo;
69}
70
72{
73 wxCHECK( m_repo, wxEmptyString );
74 git_reference* head = nullptr;
75
76 int retval = git_repository_head( &head, m_repo );
77
78 if( retval && retval != GIT_EUNBORNBRANCH && retval != GIT_ENOTFOUND )
79 return wxEmptyString;
80
81 KIGIT::GitReferencePtr headPtr( head );
82 git_reference* branch;
83
84 if( git_reference_resolve( &branch, head ) )
85 {
86 wxLogTrace( traceGit, "Failed to resolve branch" );
87 return wxEmptyString;
88 }
89
90 KIGIT::GitReferencePtr branchPtr( branch );
91 const char* branchName = "";
92
93 if( git_branch_name( &branchName, branch ) )
94 {
95 wxLogTrace( traceGit, "Failed to get branch name" );
96 return wxEmptyString;
97 }
98
99 return wxString( branchName );
100}
101
102
103std::vector<wxString> KIGIT_COMMON::GetBranchNames() const
104{
105 if( !m_repo )
106 return {};
107
108 std::vector<wxString> branchNames;
109 std::map<git_time_t, wxString> branchNamesMap;
110 wxString firstName;
111
112 git_branch_iterator* branchIterator = nullptr;
113
114 if( git_branch_iterator_new( &branchIterator, m_repo, GIT_BRANCH_LOCAL ) )
115 {
116 wxLogTrace( traceGit, "Failed to get branch iterator" );
117 return branchNames;
118 }
119
120 KIGIT::GitBranchIteratorPtr branchIteratorPtr( branchIterator );
121 git_reference* branchReference = nullptr;
122 git_branch_t branchType;
123
124 while( git_branch_next( &branchReference, &branchType, branchIterator ) != GIT_ITEROVER )
125 {
126 const char* branchName = "";
127 KIGIT::GitReferencePtr branchReferencePtr( branchReference );
128
129 if( git_branch_name( &branchName, branchReference ) )
130 {
131 wxLogTrace( traceGit, "Failed to get branch name in iter loop" );
132 continue;
133 }
134
135 const git_oid* commitId = git_reference_target( branchReference );
136
137 git_commit* commit = nullptr;
138
139 if( git_commit_lookup( &commit, m_repo, commitId ) )
140 {
141 wxLogTrace( traceGit, "Failed to get commit in iter loop" );
142 continue;
143 }
144
145 KIGIT::GitCommitPtr commitPtr( commit );
146 git_time_t commitTime = git_commit_time( commit );
147
148 if( git_branch_is_head( branchReference ) )
149 firstName = branchName;
150 else
151 branchNamesMap.emplace( commitTime, branchName );
152 }
153
154 // Add the current branch to the top of the list
155 if( !firstName.IsEmpty() )
156 branchNames.push_back( firstName );
157
158 // Add the remaining branches in order from newest to oldest
159 for( auto rit = branchNamesMap.rbegin(); rit != branchNamesMap.rend(); ++rit )
160 branchNames.push_back( rit->second );
161
162 return branchNames;
163}
164
165
166std::vector<wxString> KIGIT_COMMON::GetProjectDirs()
167{
168 wxCHECK( m_repo, {} );
169 std::vector<wxString> projDirs;
170
171 git_oid oid;
172 git_commit* commit;
173 git_tree *tree;
174
175 if( git_reference_name_to_id( &oid, m_repo, "HEAD" ) != GIT_OK )
176 {
177 wxLogTrace( traceGit, "An error occurred: %s", KIGIT_COMMON::GetLastGitError() );
178 return projDirs;
179 }
180
181 if( git_commit_lookup( &commit, m_repo, &oid ) != GIT_OK )
182 {
183 wxLogTrace( traceGit, "An error occurred: %s", KIGIT_COMMON::GetLastGitError() );
184 return projDirs;
185 }
186
187 KIGIT::GitCommitPtr commitPtr( commit );
188
189 if( git_commit_tree( &tree, commit ) != GIT_OK )
190 {
191 wxLogTrace( traceGit, "An error occurred: %s", KIGIT_COMMON::GetLastGitError() );
192 return projDirs;
193 }
194
195 KIGIT::GitTreePtr treePtr( tree );
196
197 // Define callback
198 git_tree_walk(
199 tree, GIT_TREEWALK_PRE,
200 []( const char* root, const git_tree_entry* entry, void* payload )
201 {
202 std::vector<wxString>* prjs = static_cast<std::vector<wxString>*>( payload );
203 wxFileName root_fn( git_tree_entry_name( entry ) );
204
205 root_fn.SetPath( root );
206
207 if( git_tree_entry_type( entry ) == GIT_OBJECT_BLOB
208 && ( ( root_fn.GetExt() == "kicad_pro" ) || ( root_fn.GetExt() == "pro" ) ) )
209 {
210 prjs->push_back( root_fn.GetFullPath() );
211 }
212
213 return 0; // continue walking
214 },
215 &projDirs );
216
217 std::sort( projDirs.begin(), projDirs.end(),
218 []( const wxString& a, const wxString& b )
219 {
220 int a_freq = a.Freq( wxFileName::GetPathSeparator() );
221 int b_freq = b.Freq( wxFileName::GetPathSeparator() );
222
223 if( a_freq == b_freq )
224 return a < b;
225 else
226 return a_freq < b_freq;
227
228 } );
229
230 return projDirs;
231}
232
233
234std::pair<std::set<wxString>,std::set<wxString>> KIGIT_COMMON::GetDifferentFiles() const
235{
236 auto get_modified_files = [&]( const git_oid* from_oid, const git_oid* to_oid ) -> std::set<wxString>
237 {
238 std::set<wxString> modified_set;
239 git_revwalk* walker = nullptr;
240
241 if( !m_repo || !from_oid )
242 return modified_set;
243
244 if( git_revwalk_new( &walker, m_repo ) != GIT_OK )
245 {
246 wxLogTrace( traceGit, "Failed to create revwalker: %s", KIGIT_COMMON::GetLastGitError() );
247 return modified_set;
248 }
249
250 KIGIT::GitRevWalkPtr walkerPtr( walker );
251
252 if( git_revwalk_push( walker, from_oid ) != GIT_OK )
253 {
254 wxLogTrace( traceGit, "Failed to set from commit: %s", KIGIT_COMMON::GetLastGitError() );
255 return modified_set;
256 }
257
258 if( to_oid && git_revwalk_hide( walker, to_oid ) != GIT_OK )
259 {
260 wxLogTrace( traceGit, "Failed to set end commit (maybe new repo): %s", KIGIT_COMMON::GetLastGitError() );
261 }
262
263 git_oid oid;
264 git_commit* commit;
265
266 // iterate over all local commits not in remote
267 while( git_revwalk_next( &oid, walker ) == GIT_OK )
268 {
269 if( git_commit_lookup( &commit, m_repo, &oid ) != GIT_OK )
270 {
271 wxLogTrace( traceGit, "Failed to lookup commit: %s", KIGIT_COMMON::GetLastGitError() );
272 continue;
273 }
274
275 KIGIT::GitCommitPtr commitPtr( commit );
276 git_tree *tree, *parent_tree = nullptr;
277
278 if( git_commit_tree( &tree, commit ) != GIT_OK )
279 {
280 wxLogTrace( traceGit, "Failed to get commit tree: %s", KIGIT_COMMON::GetLastGitError() );
281 continue;
282 }
283
284 KIGIT::GitTreePtr treePtr( tree );
285
286 // get parent commit tree to diff against
287 if( !git_commit_parentcount( commit ) )
288 {
289 git_tree_walk(
290 tree, GIT_TREEWALK_PRE,
291 []( const char* root, const git_tree_entry* entry, void* payload )
292 {
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 );
296 return 0; // continue walking
297 },
298 &modified_set );
299 continue;
300 }
301
302 git_commit* parent;
303
304 if( git_commit_parent( &parent, commit, 0 ) != GIT_OK )
305 {
306 wxLogTrace( traceGit, "Failed to get parent commit: %s", KIGIT_COMMON::GetLastGitError() );
307 continue;
308 }
309
310 KIGIT::GitCommitPtr parentPtr( parent );
311
312
313 if( git_commit_tree( &parent_tree, parent ) != GIT_OK )
314 {
315 wxLogTrace( traceGit, "Failed to get parent commit tree: %s", KIGIT_COMMON::GetLastGitError() );
316 continue;
317 }
318
319 KIGIT::GitTreePtr parentTreePtr( parent_tree );
320
321
322 git_diff* diff;
323 git_diff_options diff_opts;
324 git_diff_init_options( &diff_opts, GIT_DIFF_OPTIONS_VERSION );
325
326 if( !m_repo || !parent_tree || !tree )
327 continue;
328
329 if( git_diff_tree_to_tree( &diff, m_repo, parent_tree, tree, &diff_opts ) == GIT_OK )
330 {
331 size_t num_deltas = git_diff_num_deltas( diff );
332
333 for( size_t i = 0; i < num_deltas; ++i )
334 {
335 const git_diff_delta* delta = git_diff_get_delta( diff, i );
336 modified_set.insert( delta->new_file.path );
337 }
338
339 git_diff_free( diff );
340 }
341 else
342 {
343 wxLogTrace( traceGit, "Failed to diff trees: %s", KIGIT_COMMON::GetLastGitError() );
344 }
345 }
346
347 wxLogTrace( traceGit, "Finished walking commits with end: %s", KIGIT_COMMON::GetLastGitError() );
348
349 return modified_set;
350 };
351
352 std::pair<std::set<wxString>,std::set<wxString>> modified_files;
353
354 if( !m_repo )
355 return modified_files;
356
357 git_reference* head = nullptr;
358 git_reference* remote_head = nullptr;
359
360 if( git_repository_head( &head, m_repo ) != GIT_OK )
361 {
362 wxLogTrace( traceGit, "Failed to get modified HEAD" );
363 return modified_files;
364 }
365
366 KIGIT::GitReferencePtr headPtr( head );
367
368 if( git_branch_upstream( &remote_head, head ) != GIT_OK )
369 {
370 wxLogTrace( traceGit, "Failed to get modified remote HEAD" );
371 }
372
373 KIGIT::GitReferencePtr remoteHeadPtr( remote_head );
374
375 if( remote_head != nullptr && head != nullptr )
376 {
377 const git_oid* head_oid = git_reference_target( head );
378 const git_oid* remote_oid = git_reference_target( remote_head );
379
380 modified_files.first = get_modified_files( head_oid, remote_oid );
381 modified_files.second = get_modified_files( remote_oid, head_oid );
382 }
383
384 return modified_files;
385}
386
387
389{
390 if( !m_repo )
391 return false;
392
393 git_reference* head = nullptr;
394 git_reference* remote_head = nullptr;
395
396 if( git_repository_head( &head, m_repo ) != GIT_OK )
397 {
398 wxLogTrace( traceGit, "Failed to get HEAD: %s", KIGIT_COMMON::GetLastGitError() );
399 return false;
400 }
401
402 KIGIT::GitReferencePtr headPtr( head );
403
404 if( git_branch_upstream( &remote_head, head ) != GIT_OK )
405 {
406 // No remote branch, so we have local commits (new repo?)
407 wxLogTrace( traceGit, "Failed to get remote HEAD: %s", KIGIT_COMMON::GetLastGitError() );
408 return true;
409 }
410
411 KIGIT::GitReferencePtr remoteHeadPtr( remote_head );
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;
415
416 if( git_revwalk_new( &walker, m_repo ) != GIT_OK )
417 {
418 wxLogTrace( traceGit, "Failed to create revwalker: %s", KIGIT_COMMON::GetLastGitError() );
419 return false;
420 }
421
422 KIGIT::GitRevWalkPtr walkerPtr( walker );
423
424 if( !head_oid || git_revwalk_push( walker, head_oid ) != GIT_OK )
425 {
426 wxLogTrace( traceGit, "Failed to push commits: %s", KIGIT_COMMON::GetLastGitError() );
427 return false;
428 }
429
430 if( remote_oid && git_revwalk_hide( walker, remote_oid ) != GIT_OK )
431 {
432 wxLogTrace( traceGit, "Failed to push/hide commits: %s", KIGIT_COMMON::GetLastGitError() );
433 return false;
434 }
435
436 git_oid oid;
437
438 // If we can't walk to the next commit, then we are at or behind the remote
439 if( git_revwalk_next( &oid, walker ) != GIT_OK )
440 {
441 wxLogTrace( traceGit, "Failed to walk to next commit: %s", KIGIT_COMMON::GetLastGitError() );
442 return false;
443 }
444
445 return true;
446}
447
448
450{
451 wxCHECK( m_repo, false );
452 git_remote* remote = nullptr;
453
454 if( git_remote_lookup( &remote, m_repo, "origin" ) != GIT_OK )
455 {
456 wxLogTrace( traceGit, "Failed to get remote for haspushpull" );
457 return false;
458 }
459
460 KIGIT::GitRemotePtr remotePtr( remote );
461
462 // Get the URLs associated with the remote
463 const char* fetch_url = git_remote_url( remote );
464 const char* push_url = git_remote_pushurl( remote );
465
466 // If no push URL is set, libgit2 defaults to using the fetch URL for pushing
467 if( !push_url )
468 {
469 wxLogTrace( traceGit, "No push URL set, using fetch URL" );
470 push_url = fetch_url;
471 }
472
473 return fetch_url && push_url;
474}
475
476
478{
479 wxCHECK( m_repo, wxEmptyString );
480
481 wxString retval;
482 git_reference* head = nullptr;
483 git_reference* upstream = nullptr;
484
485 if( git_repository_head( &head, m_repo ) != GIT_OK )
486 {
487 wxLogTrace( traceGit, "Failed to get remote name: %s", KIGIT_COMMON::GetLastGitError() );
488 return retval;
489 }
490
491 KIGIT::GitReferencePtr headPtr( head );
492
493 if( git_branch_upstream( &upstream, head ) != GIT_OK )
494 {
495 wxLogTrace( traceGit, "Failed to get upstream branch: %s", KIGIT_COMMON::GetLastGitError() );
496 git_strarray remotes = { nullptr, 0 };
497
498 if( git_remote_list( &remotes, m_repo ) == GIT_OK )
499 {
500 if( remotes.count == 1 )
501 retval = remotes.strings[0];
502
503 git_strarray_dispose( &remotes );
504 }
505 else
506 {
507 wxLogTrace( traceGit, "Failed to list remotes: %s", KIGIT_COMMON::GetLastGitError() );
508
509 // If we can't get the remote name from the upstream branch or the list of remotes,
510 // just return the default remote name
511
512 git_remote* remote = nullptr;
513
514 if( git_remote_lookup( &remote, m_repo, "origin" ) == GIT_OK )
515 {
516 retval = git_remote_name( remote );
517 git_remote_free( remote );
518 }
519 else
520 {
521 wxLogTrace( traceGit, "Failed to get remote name from default remote: %s",
523 }
524 }
525
526 return retval;
527 }
528
529 KIGIT::GitReferencePtr upstreamPtr( upstream );
530 git_buf remote_name = GIT_BUF_INIT_CONST( nullptr, 0 );
531
532 if( git_branch_remote_name( &remote_name, m_repo, git_reference_name( upstream ) ) == GIT_OK )
533 {
534 retval = remote_name.ptr;
535 git_buf_dispose( &remote_name );
536 }
537 else
538 {
539 wxLogTrace( traceGit,
540 "Failed to get remote name from upstream branch: %s",
542 }
543
544 return retval;
545}
546
547void KIGIT_COMMON::SetSSHKey( const wxString& aKey )
548{
549 auto it = std::find( m_publicKeys.begin(), m_publicKeys.end(), aKey );
550
551 if( it != m_publicKeys.end() )
552 m_publicKeys.erase( it );
553
554 m_publicKeys.insert( m_publicKeys.begin(), aKey );
555}
556
557
559{
560 if( !m_repo )
561 return wxEmptyString;
562
563 const char *path = git_repository_path( m_repo );
564 wxString retval = path;
565 return retval;
566}
567
568
570{
571 m_publicKeys.clear();
572
573 wxFileName keyFile( wxGetHomeDir(), wxEmptyString );
574 keyFile.AppendDir( ".ssh" );
575 keyFile.SetFullName( "id_rsa" );
576
577 if( keyFile.FileExists() )
578 m_publicKeys.push_back( keyFile.GetFullPath() );
579
580 keyFile.SetFullName( "id_dsa" );
581
582 if( keyFile.FileExists() )
583 m_publicKeys.push_back( keyFile.GetFullPath() );
584
585 keyFile.SetFullName( "id_ecdsa" );
586
587 if( keyFile.FileExists() )
588 m_publicKeys.push_back( keyFile.GetFullPath() );
589
590 keyFile.SetFullName( "id_ed25519" );
591
592 if( keyFile.FileExists() )
593 m_publicKeys.push_back( keyFile.GetFullPath() );
594
595 // Parse SSH config file for hostname information
596 wxFileName sshConfig( wxGetHomeDir(), wxEmptyString );
597 sshConfig.AppendDir( ".ssh" );
598 sshConfig.SetFullName( "config" );
599
600 if( sshConfig.FileExists() )
601 {
602 wxTextFile configFile( sshConfig.GetFullPath() );
603 configFile.Open();
604
605 bool match = false;
606
607 for( wxString line = configFile.GetFirstLine(); !configFile.Eof(); line = configFile.GetNextLine() )
608 {
609 line.Trim( false ).Trim( true );
610
611 if( line.StartsWith( "Host " ) )
612 match = false;
613
614 // The difference here is that we are matching either "Hostname" or "Host" to get the
615 // match. This is because in the absence of a "Hostname" line, the "Host" line is used
616 if( line.StartsWith( "Host" ) && line.Contains( m_hostname ) )
617 match = true;
618
619 if( match && line.StartsWith( "IdentityFile" ) )
620 {
621 wxString keyPath = line.AfterFirst( ' ' ).Trim( false ).Trim( true );
622
623 // Expand ~ to home directory if present
624 if( keyPath.StartsWith( "~" ) )
625 keyPath.Replace( "~", wxGetHomeDir(), false );
626
627 // Add the public key to the beginning of the list
628 if( wxFileName::FileExists( keyPath ) )
629 SetSSHKey( keyPath );
630 }
631 }
632
633 configFile.Close();
634 }
635}
636
637
639{
640 wxCHECK( m_repo, /* void */ );
641
642 // We want to get the current branch's upstream url as well as the stored password
643 // if one exists given the url and username.
644
645 wxString remote_name = GetRemotename();
646 git_remote* remote = nullptr;
647
648 if( git_remote_lookup( &remote, m_repo, remote_name.ToStdString().c_str() ) == GIT_OK )
649 {
650 const char* url = git_remote_url( remote );
651
652 if( url )
653 m_remote = url;
654
655 git_remote_free( remote );
656 }
657
658 // Find the stored password if it exists
660
663}
664
666{
667 wxString remote = m_remote;
668
669 if( remote.IsEmpty() )
670 remote = GetRemotename();
671
672 if( remote.StartsWith( "https://" ) || remote.StartsWith( "http://" ) )
673 {
675 }
676 else if( remote.StartsWith( "ssh://" ) || remote.StartsWith( "git@" ) || remote.StartsWith( "git+ssh://" )
677 || remote.EndsWith( ".git" ) )
678 {
680 }
681
683}
684
686{
687 if( m_remote.StartsWith( "https://" ) || m_remote.StartsWith( "http://" ) )
689 else if( m_remote.StartsWith( "ssh://" ) || m_remote.StartsWith( "git@" ) || m_remote.StartsWith( "git+ssh://" ) )
691 else
693
695 {
696 wxString uri = m_remote;
697 size_t atPos = uri.find( '@' );
698
699 if( atPos != wxString::npos )
700 {
701 size_t protoEnd = uri.find( "//" );
702
703 if( protoEnd != wxString::npos )
704 {
705 wxString credentials = uri.Mid( protoEnd + 2, atPos - protoEnd - 2 );
706 size_t colonPos = credentials.find( ':' );
707
708 if( colonPos != wxString::npos )
709 {
710 m_username = credentials.Left( colonPos );
711 m_password = credentials.Mid( colonPos + 1, credentials.Length() - colonPos - 1 );
712 }
713 else
714 {
715 m_username = credentials;
716 }
717 }
718 else
719 {
720 m_username = uri.Left( atPos );
721 }
722 }
723
724 if( m_remote.StartsWith( "git@" ) )
725 {
726 // SSH format: git@hostname:path
727 size_t colonPos = m_remote.find( ':' );
728 if( colonPos != wxString::npos )
729 m_hostname = m_remote.Mid( 4, colonPos - 4 );
730 }
731 else
732 {
733 // other URL format: proto://[user@]hostname/path
734 size_t hostStart = m_remote.find( "://" ) + 2;
735 size_t hostEnd = m_remote.find( '/', hostStart );
736 wxString host;
737
738 if( hostEnd != wxString::npos )
739 host = m_remote.Mid( hostStart, hostEnd - hostStart );
740 else
741 host = m_remote.Mid( hostStart );
742
743 atPos = host.find( '@' );
744
745 if( atPos != wxString::npos )
746 m_hostname = host.Mid( atPos + 1 );
747 else
748 m_hostname = host;
749 }
750 }
751}
752
753
754int KIGIT_COMMON::HandleSSHKeyAuthentication( git_cred** aOut, const wxString& aUsername )
755{
757 return HandleSSHAgentAuthentication( aOut, aUsername );
758
759 // SSH key authentication with password
760 wxString sshKey = GetNextPublicKey();
761
762 if( sshKey.IsEmpty() )
763 {
764 wxLogTrace( traceGit, "Finished testing all possible ssh keys" );
765 m_testedTypes |= GIT_CREDENTIAL_SSH_KEY;
766 return GIT_PASSTHROUGH;
767 }
768
769 wxString sshPubKey = sshKey + ".pub";
770 wxString password = GetPassword();
771
772 wxLogTrace( traceGit, "Testing %s\n", sshKey );
773
774 if( git_credential_ssh_key_new( aOut, aUsername.mbc_str(), sshPubKey.mbc_str(), sshKey.mbc_str(),
775 password.mbc_str() ) != GIT_OK )
776 {
777 wxLogTrace( traceGit, "Failed to create SSH key credential for %s: %s",
778 aUsername, KIGIT_COMMON::GetLastGitError() );
779 return GIT_PASSTHROUGH;
780 }
781
782 return GIT_OK;
783}
784
785
786int KIGIT_COMMON::HandlePlaintextAuthentication( git_cred** aOut, const wxString& aUsername )
787{
788 wxString password = GetPassword();
789
790 git_credential_userpass_plaintext_new( aOut, aUsername.mbc_str(), password.mbc_str() );
791 m_testedTypes |= GIT_CREDENTIAL_USERPASS_PLAINTEXT;
792
793 return GIT_OK;
794}
795
796int KIGIT_COMMON::HandleSSHAgentAuthentication( git_cred** aOut, const wxString& aUsername )
797{
798 if( git_credential_ssh_key_from_agent( aOut, aUsername.mbc_str() ) != GIT_OK )
799 {
800 wxLogTrace( traceGit, "Failed to create SSH agent credential for %s: %s",
801 aUsername, KIGIT_COMMON::GetLastGitError() );
802 return GIT_PASSTHROUGH;
803 }
804
806 return GIT_OK;
807}
808
809
810extern "C" int fetchhead_foreach_cb( const char*, const char*,
811 const git_oid* aOID, unsigned int aIsMerge, void* aPayload )
812{
813 if( aIsMerge )
814 git_oid_cpy( (git_oid*) aPayload, aOID );
815
816 return 0;
817}
818
819
820extern "C" void clone_progress_cb( const char* aStr, size_t aLen, size_t aTotal, void* aPayload )
821{
822 KIGIT_REPO_MIXIN* parent = reinterpret_cast<KIGIT_REPO_MIXIN*>( aPayload );
823
824 wxString progressMessage( aStr );
825 parent->UpdateProgress( aLen, aTotal, progressMessage );
826}
827
828
829extern "C" int progress_cb( const char* str, int len, void* aPayload )
830{
831 KIGIT_REPO_MIXIN* parent = reinterpret_cast<KIGIT_REPO_MIXIN*>( aPayload );
832
833 wxString progressMessage( str, len );
834 parent->UpdateProgress( 0, 0, progressMessage );
835
836 return 0;
837}
838
839
840extern "C" int transfer_progress_cb( const git_transfer_progress* aStats, void* aPayload )
841{
842 KIGIT_REPO_MIXIN* parent = reinterpret_cast<KIGIT_REPO_MIXIN*>( aPayload );
843
844 wxString progressMessage = wxString::Format( _( "Received %u of %u objects" ),
845 aStats->received_objects,
846 aStats->total_objects );
847
848 parent->UpdateProgress( aStats->received_objects, aStats->total_objects, progressMessage );
849
850 return 0;
851}
852
853
854extern "C" int update_cb( const char* aRefname, const git_oid* aFirst, const git_oid* aSecond,
855 void* aPayload )
856{
857 constexpr int cstring_len = 8;
858 char a_str[cstring_len + 1];
859 char b_str[cstring_len + 1];
860
861 KIGIT_REPO_MIXIN* parent = reinterpret_cast<KIGIT_REPO_MIXIN*>( aPayload );
862 wxString status;
863
864 git_oid_tostr( b_str, cstring_len, aSecond );
865
866#if ( LIBGIT2_VER_MAJOR >= 1 ) || ( LIBGIT2_VER_MINOR >= 99 )
867 if( !git_oid_is_zero( aFirst ) )
868#else
869 if( !git_oid_iszero( aFirst ) )
870#endif
871 {
872 git_oid_tostr( a_str, cstring_len, aFirst );
873 status = wxString::Format( _( "* [updated] %s..%s %s" ), a_str, b_str, aRefname );
874 }
875 else
876 {
877 status = wxString::Format( _( "* [new] %s %s" ), b_str, aRefname );
878 }
879
880 parent->UpdateProgress( 0, 0, status );
881
882 return 0;
883}
884
885
886extern "C" int push_transfer_progress_cb( unsigned int aCurrent, unsigned int aTotal, size_t aBytes,
887 void* aPayload )
888{
889 long long progress = 100;
890 KIGIT_REPO_MIXIN* parent = reinterpret_cast<KIGIT_REPO_MIXIN*>( aPayload );
891
892 if( aTotal != 0 )
893 {
894 progress = ( aCurrent * 100ll ) / aTotal;
895 }
896
897 wxString progressMessage = wxString::Format( _( "Writing objects: %lld%% (%u/%u), %zu bytes" ),
898 progress, aCurrent, aTotal, aBytes );
899 parent->UpdateProgress( aCurrent, aTotal, progressMessage );
900
901 return 0;
902}
903
904
905extern "C" int push_update_reference_cb( const char* aRefname, const char* aStatus, void* aPayload )
906{
907 KIGIT_REPO_MIXIN* parent = reinterpret_cast<KIGIT_REPO_MIXIN*>( aPayload );
908 wxString status( aStatus );
909
910 if( !status.IsEmpty() )
911 {
912 wxString statusMessage = wxString::Format( _( "* [rejected] %s (%s)" ), aRefname, aStatus );
913 parent->UpdateProgress( 0, 0, statusMessage );
914 }
915 else
916 {
917 wxString statusMessage = wxString::Format( _( "[updated] %s" ), aRefname );
918 parent->UpdateProgress( 0, 0, statusMessage );
919 }
920
921 return 0;
922}
923
924
925extern "C" int credentials_cb( git_cred** aOut, const char* aUrl, const char* aUsername,
926 unsigned int aAllowedTypes, void* aPayload )
927{
928 KIGIT_REPO_MIXIN* parent = reinterpret_cast<KIGIT_REPO_MIXIN*>( aPayload );
929 KIGIT_COMMON* common = parent->GetCommon();
930
931 wxLogTrace( traceGit, "Credentials callback for %s, testing %d", aUrl, aAllowedTypes );
932
934 {
935 wxLogTrace( traceGit, "Local repository, no credentials needed" );
936 return GIT_PASSTHROUGH;
937 }
938
939 if( aAllowedTypes & GIT_CREDENTIAL_USERNAME
940 && !( parent->TestedTypes() & GIT_CREDENTIAL_USERNAME ) )
941 {
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 )
946 {
947 wxLogTrace( traceGit, "Failed to create username credential for %s: %s",
948 username, KIGIT_COMMON::GetLastGitError() );
949 }
950 else
951 {
952 wxLogTrace( traceGit, "Created username credential for %s", username );
953 }
954
955 parent->TestedTypes() |= GIT_CREDENTIAL_USERNAME;
956 }
958 && ( aAllowedTypes & GIT_CREDENTIAL_USERPASS_PLAINTEXT )
959 && !( parent->TestedTypes() & GIT_CREDENTIAL_USERPASS_PLAINTEXT ) )
960 {
961 // Plaintext authentication
962 wxLogTrace( traceGit, "Plaintext authentication for %s at %s with allowed type %d",
963 parent->GetUsername(), aUrl, aAllowedTypes );
964 return common->HandlePlaintextAuthentication( aOut, parent->GetUsername() );
965 }
967 && ( aAllowedTypes & GIT_CREDENTIAL_SSH_KEY )
968 && !( parent->TestedTypes() & GIT_CREDENTIAL_SSH_KEY ) )
969 {
970 // SSH key authentication
971 return common->HandleSSHKeyAuthentication( aOut, parent->GetUsername() );
972 }
973 else
974 {
975 // If we didn't find anything to try, then we don't have a callback set that the
976 // server likes
977 if( !parent->TestedTypes() )
978 return GIT_PASSTHROUGH;
979
980 git_error_clear();
981 git_error_set_str( GIT_ERROR_NET, _( "Unable to authenticate" ).mbc_str() );
982 // Otherwise, we did try something but we failed, so return an authentication error
983 return GIT_EAUTH;
984 }
985
986 return GIT_OK;
987};
virtual void UpdateProgress(int aCurrent, int aTotal, const wxString &aMessage)
Definition: git_progress.h:65
wxString m_username
git_repository * m_repo
std::vector< wxString > GetBranchNames() const
void updateConnectionType()
GIT_CONN_TYPE GetConnType() const
unsigned m_testedTypes
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 m_hostname
wxString m_password
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
GIT_CONN_TYPE m_connType
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.
#define _(s)
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.
bool GetSecret(const wxString &aService, const wxString &aKey, wxString &aSecret)
constexpr int delta
wxLogTrace helper definitions.