rugged 0.16.0 → 0.17.0b1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (392) hide show
  1. data/README.md +373 -243
  2. data/Rakefile +22 -0
  3. data/ext/rugged/extconf.rb +9 -6
  4. data/ext/rugged/rugged.c +123 -31
  5. data/ext/rugged/rugged.h +11 -10
  6. data/ext/rugged/rugged_blob.c +181 -18
  7. data/ext/rugged/rugged_commit.c +196 -18
  8. data/ext/rugged/rugged_config.c +94 -5
  9. data/ext/rugged/rugged_object.c +54 -1
  10. data/ext/rugged/rugged_reference.c +203 -15
  11. data/ext/rugged/{remote.c → rugged_remote.c} +35 -10
  12. data/ext/rugged/rugged_repo.c +323 -81
  13. data/ext/rugged/rugged_revwalk.c +57 -4
  14. data/ext/rugged/rugged_signature.c +3 -3
  15. data/ext/rugged/rugged_tag.c +72 -1
  16. data/ext/rugged/rugged_tree.c +70 -2
  17. data/ext/rugged/vendor/libgit2-dist/deps/regex/config.h +7 -0
  18. data/ext/rugged/vendor/libgit2-dist/deps/regex/regcomp.c +3856 -0
  19. data/ext/rugged/vendor/libgit2-dist/deps/regex/regex.c +85 -0
  20. data/ext/rugged/vendor/libgit2-dist/deps/regex/regex.h +582 -0
  21. data/ext/rugged/vendor/libgit2-dist/deps/regex/regex_internal.c +1744 -0
  22. data/ext/rugged/vendor/libgit2-dist/deps/regex/regex_internal.h +810 -0
  23. data/ext/rugged/vendor/libgit2-dist/deps/regex/regexec.c +4369 -0
  24. data/ext/rugged/vendor/libgit2-dist/examples/diff.c +238 -0
  25. data/ext/rugged/vendor/libgit2-dist/examples/general.c +4 -4
  26. data/ext/rugged/vendor/libgit2-dist/examples/network/fetch.c +101 -85
  27. data/ext/rugged/vendor/libgit2-dist/examples/network/git2.c +7 -3
  28. data/ext/rugged/vendor/libgit2-dist/examples/network/index-pack.c +80 -25
  29. data/ext/rugged/vendor/libgit2-dist/examples/network/ls-remote.c +6 -6
  30. data/ext/rugged/vendor/libgit2-dist/include/git2/attr.h +224 -0
  31. data/ext/rugged/vendor/libgit2-dist/include/git2/blob.h +59 -5
  32. data/ext/rugged/vendor/libgit2-dist/include/git2/branch.h +114 -7
  33. data/ext/rugged/vendor/libgit2-dist/include/git2/commit.h +14 -7
  34. data/ext/rugged/vendor/libgit2-dist/include/git2/common.h +4 -3
  35. data/ext/rugged/vendor/libgit2-dist/include/git2/config.h +105 -27
  36. data/ext/rugged/vendor/libgit2-dist/include/git2/diff.h +409 -0
  37. data/ext/rugged/vendor/libgit2-dist/include/git2/errors.h +47 -82
  38. data/ext/rugged/vendor/libgit2-dist/include/git2/index.h +25 -10
  39. data/ext/rugged/vendor/libgit2-dist/include/git2/indexer.h +46 -1
  40. data/ext/rugged/vendor/libgit2-dist/include/git2/merge.h +35 -0
  41. data/ext/rugged/vendor/libgit2-dist/include/git2/net.h +1 -1
  42. data/ext/rugged/vendor/libgit2-dist/include/git2/notes.h +139 -0
  43. data/ext/rugged/vendor/libgit2-dist/include/git2/object.h +5 -5
  44. data/ext/rugged/vendor/libgit2-dist/include/git2/odb.h +13 -13
  45. data/ext/rugged/vendor/libgit2-dist/include/git2/odb_backend.h +8 -8
  46. data/ext/rugged/vendor/libgit2-dist/include/git2/oid.h +14 -9
  47. data/ext/rugged/vendor/libgit2-dist/include/git2/reflog.h +5 -5
  48. data/ext/rugged/vendor/libgit2-dist/include/git2/refs.h +37 -17
  49. data/ext/rugged/vendor/libgit2-dist/include/git2/refspec.h +17 -9
  50. data/ext/rugged/vendor/libgit2-dist/include/git2/remote.h +83 -16
  51. data/ext/rugged/vendor/libgit2-dist/include/git2/repository.h +24 -10
  52. data/ext/rugged/vendor/libgit2-dist/include/git2/reset.h +44 -0
  53. data/ext/rugged/vendor/libgit2-dist/include/git2/revparse.h +36 -0
  54. data/ext/rugged/vendor/libgit2-dist/include/git2/revwalk.h +74 -6
  55. data/ext/rugged/vendor/libgit2-dist/include/git2/signature.h +3 -3
  56. data/ext/rugged/vendor/libgit2-dist/include/git2/status.h +120 -19
  57. data/ext/rugged/vendor/libgit2-dist/include/git2/submodule.h +103 -0
  58. data/ext/rugged/vendor/libgit2-dist/include/git2/tag.h +28 -10
  59. data/ext/rugged/vendor/libgit2-dist/include/git2/threads.h +1 -1
  60. data/ext/rugged/vendor/libgit2-dist/include/git2/tree.h +13 -13
  61. data/ext/rugged/vendor/libgit2-dist/include/git2/types.h +16 -2
  62. data/ext/rugged/vendor/libgit2-dist/include/git2/version.h +3 -3
  63. data/ext/rugged/vendor/libgit2-dist/include/git2/windows.h +1 -1
  64. data/ext/rugged/vendor/libgit2-dist/include/git2.h +7 -2
  65. data/ext/rugged/vendor/libgit2-dist/src/attr.c +677 -0
  66. data/ext/rugged/vendor/libgit2-dist/src/attr.h +56 -0
  67. data/ext/rugged/vendor/libgit2-dist/src/attr_file.c +609 -0
  68. data/ext/rugged/vendor/libgit2-dist/src/attr_file.h +145 -0
  69. data/ext/rugged/vendor/libgit2-dist/src/blob.c +213 -60
  70. data/ext/rugged/vendor/libgit2-dist/src/blob.h +2 -1
  71. data/ext/rugged/vendor/libgit2-dist/src/branch.c +208 -0
  72. data/ext/rugged/vendor/libgit2-dist/src/branch.h +17 -0
  73. data/ext/rugged/vendor/libgit2-dist/src/bswap.h +1 -1
  74. data/ext/rugged/vendor/libgit2-dist/src/buffer.c +395 -46
  75. data/ext/rugged/vendor/libgit2-dist/src/buffer.h +112 -9
  76. data/ext/rugged/vendor/libgit2-dist/src/cache.c +37 -49
  77. data/ext/rugged/vendor/libgit2-dist/src/cache.h +7 -17
  78. data/ext/rugged/vendor/libgit2-dist/src/cc-compat.h +18 -16
  79. data/ext/rugged/vendor/libgit2-dist/src/commit.c +56 -90
  80. data/ext/rugged/vendor/libgit2-dist/src/commit.h +1 -1
  81. data/ext/rugged/vendor/libgit2-dist/src/common.h +12 -5
  82. data/ext/rugged/vendor/libgit2-dist/src/{win32 → compat}/fnmatch.c +1 -1
  83. data/ext/rugged/vendor/libgit2-dist/src/{win32 → compat}/fnmatch.h +3 -3
  84. data/ext/rugged/vendor/libgit2-dist/src/config.c +247 -158
  85. data/ext/rugged/vendor/libgit2-dist/src/config.h +10 -1
  86. data/ext/rugged/vendor/libgit2-dist/src/config_cache.c +94 -0
  87. data/ext/rugged/vendor/libgit2-dist/src/config_file.c +606 -496
  88. data/ext/rugged/vendor/libgit2-dist/src/config_file.h +31 -0
  89. data/ext/rugged/vendor/libgit2-dist/src/crlf.c +228 -0
  90. data/ext/rugged/vendor/libgit2-dist/src/date.c +876 -0
  91. data/ext/rugged/vendor/libgit2-dist/src/delta-apply.c +15 -9
  92. data/ext/rugged/vendor/libgit2-dist/src/delta-apply.h +2 -2
  93. data/ext/rugged/vendor/libgit2-dist/src/diff.c +814 -0
  94. data/ext/rugged/vendor/libgit2-dist/src/diff.h +43 -0
  95. data/ext/rugged/vendor/libgit2-dist/src/diff_output.c +794 -0
  96. data/ext/rugged/vendor/libgit2-dist/src/errors.c +89 -74
  97. data/ext/rugged/vendor/libgit2-dist/src/fetch.c +94 -66
  98. data/ext/rugged/vendor/libgit2-dist/src/fetch.h +5 -4
  99. data/ext/rugged/vendor/libgit2-dist/src/filebuf.c +157 -100
  100. data/ext/rugged/vendor/libgit2-dist/src/filebuf.h +22 -8
  101. data/ext/rugged/vendor/libgit2-dist/src/fileops.c +330 -206
  102. data/ext/rugged/vendor/libgit2-dist/src/fileops.h +82 -51
  103. data/ext/rugged/vendor/libgit2-dist/src/filter.c +165 -0
  104. data/ext/rugged/vendor/libgit2-dist/src/filter.h +119 -0
  105. data/ext/rugged/vendor/libgit2-dist/src/global.c +4 -4
  106. data/ext/rugged/vendor/libgit2-dist/src/global.h +4 -1
  107. data/ext/rugged/vendor/libgit2-dist/src/hash.c +1 -1
  108. data/ext/rugged/vendor/libgit2-dist/src/hash.h +1 -1
  109. data/ext/rugged/vendor/libgit2-dist/src/ignore.c +203 -0
  110. data/ext/rugged/vendor/libgit2-dist/src/ignore.h +38 -0
  111. data/ext/rugged/vendor/libgit2-dist/src/index.c +220 -169
  112. data/ext/rugged/vendor/libgit2-dist/src/index.h +5 -1
  113. data/ext/rugged/vendor/libgit2-dist/src/indexer.c +601 -102
  114. data/ext/rugged/vendor/libgit2-dist/src/iterator.c +748 -0
  115. data/ext/rugged/vendor/libgit2-dist/src/iterator.h +151 -0
  116. data/ext/rugged/vendor/libgit2-dist/src/khash.h +608 -0
  117. data/ext/rugged/vendor/libgit2-dist/src/map.h +6 -1
  118. data/ext/rugged/vendor/libgit2-dist/src/message.c +61 -0
  119. data/ext/rugged/vendor/libgit2-dist/src/message.h +14 -0
  120. data/ext/rugged/vendor/libgit2-dist/src/mwindow.c +27 -29
  121. data/ext/rugged/vendor/libgit2-dist/src/mwindow.h +4 -4
  122. data/ext/rugged/vendor/libgit2-dist/src/netops.c +375 -56
  123. data/ext/rugged/vendor/libgit2-dist/src/netops.h +12 -9
  124. data/ext/rugged/vendor/libgit2-dist/src/notes.c +548 -0
  125. data/ext/rugged/vendor/libgit2-dist/src/notes.h +28 -0
  126. data/ext/rugged/vendor/libgit2-dist/src/object.c +59 -21
  127. data/ext/rugged/vendor/libgit2-dist/src/odb.c +212 -175
  128. data/ext/rugged/vendor/libgit2-dist/src/odb.h +39 -2
  129. data/ext/rugged/vendor/libgit2-dist/src/odb_loose.c +238 -241
  130. data/ext/rugged/vendor/libgit2-dist/src/odb_pack.c +94 -106
  131. data/ext/rugged/vendor/libgit2-dist/src/oid.c +59 -60
  132. data/ext/rugged/vendor/libgit2-dist/src/oidmap.h +42 -0
  133. data/ext/rugged/vendor/libgit2-dist/src/pack.c +198 -170
  134. data/ext/rugged/vendor/libgit2-dist/src/pack.h +16 -9
  135. data/ext/rugged/vendor/libgit2-dist/src/path.c +496 -106
  136. data/ext/rugged/vendor/libgit2-dist/src/path.h +214 -20
  137. data/ext/rugged/vendor/libgit2-dist/src/pkt.c +88 -159
  138. data/ext/rugged/vendor/libgit2-dist/src/pkt.h +9 -5
  139. data/ext/rugged/vendor/libgit2-dist/src/pool.c +294 -0
  140. data/ext/rugged/vendor/libgit2-dist/src/pool.h +125 -0
  141. data/ext/rugged/vendor/libgit2-dist/src/posix.c +38 -16
  142. data/ext/rugged/vendor/libgit2-dist/src/posix.h +20 -2
  143. data/ext/rugged/vendor/libgit2-dist/src/ppc/sha1.c +1 -1
  144. data/ext/rugged/vendor/libgit2-dist/src/ppc/sha1.h +1 -1
  145. data/ext/rugged/vendor/libgit2-dist/src/pqueue.c +7 -7
  146. data/ext/rugged/vendor/libgit2-dist/src/pqueue.h +1 -1
  147. data/ext/rugged/vendor/libgit2-dist/src/protocol.c +21 -13
  148. data/ext/rugged/vendor/libgit2-dist/src/protocol.h +1 -1
  149. data/ext/rugged/vendor/libgit2-dist/src/reflog.c +125 -103
  150. data/ext/rugged/vendor/libgit2-dist/src/reflog.h +1 -1
  151. data/ext/rugged/vendor/libgit2-dist/src/refs.c +713 -640
  152. data/ext/rugged/vendor/libgit2-dist/src/refs.h +27 -3
  153. data/ext/rugged/vendor/libgit2-dist/src/refspec.c +51 -17
  154. data/ext/rugged/vendor/libgit2-dist/src/refspec.h +13 -1
  155. data/ext/rugged/vendor/libgit2-dist/src/remote.c +307 -119
  156. data/ext/rugged/vendor/libgit2-dist/src/remote.h +3 -2
  157. data/ext/rugged/vendor/libgit2-dist/src/repository.c +593 -442
  158. data/ext/rugged/vendor/libgit2-dist/src/repository.h +80 -2
  159. data/ext/rugged/vendor/libgit2-dist/src/reset.c +103 -0
  160. data/ext/rugged/vendor/libgit2-dist/src/revparse.c +753 -0
  161. data/ext/rugged/vendor/libgit2-dist/src/revwalk.c +434 -158
  162. data/ext/rugged/vendor/libgit2-dist/src/sha1.c +3 -3
  163. data/ext/rugged/vendor/libgit2-dist/src/sha1.h +2 -2
  164. data/ext/rugged/vendor/libgit2-dist/src/sha1_lookup.c +3 -2
  165. data/ext/rugged/vendor/libgit2-dist/src/sha1_lookup.h +1 -1
  166. data/ext/rugged/vendor/libgit2-dist/src/signature.c +69 -80
  167. data/ext/rugged/vendor/libgit2-dist/src/signature.h +1 -1
  168. data/ext/rugged/vendor/libgit2-dist/src/status.c +184 -638
  169. data/ext/rugged/vendor/libgit2-dist/src/strmap.h +64 -0
  170. data/ext/rugged/vendor/libgit2-dist/src/submodule.c +387 -0
  171. data/ext/rugged/vendor/libgit2-dist/src/tag.c +162 -137
  172. data/ext/rugged/vendor/libgit2-dist/src/tag.h +2 -1
  173. data/ext/rugged/vendor/libgit2-dist/src/thread-utils.c +1 -1
  174. data/ext/rugged/vendor/libgit2-dist/src/thread-utils.h +8 -8
  175. data/ext/rugged/vendor/libgit2-dist/src/transport.c +31 -19
  176. data/ext/rugged/vendor/libgit2-dist/src/transport.h +31 -11
  177. data/ext/rugged/vendor/libgit2-dist/src/transports/git.c +168 -193
  178. data/ext/rugged/vendor/libgit2-dist/src/transports/http.c +192 -241
  179. data/ext/rugged/vendor/libgit2-dist/src/transports/local.c +92 -86
  180. data/ext/rugged/vendor/libgit2-dist/src/tree-cache.c +32 -49
  181. data/ext/rugged/vendor/libgit2-dist/src/tree-cache.h +1 -1
  182. data/ext/rugged/vendor/libgit2-dist/src/tree.c +172 -145
  183. data/ext/rugged/vendor/libgit2-dist/src/tree.h +16 -2
  184. data/ext/rugged/vendor/libgit2-dist/src/tsort.c +16 -14
  185. data/ext/rugged/vendor/libgit2-dist/src/unix/map.c +8 -24
  186. data/ext/rugged/vendor/libgit2-dist/src/unix/posix.h +9 -3
  187. data/ext/rugged/vendor/libgit2-dist/src/util.c +94 -38
  188. data/ext/rugged/vendor/libgit2-dist/src/util.h +119 -13
  189. data/ext/rugged/vendor/libgit2-dist/src/vector.c +84 -31
  190. data/ext/rugged/vendor/libgit2-dist/src/vector.h +37 -4
  191. data/ext/rugged/vendor/libgit2-dist/src/win32/dir.c +81 -41
  192. data/ext/rugged/vendor/libgit2-dist/src/{dir.h → win32/dir.h} +4 -9
  193. data/ext/rugged/vendor/libgit2-dist/src/win32/map.c +19 -35
  194. data/ext/rugged/vendor/libgit2-dist/src/win32/mingw-compat.h +1 -1
  195. data/ext/rugged/vendor/libgit2-dist/src/win32/msvc-compat.h +5 -1
  196. data/ext/rugged/vendor/libgit2-dist/src/win32/posix.h +10 -8
  197. data/ext/rugged/vendor/libgit2-dist/src/win32/posix_w32.c +262 -118
  198. data/ext/rugged/vendor/libgit2-dist/src/win32/pthread.c +12 -9
  199. data/ext/rugged/vendor/libgit2-dist/src/win32/pthread.h +1 -1
  200. data/ext/rugged/vendor/libgit2-dist/src/win32/utf-conv.c +30 -26
  201. data/ext/rugged/vendor/libgit2-dist/src/win32/utf-conv.h +2 -1
  202. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xdiff.h +135 -0
  203. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xdiffi.c +572 -0
  204. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xdiffi.h +63 -0
  205. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xemit.c +253 -0
  206. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xemit.h +36 -0
  207. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xhistogram.c +371 -0
  208. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xinclude.h +46 -0
  209. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xmacros.h +54 -0
  210. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xmerge.c +619 -0
  211. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xpatience.c +358 -0
  212. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xprepare.c +483 -0
  213. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xprepare.h +34 -0
  214. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xtypes.h +67 -0
  215. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xutils.c +419 -0
  216. data/ext/rugged/vendor/libgit2-dist/src/xdiff/xutils.h +49 -0
  217. data/ext/rugged/vendor/libgit2-dist/tests-clar/attr/attr_expect.h +43 -0
  218. data/ext/rugged/vendor/libgit2-dist/tests-clar/attr/file.c +226 -0
  219. data/ext/rugged/vendor/libgit2-dist/tests-clar/attr/flags.c +108 -0
  220. data/ext/rugged/vendor/libgit2-dist/tests-clar/attr/lookup.c +262 -0
  221. data/ext/rugged/vendor/libgit2-dist/tests-clar/attr/repo.c +273 -0
  222. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/buf/basic.c +5 -5
  223. data/ext/rugged/vendor/libgit2-dist/tests-clar/clar_helpers.c +181 -0
  224. data/ext/rugged/vendor/libgit2-dist/tests-clar/clar_libgit2.h +55 -0
  225. data/ext/rugged/vendor/libgit2-dist/tests-clar/commit/commit.c +44 -0
  226. data/ext/rugged/vendor/libgit2-dist/tests-clar/commit/parse.c +350 -0
  227. data/ext/rugged/vendor/libgit2-dist/tests-clar/commit/signature.c +65 -0
  228. data/ext/rugged/vendor/libgit2-dist/tests-clar/commit/write.c +140 -0
  229. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/config/add.c +3 -3
  230. data/ext/rugged/vendor/libgit2-dist/tests-clar/config/multivar.c +151 -0
  231. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/config/new.c +5 -5
  232. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/config/read.c +44 -32
  233. data/ext/rugged/vendor/libgit2-dist/tests-clar/config/stress.c +61 -0
  234. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/config/write.c +20 -5
  235. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/buffer.c +613 -0
  236. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/core/dirent.c +39 -26
  237. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/env.c +115 -0
  238. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/errors.c +60 -0
  239. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/core/filebuf.c +4 -18
  240. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/hex.c +22 -0
  241. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/core/oid.c +6 -6
  242. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/path.c +420 -0
  243. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/pool.c +85 -0
  244. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/rmdir.c +68 -0
  245. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/core/string.c +1 -1
  246. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/strmap.c +102 -0
  247. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/core/strtol.c +1 -1
  248. data/ext/rugged/vendor/libgit2-dist/tests-clar/core/vector.c +191 -0
  249. data/ext/rugged/vendor/libgit2-dist/tests-clar/date/date.c +15 -0
  250. data/ext/rugged/vendor/libgit2-dist/tests-clar/diff/blob.c +254 -0
  251. data/ext/rugged/vendor/libgit2-dist/tests-clar/diff/diff_helpers.c +104 -0
  252. data/ext/rugged/vendor/libgit2-dist/tests-clar/diff/diff_helpers.h +47 -0
  253. data/ext/rugged/vendor/libgit2-dist/tests-clar/diff/index.c +92 -0
  254. data/ext/rugged/vendor/libgit2-dist/tests-clar/diff/iterator.c +572 -0
  255. data/ext/rugged/vendor/libgit2-dist/tests-clar/diff/patch.c +99 -0
  256. data/ext/rugged/vendor/libgit2-dist/tests-clar/diff/tree.c +210 -0
  257. data/ext/rugged/vendor/libgit2-dist/tests-clar/diff/workdir.c +543 -0
  258. data/ext/rugged/vendor/libgit2-dist/tests-clar/index/read_tree.c +46 -0
  259. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/index/rename.c +2 -12
  260. data/ext/rugged/vendor/libgit2-dist/tests-clar/index/tests.c +246 -0
  261. data/ext/rugged/vendor/libgit2-dist/tests-clar/network/createremotethenload.c +33 -0
  262. data/ext/rugged/vendor/libgit2-dist/tests-clar/network/remotelocal.c +137 -0
  263. data/ext/rugged/vendor/libgit2-dist/tests-clar/network/remotes.c +183 -0
  264. data/ext/rugged/vendor/libgit2-dist/tests-clar/notes/notes.c +133 -0
  265. data/ext/rugged/vendor/libgit2-dist/tests-clar/notes/notesref.c +57 -0
  266. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/blob/filter.c +125 -0
  267. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/blob/fromchunks.c +87 -0
  268. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/blob/write.c +69 -0
  269. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/commit/commitstagedfile.c +126 -0
  270. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/lookup.c +63 -0
  271. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/message.c +171 -0
  272. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/chars.c +3 -14
  273. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/compare.c +4 -4
  274. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/convert.c +10 -10
  275. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/data.h +0 -0
  276. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/fromstr.c +1 -1
  277. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/hash.c +21 -17
  278. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/short.c +1 -1
  279. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/size.c +1 -1
  280. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/raw/type2string.c +14 -14
  281. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/raw/write.c +455 -0
  282. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/tag/peel.c +56 -0
  283. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/tag/read.c +130 -0
  284. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/tag/write.c +192 -0
  285. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/object/tree/frompath.c +22 -16
  286. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/tree/read.c +75 -0
  287. data/ext/rugged/vendor/libgit2-dist/tests-clar/object/tree/write.c +84 -0
  288. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/odb/loose.c +1 -1
  289. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/odb/loose_data.h +0 -0
  290. data/ext/rugged/vendor/libgit2-dist/tests-clar/odb/mixed.c +24 -0
  291. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/odb/pack_data.h +0 -0
  292. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/odb/packed.c +1 -1
  293. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/odb/sorting.c +1 -1
  294. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/branches/create.c +113 -0
  295. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/branches/delete.c +91 -0
  296. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/branches/listall.c +78 -0
  297. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/branches/move.c +72 -0
  298. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/crashes.c +17 -0
  299. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/create.c +149 -0
  300. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/delete.c +85 -0
  301. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/list.c +53 -0
  302. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/listall.c +36 -0
  303. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/lookup.c +42 -0
  304. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/normalize.c +200 -0
  305. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/overwrite.c +136 -0
  306. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/pack.c +67 -0
  307. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/read.c +194 -0
  308. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/reflog.c +123 -0
  309. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/rename.c +339 -0
  310. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/revparse.c +174 -0
  311. data/ext/rugged/vendor/libgit2-dist/tests-clar/refs/unicode.c +42 -0
  312. data/ext/rugged/vendor/libgit2-dist/tests-clar/repo/discover.c +142 -0
  313. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/repo/getters.c +19 -1
  314. data/ext/rugged/vendor/libgit2-dist/tests-clar/repo/init.c +235 -0
  315. data/ext/rugged/vendor/libgit2-dist/tests-clar/repo/open.c +282 -0
  316. data/ext/rugged/vendor/libgit2-dist/tests-clar/repo/setters.c +80 -0
  317. data/ext/rugged/vendor/libgit2-dist/tests-clar/reset/mixed.c +47 -0
  318. data/ext/rugged/vendor/libgit2-dist/tests-clar/reset/reset_helpers.c +10 -0
  319. data/ext/rugged/vendor/libgit2-dist/tests-clar/reset/reset_helpers.h +6 -0
  320. data/ext/rugged/vendor/libgit2-dist/tests-clar/reset/soft.c +102 -0
  321. data/ext/rugged/vendor/libgit2-dist/{tests/t05-revwalk.c → tests-clar/revwalk/basic.c} +85 -44
  322. data/ext/rugged/vendor/libgit2-dist/tests-clar/revwalk/mergebase.c +148 -0
  323. data/ext/rugged/vendor/libgit2-dist/tests-clar/status/ignore.c +133 -0
  324. data/ext/rugged/vendor/libgit2-dist/{tests-clay → tests-clar}/status/single.c +2 -11
  325. data/ext/rugged/vendor/libgit2-dist/tests-clar/status/status_data.h +202 -0
  326. data/ext/rugged/vendor/libgit2-dist/tests-clar/status/status_helpers.c +49 -0
  327. data/ext/rugged/vendor/libgit2-dist/tests-clar/status/status_helpers.h +33 -0
  328. data/ext/rugged/vendor/libgit2-dist/tests-clar/status/submodules.c +112 -0
  329. data/ext/rugged/vendor/libgit2-dist/tests-clar/status/worktree.c +649 -0
  330. data/ext/rugged/vendor/libgit2-dist/tests-clar/threads/basic.c +20 -0
  331. data/ext/rugged/vendor/libgit2-dist.tar.gz +0 -0
  332. data/lib/rugged/commit.rb +35 -0
  333. data/lib/rugged/object.rb +7 -0
  334. data/lib/rugged/reference.rb +9 -0
  335. data/lib/rugged/repository.rb +99 -3
  336. data/lib/rugged/tag.rb +22 -0
  337. data/lib/rugged/tree.rb +8 -0
  338. data/lib/rugged/version.rb +1 -1
  339. data/lib/rugged.rb +4 -1
  340. data/test/blob_test.rb +2 -2
  341. data/test/commit_test.rb +27 -13
  342. data/test/config_test.rb +44 -0
  343. data/test/coverage/HEAD.json +1 -1
  344. data/test/fixtures/testrepo.git/config +2 -0
  345. data/test/fixtures/testrepo.git/objects/7f/043268ea43ce18e3540acaabf9e090c91965b0 +0 -0
  346. data/test/lib_test.rb +7 -0
  347. data/test/object_test.rb +3 -3
  348. data/test/reference_test.rb +46 -8
  349. data/test/remote_test.rb +8 -2
  350. data/test/repo_pack_test.rb +3 -3
  351. data/test/repo_test.rb +79 -11
  352. data/test/tag_test.rb +9 -2
  353. data/test/test_helper.rb +21 -0
  354. data/test/tree_test.rb +18 -6
  355. metadata +205 -81
  356. data/ext/rugged/vendor/libgit2-dist/include/git2/zlib.h +0 -40
  357. data/ext/rugged/vendor/libgit2-dist/src/hashtable.c +0 -243
  358. data/ext/rugged/vendor/libgit2-dist/src/hashtable.h +0 -80
  359. data/ext/rugged/vendor/libgit2-dist/tests/t00-core.c +0 -628
  360. data/ext/rugged/vendor/libgit2-dist/tests/t01-data.h +0 -322
  361. data/ext/rugged/vendor/libgit2-dist/tests/t01-rawobj.c +0 -638
  362. data/ext/rugged/vendor/libgit2-dist/tests/t03-data.h +0 -344
  363. data/ext/rugged/vendor/libgit2-dist/tests/t03-objwrite.c +0 -255
  364. data/ext/rugged/vendor/libgit2-dist/tests/t04-commit.c +0 -788
  365. data/ext/rugged/vendor/libgit2-dist/tests/t06-index.c +0 -219
  366. data/ext/rugged/vendor/libgit2-dist/tests/t07-hashtable.c +0 -192
  367. data/ext/rugged/vendor/libgit2-dist/tests/t08-tag.c +0 -357
  368. data/ext/rugged/vendor/libgit2-dist/tests/t09-tree.c +0 -221
  369. data/ext/rugged/vendor/libgit2-dist/tests/t10-refs.c +0 -1294
  370. data/ext/rugged/vendor/libgit2-dist/tests/t12-repo.c +0 -174
  371. data/ext/rugged/vendor/libgit2-dist/tests/t13-threads.c +0 -41
  372. data/ext/rugged/vendor/libgit2-dist/tests/t17-bufs.c +0 -61
  373. data/ext/rugged/vendor/libgit2-dist/tests/t18-status.c +0 -448
  374. data/ext/rugged/vendor/libgit2-dist/tests/test_helpers.c +0 -310
  375. data/ext/rugged/vendor/libgit2-dist/tests/test_helpers.h +0 -83
  376. data/ext/rugged/vendor/libgit2-dist/tests/test_lib.c +0 -198
  377. data/ext/rugged/vendor/libgit2-dist/tests/test_lib.h +0 -54
  378. data/ext/rugged/vendor/libgit2-dist/tests/test_main.c +0 -89
  379. data/ext/rugged/vendor/libgit2-dist/tests-clay/clay.h +0 -187
  380. data/ext/rugged/vendor/libgit2-dist/tests-clay/clay_libgit2.h +0 -28
  381. data/ext/rugged/vendor/libgit2-dist/tests-clay/clay_main.c +0 -1073
  382. data/ext/rugged/vendor/libgit2-dist/tests-clay/config/stress.c +0 -39
  383. data/ext/rugged/vendor/libgit2-dist/tests-clay/core/path.c +0 -139
  384. data/ext/rugged/vendor/libgit2-dist/tests-clay/core/rmdir.c +0 -50
  385. data/ext/rugged/vendor/libgit2-dist/tests-clay/core/vector.c +0 -66
  386. data/ext/rugged/vendor/libgit2-dist/tests-clay/network/remotes.c +0 -50
  387. data/ext/rugged/vendor/libgit2-dist/tests-clay/repo/init.c +0 -104
  388. data/ext/rugged/vendor/libgit2-dist/tests-clay/repo/open.c +0 -54
  389. data/ext/rugged/vendor/libgit2-dist/tests-clay/status/status_data.h +0 -48
  390. data/ext/rugged/vendor/libgit2-dist/tests-clay/status/worktree.c +0 -124
  391. data/lib/rugged/objects.rb +0 -45
  392. data/test/fixtures/testrepo.git/refs/heads/new_name +0 -1
