gitrepository.cpp 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. #include "gitrepository.h"
  2. #include <QDebug>
  3. #include <QFileInfo>
  4. #include <QDir>
  5. #include <QCryptographicHash>
  6. #include <gitbranch.h>
  7. #include <gitcommit.h>
  8. #include <gittag.h>
  9. #include <gitremote.h>
  10. #include <gitdiff.h>
  11. #include <git2.h>
  12. GitRepository::GitRepository(const QString& root) : QObject(nullptr)
  13. {
  14. if(git_repository_open(&m_raw, root.toUtf8().data()) != 0) {
  15. qDebug() << "Cannot open repository";
  16. close();
  17. return;
  18. }
  19. m_root = root;
  20. m_path = git_repository_workdir(m_raw);
  21. m_name = m_path.split("/", QString::SkipEmptyParts).last();
  22. qDebug() << "New repo:" << m_name << m_root << m_path;
  23. readBranches();
  24. readTags();
  25. readRemotes();
  26. updateHead();
  27. }
  28. GitRepository::~GitRepository()
  29. {
  30. qDebug() << "GitRepository::~GitRepository";
  31. close();
  32. }
  33. void GitRepository::close()
  34. {
  35. if(m_raw) {
  36. git_repository_free(m_raw);
  37. }
  38. m_raw = nullptr;
  39. }
  40. void GitRepository::readBranches()
  41. {
  42. git_reference *branchRef;
  43. git_branch_t branchType;
  44. git_branch_iterator* iter;
  45. git_branch_iterator_new(&iter, m_raw, GIT_BRANCH_ALL);
  46. while(git_branch_next(&branchRef, &branchType, iter) == 0) {
  47. GitBranch* branch = new GitBranch(branchRef, branchType, this);
  48. m_branches.insert(branch->fullName(), QPointer<GitBranch>(branch));
  49. qDebug() << branch->fullName();
  50. qDebug() << branch->type();
  51. }
  52. emit branchesChanged();
  53. }
  54. void GitRepository::readTags()
  55. {
  56. git_tag_foreach(raw(),
  57. [](const char *name, git_oid *oid, void *payload) -> int {
  58. Q_UNUSED(payload)
  59. Q_UNUSED(name)
  60. GitRepository* repo = static_cast<GitRepository*>(payload);
  61. git_tag* tagraw = 0;
  62. if(git_tag_lookup(&tagraw, repo->raw(), oid) != 0) {
  63. qCritical() << "Invalid tag found. Broken repository";
  64. return 1;
  65. }
  66. GitTag* tag = new GitTag(tagraw, repo);
  67. if(tag->isValid()) {
  68. repo->m_tags.insert(tag->targetId(), tag);
  69. }
  70. qDebug() << "Tag found: " << tag->name() << tag->sha1();
  71. return 0;
  72. },
  73. this);
  74. }
  75. void GitRepository::readRemotes()
  76. {
  77. git_strarray str_array;
  78. git_remote_list(&str_array, raw());
  79. for(int i = 0; i < str_array.count; i++) {
  80. GitRemote* remote = GitRemote::fromName(QString::fromLatin1(str_array.strings[i]), this);
  81. if(remote) {
  82. m_remotes.insert(remote->name(), QPointer<GitRemote>(remote));
  83. }
  84. }
  85. // git_reference_foreach_glob(raw(),"refs/remotes/*", [](const char *name, void *payload) -> int
  86. // {
  87. // qDebug() << "Remotes: " << name;
  88. // return 0;
  89. //// if(git_reference_is_remote(reference)) {
  90. //// }
  91. // }, this);
  92. // git_remote* remoteRaw;
  93. // git_reference_is_remote()
  94. }
  95. void GitRepository::checkout(QObject* object)
  96. {
  97. git_checkout_options opts;
  98. git_checkout_init_options(&opts,GIT_CHECKOUT_OPTIONS_VERSION);
  99. opts.checkout_strategy = GIT_CHECKOUT_FORCE;//TODO: modify to merge policy
  100. opts.notify_flags = GIT_CHECKOUT_NOTIFY_ALL;
  101. opts.notify_cb = [](
  102. git_checkout_notify_t why,
  103. const char *path,
  104. const git_diff_file *baseline,
  105. const git_diff_file *target,
  106. const git_diff_file *workdir,
  107. void *payload) -> int {
  108. //TODO: make popup with progressbar
  109. // qDebug() << "path:" << path;
  110. // switch (why) {
  111. // case GIT_CHECKOUT_NOTIFY_CONFLICT:
  112. // qDebug() << "conflict";
  113. // break;
  114. // case GIT_CHECKOUT_NOTIFY_DIRTY:
  115. // qDebug() << "dirty";
  116. // break;
  117. // case GIT_CHECKOUT_NOTIFY_UPDATED:
  118. // qDebug() << "updated";
  119. // break;
  120. // case GIT_CHECKOUT_NOTIFY_UNTRACKED:
  121. // qDebug() << "untracked";
  122. // break;
  123. // case GIT_CHECKOUT_NOTIFY_IGNORED:
  124. // qDebug() << "ignored";
  125. // break;
  126. // default:
  127. // break;
  128. // }
  129. return 0;
  130. };
  131. GitOid oid;
  132. GitBranch* branch = dynamic_cast<GitBranch*>(object);
  133. GitCommit* commit = dynamic_cast<GitCommit*>(object);
  134. if(branch != nullptr) {
  135. commit = GitCommit::fromOid(branch->oid());
  136. qDebug() << "Checkout branch: " << git_checkout_tree(raw(), (git_object*)(commit->raw()), &opts);
  137. switch(branch->type()) {
  138. case GitBranch::Local:
  139. break;
  140. case GitBranch::Remote:
  141. git_reference* ref;
  142. //TODO: better to create API for local branch creation
  143. if(0 == git_branch_create_from_annotated(&ref, raw(), branch->name().toUtf8().data(), branch->annontatedCommit(), 0)) {
  144. branch = new GitBranch(ref, GIT_BRANCH_LOCAL, this);
  145. git_branch_set_upstream(ref, branch->fullName().toUtf8().data());
  146. m_branches.insert(branch->fullName(), QPointer<GitBranch>(branch));
  147. emit branchesChanged();
  148. } else if(m_branches.contains(branch->name())) { /*
  149. TODO: need to think how valid is this.
  150. In case if local branch already exists
  151. it will be checked out.
  152. But from commit tree perspective it doesn't
  153. look as valid behavior.
  154. */
  155. branch = m_branches.value(branch->name());
  156. }
  157. break;
  158. }
  159. git_repository_set_head(raw(), branch->refName().toLatin1().data());
  160. oid = branch->oid();
  161. delete commit;
  162. } else if(commit != nullptr) {
  163. qDebug() << "Checkout commit: " << git_checkout_tree(raw(), (git_object*)(commit->raw()), &opts);
  164. git_repository_set_head_detached(raw(), commit->oid().raw());
  165. oid = commit->oid();
  166. } else {
  167. qDebug() << "Invalid object for checkout";
  168. return;
  169. }
  170. setHead(oid);
  171. }
  172. void GitRepository::updateHead()
  173. {
  174. //Read head
  175. git_reference* ref = nullptr;
  176. git_repository_head(&ref, raw());
  177. git_annotated_commit* commit = nullptr;
  178. git_annotated_commit_from_ref(&commit, raw(), ref);
  179. setHead(GitOid(git_annotated_commit_id(commit), this));
  180. qDebug() << "Repo head" << m_head.toString();
  181. }
  182. QString GitRepository::id() const
  183. {
  184. return QCryptographicHash::hash(name().toUtf8() + path().toUtf8(), QCryptographicHash::Sha1).toHex();
  185. }