@@ -0,0 +1,814 @@
1
+ /*
2
+ * Copyright (C) 2012 the libgit2 contributors
3
+ *
4
+ * This file is part of libgit2, distributed under the GNU GPL v2 with
5
+ * a Linking Exception. For full terms see the included COPYING file.
6
+ */
7
+ #include "common.h"
8
+ #include "git2/diff.h"
9
+ #include "diff.h"
10
+ #include "fileops.h"
11
+ #include "config.h"
12
+ #include "attr_file.h"
13
+
14
+ static char *diff_prefix_from_pathspec(const git_strarray *pathspec)
15
+ {
16
+ git_buf prefix = GIT_BUF_INIT;
17
+ const char *scan;
18
+
19
+ if (git_buf_common_prefix(&prefix, pathspec) < 0)
20
+ return NULL;
21
+
22
+ /* diff prefix will only be leading non-wildcards */
23
+ for (scan = prefix.ptr; *scan && !git__iswildcard(*scan); ++scan);
24
+ git_buf_truncate(&prefix, scan - prefix.ptr);
25
+
26
+ if (prefix.size > 0)
27
+ return git_buf_detach(&prefix);
28
+
29
+ git_buf_free(&prefix);
30
+ return NULL;
31
+ }
32
+
33
+ static bool diff_pathspec_is_interesting(const git_strarray *pathspec)
34
+ {
35
+ const char *str;
36
+
37
+ if (pathspec == NULL || pathspec->count == 0)
38
+ return false;
39
+ if (pathspec->count > 1)
40
+ return true;
41
+
42
+ str = pathspec->strings[0];
43
+ if (!str || !str[0] || (!str[1] && (str[0] == '*' || str[0] == '.')))
44
+ return false;
45
+ return true;
46
+ }
47
+
48
+ static bool diff_path_matches_pathspec(git_diff_list *diff, const char *path)
49
+ {
50
+ unsigned int i;
51
+ git_attr_fnmatch *match;
52
+
53
+ if (!diff->pathspec.length)
54
+ return true;
55
+
56
+ git_vector_foreach(&diff->pathspec, i, match) {
57
+ int result = p_fnmatch(match->pattern, path, 0);
58
+
59
+ /* if we didn't match, look for exact dirname prefix match */
60
+ if (result == FNM_NOMATCH &&
61
+ (match->flags & GIT_ATTR_FNMATCH_HASWILD) == 0 &&
62
+ strncmp(path, match->pattern, match->length) == 0 &&
63
+ path[match->length] == '/')
64
+ result = 0;
65
+
66
+ if (result == 0)
67
+ return (match->flags & GIT_ATTR_FNMATCH_NEGATIVE) ? false : true;
68
+ }
69
+
70
+ return false;
71
+ }
72
+
73
+ static git_diff_delta *diff_delta__alloc(
74
+ git_diff_list *diff,
75
+ git_delta_t status,
76
+ const char *path)
77
+ {
78
+ git_diff_delta *delta = git__calloc(1, sizeof(git_diff_delta));
79
+ if (!delta)
80
+ return NULL;
81
+
82
+ delta->old_file.path = git_pool_strdup(&diff->pool, path);
83
+ if (delta->old_file.path == NULL) {
84
+ git__free(delta);
85
+ return NULL;
86
+ }
87
+
88
+ delta->new_file.path = delta->old_file.path;
89
+
90
+ if (diff->opts.flags & GIT_DIFF_REVERSE) {
91
+ switch (status) {
92
+ case GIT_DELTA_ADDED: status = GIT_DELTA_DELETED; break;
93
+ case GIT_DELTA_DELETED: status = GIT_DELTA_ADDED; break;
94
+ default: break; /* leave other status values alone */
95
+ }
96
+ }
97
+ delta->status = status;
98
+
99
+ return delta;
100
+ }
101
+
102
+ static git_diff_delta *diff_delta__dup(
103
+ const git_diff_delta *d, git_pool *pool)
104
+ {
105
+ git_diff_delta *delta = git__malloc(sizeof(git_diff_delta));
106
+ if (!delta)
107
+ return NULL;
108
+
109
+ memcpy(delta, d, sizeof(git_diff_delta));
110
+
111
+ delta->old_file.path = git_pool_strdup(pool, d->old_file.path);
112
+ if (delta->old_file.path == NULL)
113
+ goto fail;
114
+
115
+ if (d->new_file.path != d->old_file.path) {
116
+ delta->new_file.path = git_pool_strdup(pool, d->new_file.path);
117
+ if (delta->new_file.path == NULL)
118
+ goto fail;
119
+ } else {
120
+ delta->new_file.path = delta->old_file.path;
121
+ }
122
+
123
+ return delta;
124
+
125
+ fail:
126
+ git__free(delta);
127
+ return NULL;
128
+ }
129
+
130
+ static git_diff_delta *diff_delta__merge_like_cgit(
131
+ const git_diff_delta *a, const git_diff_delta *b, git_pool *pool)
132
+ {
133
+ git_diff_delta *dup;
134
+
135
+ /* Emulate C git for merging two diffs (a la 'git diff <sha>').
136
+ *
137
+ * When C git does a diff between the work dir and a tree, it actually
138
+ * diffs with the index but uses the workdir contents. This emulates
139
+ * those choices so we can emulate the type of diff.
140
+ *
141
+ * We have three file descriptions here, let's call them:
142
+ * f1 = a->old_file
143
+ * f2 = a->new_file AND b->old_file
144
+ * f3 = b->new_file
145
+ */
146
+
147
+ /* if f2 == f3 or f2 is deleted, then just dup the 'a' diff */
148
+ if (b->status == GIT_DELTA_UNMODIFIED || a->status == GIT_DELTA_DELETED)
149
+ return diff_delta__dup(a, pool);
150
+
151
+ /* otherwise, base this diff on the 'b' diff */
152
+ if ((dup = diff_delta__dup(b, pool)) == NULL)
153
+ return NULL;
154
+
155
+ /* If 'a' status is uninteresting, then we're done */
156
+ if (a->status == GIT_DELTA_UNMODIFIED)
157
+ return dup;
158
+
159
+ assert(a->status != GIT_DELTA_UNMODIFIED);
160
+ assert(b->status != GIT_DELTA_UNMODIFIED);
161
+
162
+ /* A cgit exception is that the diff of a file that is only in the
163
+ * index (i.e. not in HEAD nor workdir) is given as empty.
164
+ */
165
+ if (dup->status == GIT_DELTA_DELETED) {
166
+ if (a->status == GIT_DELTA_ADDED)
167
+ dup->status = GIT_DELTA_UNMODIFIED;
168
+ /* else don't overwrite DELETE status */
169
+ } else {
170
+ dup->status = a->status;
171
+ }
172
+
173
+ git_oid_cpy(&dup->old_file.oid, &a->old_file.oid);
174
+ dup->old_file.mode = a->old_file.mode;
175
+ dup->old_file.size = a->old_file.size;
176
+ dup->old_file.flags = a->old_file.flags;
177
+
178
+ return dup;
179
+ }
180
+
181
+ static int diff_delta__from_one(
182
+ git_diff_list *diff,
183
+ git_delta_t status,
184
+ const git_index_entry *entry)
185
+ {
186
+ git_diff_delta *delta;
187
+
188
+ if (status == GIT_DELTA_IGNORED &&
189
+ (diff->opts.flags & GIT_DIFF_INCLUDE_IGNORED) == 0)
190
+ return 0;
191
+
192
+ if (status == GIT_DELTA_UNTRACKED &&
193
+ (diff->opts.flags & GIT_DIFF_INCLUDE_UNTRACKED) == 0)
194
+ return 0;
195
+
196
+ if (!diff_path_matches_pathspec(diff, entry->path))
197
+ return 0;
198
+
199
+ delta = diff_delta__alloc(diff, status, entry->path);
200
+ GITERR_CHECK_ALLOC(delta);
201
+
202
+ /* This fn is just for single-sided diffs */
203
+ assert(status != GIT_DELTA_MODIFIED);
204
+
205
+ if (delta->status == GIT_DELTA_DELETED) {
206
+ delta->old_file.mode = entry->mode;
207
+ delta->old_file.size = entry->file_size;
208
+ git_oid_cpy(&delta->old_file.oid, &entry->oid);
209
+ } else /* ADDED, IGNORED, UNTRACKED */ {
210
+ delta->new_file.mode = entry->mode;
211
+ delta->new_file.size = entry->file_size;
212
+ git_oid_cpy(&delta->new_file.oid, &entry->oid);
213
+ }
214
+
215
+ delta->old_file.flags |= GIT_DIFF_FILE_VALID_OID;
216
+ delta->new_file.flags |= GIT_DIFF_FILE_VALID_OID;
217
+
218
+ if (git_vector_insert(&diff->deltas, delta) < 0) {
219
+ git__free(delta);
220
+ return -1;
221
+ }
222
+
223
+ return 0;
224
+ }
225
+
226
+ static int diff_delta__from_two(
227
+ git_diff_list *diff,
228
+ git_delta_t status,
229
+ const git_index_entry *old_entry,
230
+ uint32_t old_mode,
231
+ const git_index_entry *new_entry,
232
+ uint32_t new_mode,
233
+ git_oid *new_oid)
234
+ {
235
+ git_diff_delta *delta;
236
+
237
+ if (status == GIT_DELTA_UNMODIFIED &&
238
+ (diff->opts.flags & GIT_DIFF_INCLUDE_UNMODIFIED) == 0)
239
+ return 0;
240
+
241
+ if ((diff->opts.flags & GIT_DIFF_REVERSE) != 0) {
242
+ uint32_t temp_mode = old_mode;
243
+ const git_index_entry *temp_entry = old_entry;
244
+ old_entry = new_entry;
245
+ new_entry = temp_entry;
246
+ old_mode = new_mode;
247
+ new_mode = temp_mode;
248
+ }
249
+
250
+ delta = diff_delta__alloc(diff, status, old_entry->path);
251
+ GITERR_CHECK_ALLOC(delta);
252
+
253
+ delta->old_file.mode = old_mode;
254
+ git_oid_cpy(&delta->old_file.oid, &old_entry->oid);
255
+ delta->old_file.flags |= GIT_DIFF_FILE_VALID_OID;
256
+
257
+ delta->new_file.mode = new_mode;
258
+ git_oid_cpy(&delta->new_file.oid, new_oid ? new_oid : &new_entry->oid);
259
+ if (new_oid || !git_oid_iszero(&new_entry->oid))
260
+ delta->new_file.flags |= GIT_DIFF_FILE_VALID_OID;
261
+
262
+ if (git_vector_insert(&diff->deltas, delta) < 0) {
263
+ git__free(delta);
264
+ return -1;
265
+ }
266
+
267
+ return 0;
268
+ }
269
+
270
+ static char *diff_strdup_prefix(git_pool *pool, const char *prefix)
271
+ {
272
+ size_t len = strlen(prefix);
273
+
274
+ /* append '/' at end if needed */
275
+ if (len > 0 && prefix[len - 1] != '/')
276
+ return git_pool_strcat(pool, prefix, "/");
277
+ else
278
+ return git_pool_strndup(pool, prefix, len + 1);
279
+ }
280
+
281
+ static int diff_delta__cmp(const void *a, const void *b)
282
+ {
283
+ const git_diff_delta *da = a, *db = b;
284
+ int val = strcmp(da->old_file.path, db->old_file.path);
285
+ return val ? val : ((int)da->status - (int)db->status);
286
+ }
287
+
288
+ static int config_bool(git_config *cfg, const char *name, int defvalue)
289
+ {
290
+ int val = defvalue;
291
+
292
+ if (git_config_get_bool(&val, cfg, name) < 0)
293
+ giterr_clear();
294
+
295
+ return val;
296
+ }
297
+
298
+ static git_diff_list *git_diff_list_alloc(
299
+ git_repository *repo, const git_diff_options *opts)
300
+ {
301
+ git_config *cfg;
302
+ size_t i;
303
+ git_diff_list *diff = git__calloc(1, sizeof(git_diff_list));
304
+ if (diff == NULL)
305
+ return NULL;
306
+
307
+ diff->repo = repo;
308
+
309
+ if (git_vector_init(&diff->deltas, 0, diff_delta__cmp) < 0 ||
310
+ git_pool_init(&diff->pool, 1, 0) < 0)
311
+ goto fail;
312
+
313
+ /* load config values that affect diff behavior */
314
+ if (git_repository_config__weakptr(&cfg, repo) < 0)
315
+ goto fail;
316
+ if (config_bool(cfg, "core.symlinks", 1))
317
+ diff->diffcaps = diff->diffcaps | GIT_DIFFCAPS_HAS_SYMLINKS;
318
+ if (config_bool(cfg, "core.ignorestat", 0))
319
+ diff->diffcaps = diff->diffcaps | GIT_DIFFCAPS_ASSUME_UNCHANGED;
320
+ if (config_bool(cfg, "core.filemode", 1))
321
+ diff->diffcaps = diff->diffcaps | GIT_DIFFCAPS_TRUST_MODE_BITS;
322
+ if (config_bool(cfg, "core.trustctime", 1))
323
+ diff->diffcaps = diff->diffcaps | GIT_DIFFCAPS_TRUST_CTIME;
324
+ /* Don't set GIT_DIFFCAPS_USE_DEV - compile time option in core git */
325
+
326
+ if (opts == NULL)
327
+ return diff;
328
+
329
+ memcpy(&diff->opts, opts, sizeof(git_diff_options));
330
+ memset(&diff->opts.pathspec, 0, sizeof(diff->opts.pathspec));
331
+
332
+ diff->opts.old_prefix = diff_strdup_prefix(&diff->pool,
333
+ opts->old_prefix ? opts->old_prefix : DIFF_OLD_PREFIX_DEFAULT);
334
+ diff->opts.new_prefix = diff_strdup_prefix(&diff->pool,
335
+ opts->new_prefix ? opts->new_prefix : DIFF_NEW_PREFIX_DEFAULT);
336
+
337
+ if (!diff->opts.old_prefix || !diff->opts.new_prefix)
338
+ goto fail;
339
+
340
+ if (diff->opts.flags & GIT_DIFF_REVERSE) {
341
+ char *swap = diff->opts.old_prefix;
342
+ diff->opts.old_prefix = diff->opts.new_prefix;
343
+ diff->opts.new_prefix = swap;
344
+ }
345
+
346
+ /* only copy pathspec if it is "interesting" so we can test
347
+ * diff->pathspec.length > 0 to know if it is worth calling
348
+ * fnmatch as we iterate.
349
+ */
350
+ if (!diff_pathspec_is_interesting(&opts->pathspec))
351
+ return diff;
352
+
353
+ if (git_vector_init(
354
+ &diff->pathspec, (unsigned int)opts->pathspec.count, NULL) < 0)
355
+ goto fail;
356
+
357
+ for (i = 0; i < opts->pathspec.count; ++i) {
358
+ int ret;
359
+ const char *pattern = opts->pathspec.strings[i];
360
+ git_attr_fnmatch *match = git__calloc(1, sizeof(git_attr_fnmatch));
361
+ if (!match)
362
+ goto fail;
363
+ match->flags = GIT_ATTR_FNMATCH_ALLOWSPACE;
364
+ ret = git_attr_fnmatch__parse(match, &diff->pool, NULL, &pattern);
365
+ if (ret == GIT_ENOTFOUND) {
366
+ git__free(match);
367
+ continue;
368
+ } else if (ret < 0)
369
+ goto fail;
370
+
371
+ if (git_vector_insert(&diff->pathspec, match) < 0)
372
+ goto fail;
373
+ }
374
+
375
+ return diff;
376
+
377
+ fail:
378
+ git_diff_list_free(diff);
379
+ return NULL;
380
+ }
381
+
382
+ void git_diff_list_free(git_diff_list *diff)
383
+ {
384
+ git_diff_delta *delta;
385
+ git_attr_fnmatch *match;
386
+ unsigned int i;
387
+
388
+ if (!diff)
389
+ return;
390
+
391
+ git_vector_foreach(&diff->deltas, i, delta) {
392
+ git__free(delta);
393
+ diff->deltas.contents[i] = NULL;
394
+ }
395
+ git_vector_free(&diff->deltas);
396
+
397
+ git_vector_foreach(&diff->pathspec, i, match) {
398
+ git__free(match);
399
+ diff->pathspec.contents[i] = NULL;
400
+ }
401
+ git_vector_free(&diff->pathspec);
402
+
403
+ git_pool_clear(&diff->pool);
404
+ git__free(diff);
405
+ }
406
+
407
+ static int oid_for_workdir_item(
408
+ git_repository *repo,
409
+ const git_index_entry *item,
410
+ git_oid *oid)
411
+ {
412
+ int result;
413
+ git_buf full_path = GIT_BUF_INIT;
414
+
415
+ if (git_buf_joinpath(&full_path, git_repository_workdir(repo), item->path) < 0)
416
+ return -1;
417
+
418
+ /* calculate OID for file if possible*/
419
+ if (S_ISLNK(item->mode))
420
+ result = git_odb__hashlink(oid, full_path.ptr);
421
+ else if (!git__is_sizet(item->file_size)) {
422
+ giterr_set(GITERR_OS, "File size overflow for 32-bit systems");
423
+ result = -1;
424
+ } else {
425
+ int fd = git_futils_open_ro(full_path.ptr);
426
+ if (fd < 0)
427
+ result = fd;
428
+ else {
429
+ result = git_odb__hashfd(
430
+ oid, fd, (size_t)item->file_size, GIT_OBJ_BLOB);
431
+ p_close(fd);
432
+ }
433
+ }
434
+
435
+ git_buf_free(&full_path);
436
+
437
+ return result;
438
+ }
439
+
440
+ #define MODE_BITS_MASK 0000777
441
+
442
+ static int maybe_modified(
443
+ git_iterator *old_iter,
444
+ const git_index_entry *oitem,
445
+ git_iterator *new_iter,
446
+ const git_index_entry *nitem,
447
+ git_diff_list *diff)
448
+ {
449
+ git_oid noid, *use_noid = NULL;
450
+ git_delta_t status = GIT_DELTA_MODIFIED;
451
+ unsigned int omode = oitem->mode;
452
+ unsigned int nmode = nitem->mode;
453
+
454
+ GIT_UNUSED(old_iter);
455
+
456
+ if (!diff_path_matches_pathspec(diff, oitem->path))
457
+ return 0;
458
+
459
+ /* on platforms with no symlinks, promote plain files to symlinks */
460
+ if (S_ISLNK(omode) && S_ISREG(nmode) &&
461
+ !(diff->diffcaps & GIT_DIFFCAPS_HAS_SYMLINKS))
462
+ nmode = GIT_MODE_TYPE(omode) | (nmode & GIT_MODE_PERMS_MASK);
463
+
464
+ /* on platforms with no execmode, just preserve old mode */
465
+ if (!(diff->diffcaps & GIT_DIFFCAPS_TRUST_MODE_BITS) &&
466
+ (nmode & MODE_BITS_MASK) != (omode & MODE_BITS_MASK) &&
467
+ new_iter->type == GIT_ITERATOR_WORKDIR)
468
+ nmode = (nmode & ~MODE_BITS_MASK) | (omode & MODE_BITS_MASK);
469
+
470
+ /* support "assume unchanged" (poorly, b/c we still stat everything) */
471
+ if ((diff->diffcaps & GIT_DIFFCAPS_ASSUME_UNCHANGED) != 0)
472
+ status = (oitem->flags_extended & GIT_IDXENTRY_INTENT_TO_ADD) ?
473
+ GIT_DELTA_MODIFIED : GIT_DELTA_UNMODIFIED;
474
+
475
+ /* support "skip worktree" index bit */
476
+ else if ((oitem->flags_extended & GIT_IDXENTRY_SKIP_WORKTREE) != 0)
477
+ status = GIT_DELTA_UNMODIFIED;
478
+
479
+ /* if basic type of file changed, then split into delete and add */
480
+ else if (GIT_MODE_TYPE(omode) != GIT_MODE_TYPE(nmode)) {
481
+ if (diff_delta__from_one(diff, GIT_DELTA_DELETED, oitem) < 0 ||
482
+ diff_delta__from_one(diff, GIT_DELTA_ADDED, nitem) < 0)
483
+ return -1;
484
+ return 0;
485
+ }
486
+
487
+ /* if oids and modes match, then file is unmodified */
488
+ else if (git_oid_cmp(&oitem->oid, &nitem->oid) == 0 &&
489
+ omode == nmode)
490
+ status = GIT_DELTA_UNMODIFIED;
491
+
492
+ /* if modes match and we have an unknown OID and a workdir iterator,
493
+ * then check deeper for matching
494
+ */
495
+ else if (omode == nmode &&
496
+ git_oid_iszero(&nitem->oid) &&
497
+ new_iter->type == GIT_ITERATOR_WORKDIR)
498
+ {
499
+ /* TODO: add check against index file st_mtime to avoid racy-git */
500
+
501
+ /* if they files look exactly alike, then we'll assume the same */
502
+ if (oitem->file_size == nitem->file_size &&
503
+ (!(diff->diffcaps & GIT_DIFFCAPS_TRUST_CTIME) ||
504
+ (oitem->ctime.seconds == nitem->ctime.seconds)) &&
505
+ oitem->mtime.seconds == nitem->mtime.seconds &&
506
+ (!(diff->diffcaps & GIT_DIFFCAPS_USE_DEV) ||
507
+ (oitem->dev == nitem->dev)) &&
508
+ oitem->ino == nitem->ino &&
509
+ oitem->uid == nitem->uid &&
510
+ oitem->gid == nitem->gid)
511
+ status = GIT_DELTA_UNMODIFIED;
512
+
513
+ else if (S_ISGITLINK(nmode)) {
514
+ git_submodule *sub;
515
+
516
+ if ((diff->opts.flags & GIT_DIFF_IGNORE_SUBMODULES) != 0)
517
+ status = GIT_DELTA_UNMODIFIED;
518
+ else if (git_submodule_lookup(&sub, diff->repo, nitem->path) < 0)
519
+ return -1;
520
+ else if (sub->ignore == GIT_SUBMODULE_IGNORE_ALL)
521
+ status = GIT_DELTA_UNMODIFIED;
522
+ else {
523
+ /* TODO: support other GIT_SUBMODULE_IGNORE values */
524
+ status = GIT_DELTA_UNMODIFIED;
525
+ }
526
+ }
527
+
528
+ /* TODO: check git attributes so we will not have to read the file
529
+ * in if it is marked binary.
530
+ */
531
+
532
+ else if (oid_for_workdir_item(diff->repo, nitem, &noid) < 0)
533
+ return -1;
534
+
535
+ else if (git_oid_cmp(&oitem->oid, &noid) == 0 &&
536
+ omode == nmode)
537
+ status = GIT_DELTA_UNMODIFIED;
538
+
539
+ /* store calculated oid so we don't have to recalc later */
540
+ use_noid = &noid;
541
+ }
542
+
543
+ return diff_delta__from_two(
544
+ diff, status, oitem, omode, nitem, nmode, use_noid);
545
+ }
546
+
547
+ static int diff_from_iterators(
548
+ git_repository *repo,
549
+ const git_diff_options *opts, /**< can be NULL for defaults */
550
+ git_iterator *old_iter,
551
+ git_iterator *new_iter,
552
+ git_diff_list **diff_ptr)
553
+ {
554
+ const git_index_entry *oitem, *nitem;
555
+ git_buf ignore_prefix = GIT_BUF_INIT;
556
+ git_diff_list *diff = git_diff_list_alloc(repo, opts);
557
+ if (!diff)
558
+ goto fail;
559
+
560
+ diff->old_src = old_iter->type;
561
+ diff->new_src = new_iter->type;
562
+
563
+ if (git_iterator_current(old_iter, &oitem) < 0 ||
564
+ git_iterator_current(new_iter, &nitem) < 0)
565
+ goto fail;
566
+
567
+ /* run iterators building diffs */
568
+ while (oitem || nitem) {
569
+
570
+ /* create DELETED records for old items not matched in new */
571
+ if (oitem && (!nitem || strcmp(oitem->path, nitem->path) < 0)) {
572
+ if (diff_delta__from_one(diff, GIT_DELTA_DELETED, oitem) < 0 ||
573
+ git_iterator_advance(old_iter, &oitem) < 0)
574
+ goto fail;
575
+ }
576
+
577
+ /* create ADDED, TRACKED, or IGNORED records for new items not
578
+ * matched in old (and/or descend into directories as needed)
579
+ */
580
+ else if (nitem && (!oitem || strcmp(oitem->path, nitem->path) > 0)) {
581
+ git_delta_t delta_type = GIT_DELTA_UNTRACKED;
582
+
583
+ /* check if contained in ignored parent directory */
584
+ if (git_buf_len(&ignore_prefix) &&
585
+ git__prefixcmp(nitem->path, git_buf_cstr(&ignore_prefix)) == 0)
586
+ delta_type = GIT_DELTA_IGNORED;
587
+
588
+ if (S_ISDIR(nitem->mode)) {
589
+ /* recurse into directory only if there are tracked items in
590
+ * it or if the user requested the contents of untracked
591
+ * directories and it is not under an ignored directory.
592
+ */
593
+ if ((oitem && git__prefixcmp(oitem->path, nitem->path) == 0) ||
594
+ (delta_type == GIT_DELTA_UNTRACKED &&
595
+ (diff->opts.flags & GIT_DIFF_RECURSE_UNTRACKED_DIRS) != 0))
596
+ {
597
+ /* if this directory is ignored, remember it as the
598
+ * "ignore_prefix" for processing contained items
599
+ */
600
+ if (delta_type == GIT_DELTA_UNTRACKED &&
601
+ git_iterator_current_is_ignored(new_iter))
602
+ git_buf_sets(&ignore_prefix, nitem->path);
603
+
604
+ if (git_iterator_advance_into_directory(new_iter, &nitem) < 0)
605
+ goto fail;
606
+
607
+ continue;
608
+ }
609
+ }
610
+
611
+ /* In core git, the next two "else if" clauses are effectively
612
+ * reversed -- i.e. when an untracked file contained in an
613
+ * ignored directory is individually ignored, it shows up as an
614
+ * ignored file in the diff list, even though other untracked
615
+ * files in the same directory are skipped completely.
616
+ *
617
+ * To me, this is odd. If the directory is ignored and the file
618
+ * is untracked, we should skip it consistently, regardless of
619
+ * whether it happens to match a pattern in the ignore file.
620
+ *
621
+ * To match the core git behavior, just reverse the following
622
+ * two "else if" cases so that individual file ignores are
623
+ * checked before container directory exclusions are used to
624
+ * skip the file.
625
+ */
626
+ else if (delta_type == GIT_DELTA_IGNORED) {
627
+ if (git_iterator_advance(new_iter, &nitem) < 0)
628
+ goto fail;
629
+ continue; /* ignored parent directory, so skip completely */
630
+ }
631
+
632
+ else if (git_iterator_current_is_ignored(new_iter))
633
+ delta_type = GIT_DELTA_IGNORED;
634
+
635
+ else if (new_iter->type != GIT_ITERATOR_WORKDIR)
636
+ delta_type = GIT_DELTA_ADDED;
637
+
638
+ if (diff_delta__from_one(diff, delta_type, nitem) < 0 ||
639
+ git_iterator_advance(new_iter, &nitem) < 0)
640
+ goto fail;
641
+ }
642
+
643
+ /* otherwise item paths match, so create MODIFIED record
644
+ * (or ADDED and DELETED pair if type changed)
645
+ */
646
+ else {
647
+ assert(oitem && nitem && strcmp(oitem->path, nitem->path) == 0);
648
+
649
+ if (maybe_modified(old_iter, oitem, new_iter, nitem, diff) < 0 ||
650
+ git_iterator_advance(old_iter, &oitem) < 0 ||
651
+ git_iterator_advance(new_iter, &nitem) < 0)
652
+ goto fail;
653
+ }
654
+ }
655
+
656
+ git_iterator_free(old_iter);
657
+ git_iterator_free(new_iter);
658
+ git_buf_free(&ignore_prefix);
659
+
660
+ *diff_ptr = diff;
661
+ return 0;
662
+
663
+ fail:
664
+ git_iterator_free(old_iter);
665
+ git_iterator_free(new_iter);
666
+ git_buf_free(&ignore_prefix);
667
+
668
+ git_diff_list_free(diff);
669
+ *diff_ptr = NULL;
670
+ return -1;
671
+ }
672
+
673
+
674
+ int git_diff_tree_to_tree(
675
+ git_repository *repo,
676
+ const git_diff_options *opts, /**< can be NULL for defaults */
677
+ git_tree *old_tree,
678
+ git_tree *new_tree,
679
+ git_diff_list **diff)
680
+ {
681
+ git_iterator *a = NULL, *b = NULL;
682
+ char *prefix = opts ? diff_prefix_from_pathspec(&opts->pathspec) : NULL;
683
+
684
+ assert(repo && old_tree && new_tree && diff);
685
+
686
+ if (git_iterator_for_tree_range(&a, repo, old_tree, prefix, prefix) < 0 ||
687
+ git_iterator_for_tree_range(&b, repo, new_tree, prefix, prefix) < 0)
688
+ return -1;
689
+
690
+ git__free(prefix);
691
+
692
+ return diff_from_iterators(repo, opts, a, b, diff);
693
+ }
694
+
695
+ int git_diff_index_to_tree(
696
+ git_repository *repo,
697
+ const git_diff_options *opts,
698
+ git_tree *old_tree,
699
+ git_diff_list **diff)
700
+ {
701
+ git_iterator *a = NULL, *b = NULL;
702
+ char *prefix = opts ? diff_prefix_from_pathspec(&opts->pathspec) : NULL;
703
+
704
+ assert(repo && diff);
705
+
706
+ if (git_iterator_for_tree_range(&a, repo, old_tree, prefix, prefix) < 0 ||
707
+ git_iterator_for_index_range(&b, repo, prefix, prefix) < 0)
708
+ return -1;
709
+
710
+ git__free(prefix);
711
+
712
+ return diff_from_iterators(repo, opts, a, b, diff);
713
+ }
714
+
715
+ int git_diff_workdir_to_index(
716
+ git_repository *repo,
717
+ const git_diff_options *opts,
718
+ git_diff_list **diff)
719
+ {
720
+ git_iterator *a = NULL, *b = NULL;
721
+ char *prefix = opts ? diff_prefix_from_pathspec(&opts->pathspec) : NULL;
722
+
723
+ assert(repo && diff);
724
+
725
+ if (git_iterator_for_index_range(&a, repo, prefix, prefix) < 0 ||
726
+ git_iterator_for_workdir_range(&b, repo, prefix, prefix) < 0)
727
+ return -1;
728
+
729
+ git__free(prefix);
730
+
731
+ return diff_from_iterators(repo, opts, a, b, diff);
732
+ }
733
+
734
+
735
+ int git_diff_workdir_to_tree(
736
+ git_repository *repo,
737
+ const git_diff_options *opts,
738
+ git_tree *old_tree,
739
+ git_diff_list **diff)
740
+ {
741
+ git_iterator *a = NULL, *b = NULL;
742
+ char *prefix = opts ? diff_prefix_from_pathspec(&opts->pathspec) : NULL;
743
+
744
+ assert(repo && old_tree && diff);
745
+
746
+ if (git_iterator_for_tree_range(&a, repo, old_tree, prefix, prefix) < 0 ||
747
+ git_iterator_for_workdir_range(&b, repo, prefix, prefix) < 0)
748
+ return -1;
749
+
750
+ git__free(prefix);
751
+
752
+ return diff_from_iterators(repo, opts, a, b, diff);
753
+ }
754
+
755
+ int git_diff_merge(
756
+ git_diff_list *onto,
757
+ const git_diff_list *from)
758
+ {
759
+ int error = 0;
760
+ git_pool onto_pool;
761
+ git_vector onto_new;
762
+ git_diff_delta *delta;
763
+ unsigned int i, j;
764
+
765
+ assert(onto && from);
766
+
767
+ if (!from->deltas.length)
768
+ return 0;
769
+
770
+ if (git_vector_init(&onto_new, onto->deltas.length, diff_delta__cmp) < 0 ||
771
+ git_pool_init(&onto_pool, 1, 0) < 0)
772
+ return -1;
773
+
774
+ for (i = 0, j = 0; i < onto->deltas.length || j < from->deltas.length; ) {
775
+ git_diff_delta *o = GIT_VECTOR_GET(&onto->deltas, i);
776
+ const git_diff_delta *f = GIT_VECTOR_GET(&from->deltas, j);
777
+ int cmp = !f ? -1 : !o ? 1 : strcmp(o->old_file.path, f->old_file.path);
778
+
779
+ if (cmp < 0) {
780
+ delta = diff_delta__dup(o, &onto_pool);
781
+ i++;
782
+ } else if (cmp > 0) {
783
+ delta = diff_delta__dup(f, &onto_pool);
784
+ j++;
785
+ } else {
786
+ delta = diff_delta__merge_like_cgit(o, f, &onto_pool);
787
+ i++;
788
+ j++;
789
+ }
790
+
791
+ if ((error = !delta ? -1 : git_vector_insert(&onto_new, delta)) < 0)
792
+ break;
793
+ }
794
+
795
+ if (!error) {
796
+ git_vector_swap(&onto->deltas, &onto_new);
797
+ git_pool_swap(&onto->pool, &onto_pool);
798
+ onto->new_src = from->new_src;
799
+
800
+ /* prefix strings also come from old pool, so recreate those.*/
801
+ onto->opts.old_prefix =
802
+ git_pool_strdup(&onto->pool, onto->opts.old_prefix);
803
+ onto->opts.new_prefix =
804
+ git_pool_strdup(&onto->pool, onto->opts.new_prefix);
805
+ }
806
+
807
+ git_vector_foreach(&onto_new, i, delta)
808
+ git__free(delta);
809
+ git_vector_free(&onto_new);
810
+ git_pool_clear(&onto_pool);
811
+
812
+ return error;
813
+ }
814
+