rugged 0.19.0 → 0.21.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (453) hide show
  1. checksums.yaml +7 -0
  2. data/LICENSE +1 -1
  3. data/README.md +151 -25
  4. data/ext/rugged/extconf.rb +48 -27
  5. data/ext/rugged/rugged.c +107 -35
  6. data/ext/rugged/rugged.h +44 -8
  7. data/ext/rugged/rugged_blame.c +292 -0
  8. data/ext/rugged/rugged_blob.c +84 -31
  9. data/ext/rugged/rugged_branch.c +90 -223
  10. data/ext/rugged/rugged_branch_collection.c +445 -0
  11. data/ext/rugged/rugged_commit.c +190 -9
  12. data/ext/rugged/rugged_config.c +1 -1
  13. data/ext/rugged/rugged_cred.c +125 -0
  14. data/ext/rugged/rugged_diff.c +267 -94
  15. data/ext/rugged/rugged_diff_delta.c +14 -3
  16. data/ext/rugged/rugged_diff_hunk.c +31 -28
  17. data/ext/rugged/rugged_diff_line.c +21 -17
  18. data/ext/rugged/rugged_index.c +326 -6
  19. data/ext/rugged/rugged_note.c +39 -34
  20. data/ext/rugged/rugged_object.c +9 -9
  21. data/ext/rugged/rugged_patch.c +245 -0
  22. data/ext/rugged/rugged_reference.c +67 -332
  23. data/ext/rugged/rugged_reference_collection.c +490 -0
  24. data/ext/rugged/rugged_remote.c +447 -351
  25. data/ext/rugged/rugged_remote_collection.c +285 -0
  26. data/ext/rugged/rugged_repo.c +752 -203
  27. data/ext/rugged/rugged_revwalk.c +119 -27
  28. data/ext/rugged/rugged_settings.c +48 -1
  29. data/ext/rugged/rugged_signature.c +30 -16
  30. data/ext/rugged/rugged_tag.c +86 -191
  31. data/ext/rugged/rugged_tag_collection.c +279 -0
  32. data/ext/rugged/rugged_tree.c +159 -22
  33. data/lib/rugged/branch.rb +8 -17
  34. data/lib/rugged/credentials.rb +43 -0
  35. data/lib/rugged/diff/delta.rb +1 -2
  36. data/lib/rugged/diff/hunk.rb +4 -9
  37. data/lib/rugged/diff/line.rb +23 -3
  38. data/lib/rugged/diff.rb +3 -1
  39. data/lib/rugged/patch.rb +26 -0
  40. data/lib/rugged/reference.rb +1 -3
  41. data/lib/rugged/remote.rb +0 -9
  42. data/lib/rugged/repository.rb +70 -13
  43. data/lib/rugged/tag.rb +23 -18
  44. data/lib/rugged/tree.rb +7 -0
  45. data/lib/rugged/version.rb +1 -1
  46. data/lib/rugged.rb +8 -1
  47. data/vendor/libgit2/AUTHORS +75 -0
  48. data/vendor/libgit2/CMakeLists.txt +490 -0
  49. data/vendor/libgit2/COPYING +920 -0
  50. data/vendor/libgit2/Makefile.embed +25 -7
  51. data/vendor/libgit2/cmake/Modules/AddCFlagIfSupported.cmake +16 -0
  52. data/vendor/libgit2/cmake/Modules/FindHTTP_Parser.cmake +39 -0
  53. data/vendor/libgit2/cmake/Modules/FindIconv.cmake +43 -0
  54. data/vendor/libgit2/cmake/Modules/FindLIBSSH2.cmake +44 -0
  55. data/vendor/libgit2/deps/http-parser/LICENSE-MIT +23 -0
  56. data/vendor/libgit2/deps/regex/regex.c +10 -3
  57. data/vendor/libgit2/include/git2/attr.h +2 -1
  58. data/vendor/libgit2/include/git2/blame.h +213 -0
  59. data/vendor/libgit2/include/git2/blob.h +57 -29
  60. data/vendor/libgit2/include/git2/branch.h +56 -44
  61. data/vendor/libgit2/include/git2/buffer.h +112 -0
  62. data/vendor/libgit2/include/git2/checkout.h +72 -37
  63. data/vendor/libgit2/include/git2/cherrypick.h +87 -0
  64. data/vendor/libgit2/include/git2/clone.h +153 -56
  65. data/vendor/libgit2/include/git2/commit.h +82 -12
  66. data/vendor/libgit2/include/git2/common.h +31 -19
  67. data/vendor/libgit2/include/git2/config.h +124 -19
  68. data/vendor/libgit2/include/git2/cred_helpers.h +1 -1
  69. data/vendor/libgit2/include/git2/diff.h +636 -494
  70. data/vendor/libgit2/include/git2/errors.h +48 -15
  71. data/vendor/libgit2/include/git2/filter.h +149 -0
  72. data/vendor/libgit2/include/git2/graph.h +14 -0
  73. data/vendor/libgit2/include/git2/index.h +95 -54
  74. data/vendor/libgit2/include/git2/indexer.h +15 -9
  75. data/vendor/libgit2/include/git2/merge.h +402 -39
  76. data/vendor/libgit2/include/git2/message.h +9 -14
  77. data/vendor/libgit2/include/git2/net.h +5 -0
  78. data/vendor/libgit2/include/git2/notes.h +6 -6
  79. data/vendor/libgit2/include/git2/object.h +34 -2
  80. data/vendor/libgit2/include/git2/odb.h +77 -14
  81. data/vendor/libgit2/include/git2/odb_backend.h +50 -6
  82. data/vendor/libgit2/include/git2/oid.h +4 -8
  83. data/vendor/libgit2/include/git2/pack.h +58 -4
  84. data/vendor/libgit2/include/git2/patch.h +277 -0
  85. data/vendor/libgit2/include/git2/pathspec.h +263 -0
  86. data/vendor/libgit2/include/git2/push.h +55 -5
  87. data/vendor/libgit2/include/git2/reflog.h +11 -8
  88. data/vendor/libgit2/include/git2/refs.h +219 -33
  89. data/vendor/libgit2/include/git2/refspec.h +3 -4
  90. data/vendor/libgit2/include/git2/remote.h +216 -76
  91. data/vendor/libgit2/include/git2/repository.h +85 -40
  92. data/vendor/libgit2/include/git2/reset.h +15 -4
  93. data/vendor/libgit2/include/git2/revert.h +86 -0
  94. data/vendor/libgit2/include/git2/revparse.h +27 -16
  95. data/vendor/libgit2/include/git2/revwalk.h +44 -6
  96. data/vendor/libgit2/include/git2/signature.h +17 -3
  97. data/vendor/libgit2/include/git2/stash.h +8 -12
  98. data/vendor/libgit2/include/git2/status.h +67 -18
  99. data/vendor/libgit2/include/git2/submodule.h +100 -85
  100. data/vendor/libgit2/include/git2/sys/commit.h +38 -4
  101. data/vendor/libgit2/include/git2/sys/config.h +44 -3
  102. data/vendor/libgit2/include/git2/sys/diff.h +91 -0
  103. data/vendor/libgit2/include/git2/sys/filter.h +301 -0
  104. data/vendor/libgit2/include/git2/sys/index.h +0 -2
  105. data/vendor/libgit2/include/git2/sys/mempack.h +85 -0
  106. data/vendor/libgit2/include/git2/sys/odb_backend.h +33 -11
  107. data/vendor/libgit2/include/git2/sys/refdb_backend.h +51 -5
  108. data/vendor/libgit2/include/git2/sys/reflog.h +21 -0
  109. data/vendor/libgit2/include/git2/sys/refs.h +2 -2
  110. data/vendor/libgit2/include/git2/sys/repository.h +19 -1
  111. data/vendor/libgit2/include/git2/transport.h +216 -75
  112. data/vendor/libgit2/include/git2/tree.h +32 -11
  113. data/vendor/libgit2/include/git2/types.h +124 -10
  114. data/vendor/libgit2/include/git2/version.h +4 -2
  115. data/vendor/libgit2/include/git2.h +41 -40
  116. data/vendor/libgit2/libgit2.pc.in +10 -0
  117. data/vendor/libgit2/src/array.h +16 -8
  118. data/vendor/libgit2/src/attr.c +140 -402
  119. data/vendor/libgit2/src/attr.h +1 -33
  120. data/vendor/libgit2/src/attr_file.c +298 -135
  121. data/vendor/libgit2/src/attr_file.h +61 -22
  122. data/vendor/libgit2/src/attrcache.c +449 -0
  123. data/vendor/libgit2/src/attrcache.h +38 -6
  124. data/vendor/libgit2/src/bitvec.h +75 -0
  125. data/vendor/libgit2/src/blame.c +488 -0
  126. data/vendor/libgit2/src/blame.h +93 -0
  127. data/vendor/libgit2/src/blame_git.c +624 -0
  128. data/vendor/libgit2/src/blame_git.h +20 -0
  129. data/vendor/libgit2/src/blob.c +137 -92
  130. data/vendor/libgit2/src/blob.h +9 -0
  131. data/vendor/libgit2/src/branch.c +175 -156
  132. data/vendor/libgit2/src/buf_text.c +29 -14
  133. data/vendor/libgit2/src/buf_text.h +5 -4
  134. data/vendor/libgit2/src/buffer.c +158 -15
  135. data/vendor/libgit2/src/buffer.h +33 -20
  136. data/vendor/libgit2/src/cc-compat.h +8 -2
  137. data/vendor/libgit2/src/checkout.c +1081 -287
  138. data/vendor/libgit2/src/checkout.h +1 -1
  139. data/vendor/libgit2/src/cherrypick.c +226 -0
  140. data/vendor/libgit2/src/clone.c +297 -245
  141. data/vendor/libgit2/src/{compress.h → clone.h} +4 -8
  142. data/vendor/libgit2/src/commit.c +313 -101
  143. data/vendor/libgit2/src/commit.h +6 -3
  144. data/vendor/libgit2/src/commit_list.c +3 -3
  145. data/vendor/libgit2/src/commit_list.h +1 -1
  146. data/vendor/libgit2/src/common.h +74 -1
  147. data/vendor/libgit2/src/config.c +570 -145
  148. data/vendor/libgit2/src/config.h +39 -6
  149. data/vendor/libgit2/src/config_cache.c +32 -18
  150. data/vendor/libgit2/src/config_file.c +711 -424
  151. data/vendor/libgit2/src/config_file.h +4 -3
  152. data/vendor/libgit2/src/crlf.c +149 -109
  153. data/vendor/libgit2/src/date.c +35 -7
  154. data/vendor/libgit2/src/delta.c +1 -1
  155. data/vendor/libgit2/src/diff.c +719 -414
  156. data/vendor/libgit2/src/diff.h +52 -7
  157. data/vendor/libgit2/src/diff_driver.c +183 -85
  158. data/vendor/libgit2/src/diff_driver.h +1 -1
  159. data/vendor/libgit2/src/diff_file.c +65 -84
  160. data/vendor/libgit2/src/diff_file.h +12 -10
  161. data/vendor/libgit2/src/diff_patch.c +274 -333
  162. data/vendor/libgit2/src/diff_patch.h +10 -9
  163. data/vendor/libgit2/src/diff_print.c +381 -179
  164. data/vendor/libgit2/src/diff_stats.c +336 -0
  165. data/vendor/libgit2/src/diff_tform.c +393 -215
  166. data/vendor/libgit2/src/diff_xdiff.c +117 -42
  167. data/vendor/libgit2/src/errors.c +59 -4
  168. data/vendor/libgit2/src/fetch.c +40 -34
  169. data/vendor/libgit2/src/fetch.h +2 -5
  170. data/vendor/libgit2/src/fetchhead.c +14 -7
  171. data/vendor/libgit2/src/filebuf.c +13 -24
  172. data/vendor/libgit2/src/filebuf.h +3 -3
  173. data/vendor/libgit2/src/fileops.c +104 -296
  174. data/vendor/libgit2/src/fileops.h +23 -94
  175. data/vendor/libgit2/src/filter.c +642 -43
  176. data/vendor/libgit2/src/filter.h +7 -66
  177. data/vendor/libgit2/src/fnmatch.c +46 -3
  178. data/vendor/libgit2/src/fnmatch.h +24 -3
  179. data/vendor/libgit2/src/global.c +158 -42
  180. data/vendor/libgit2/src/global.h +9 -0
  181. data/vendor/libgit2/src/graph.c +34 -20
  182. data/vendor/libgit2/src/hash/hash_generic.h +0 -1
  183. data/vendor/libgit2/src/hash/hash_openssl.h +0 -1
  184. data/vendor/libgit2/src/hash/hash_win32.c +14 -29
  185. data/vendor/libgit2/src/hash.h +0 -2
  186. data/vendor/libgit2/src/hashsig.c +97 -117
  187. data/vendor/libgit2/src/ident.c +125 -0
  188. data/vendor/libgit2/src/ignore.c +159 -110
  189. data/vendor/libgit2/src/ignore.h +13 -3
  190. data/vendor/libgit2/src/index.c +803 -445
  191. data/vendor/libgit2/src/index.h +43 -6
  192. data/vendor/libgit2/src/indexer.c +475 -157
  193. data/vendor/libgit2/src/iterator.c +198 -55
  194. data/vendor/libgit2/src/iterator.h +28 -4
  195. data/vendor/libgit2/src/map.h +1 -0
  196. data/vendor/libgit2/src/merge.c +849 -142
  197. data/vendor/libgit2/src/merge.h +11 -4
  198. data/vendor/libgit2/src/merge_file.c +183 -78
  199. data/vendor/libgit2/src/merge_file.h +0 -57
  200. data/vendor/libgit2/src/message.c +4 -28
  201. data/vendor/libgit2/src/mwindow.c +117 -8
  202. data/vendor/libgit2/src/mwindow.h +9 -1
  203. data/vendor/libgit2/src/netops.c +164 -59
  204. data/vendor/libgit2/src/netops.h +37 -1
  205. data/vendor/libgit2/src/notes.c +9 -18
  206. data/vendor/libgit2/src/notes.h +1 -1
  207. data/vendor/libgit2/src/object.c +78 -2
  208. data/vendor/libgit2/src/odb.c +191 -59
  209. data/vendor/libgit2/src/odb.h +2 -1
  210. data/vendor/libgit2/src/odb_loose.c +66 -51
  211. data/vendor/libgit2/src/odb_mempack.c +182 -0
  212. data/vendor/libgit2/src/odb_pack.c +151 -61
  213. data/vendor/libgit2/src/oid.c +30 -19
  214. data/vendor/libgit2/src/oid.h +13 -10
  215. data/vendor/libgit2/src/pack-objects.c +198 -147
  216. data/vendor/libgit2/src/pack-objects.h +7 -0
  217. data/vendor/libgit2/src/pack.c +272 -101
  218. data/vendor/libgit2/src/pack.h +15 -1
  219. data/vendor/libgit2/src/path.c +359 -117
  220. data/vendor/libgit2/src/path.h +110 -20
  221. data/vendor/libgit2/src/pathspec.c +583 -57
  222. data/vendor/libgit2/src/pathspec.h +36 -15
  223. data/vendor/libgit2/src/pool.c +4 -5
  224. data/vendor/libgit2/src/posix.c +45 -2
  225. data/vendor/libgit2/src/posix.h +13 -5
  226. data/vendor/libgit2/src/pqueue.c +73 -119
  227. data/vendor/libgit2/src/pqueue.h +35 -82
  228. data/vendor/libgit2/src/push.c +116 -48
  229. data/vendor/libgit2/src/push.h +5 -0
  230. data/vendor/libgit2/src/refdb.c +45 -6
  231. data/vendor/libgit2/src/refdb.h +13 -3
  232. data/vendor/libgit2/src/refdb_fs.c +1130 -551
  233. data/vendor/libgit2/src/reflog.c +36 -327
  234. data/vendor/libgit2/src/reflog.h +6 -1
  235. data/vendor/libgit2/src/refs.c +345 -142
  236. data/vendor/libgit2/src/refs.h +9 -2
  237. data/vendor/libgit2/src/refspec.c +90 -62
  238. data/vendor/libgit2/src/refspec.h +7 -24
  239. data/vendor/libgit2/src/remote.c +815 -415
  240. data/vendor/libgit2/src/remote.h +3 -4
  241. data/vendor/libgit2/src/repository.c +360 -207
  242. data/vendor/libgit2/src/repository.h +16 -10
  243. data/vendor/libgit2/src/reset.c +28 -13
  244. data/vendor/libgit2/src/revert.c +228 -0
  245. data/vendor/libgit2/src/revparse.c +29 -30
  246. data/vendor/libgit2/src/revwalk.c +141 -96
  247. data/vendor/libgit2/src/revwalk.h +6 -1
  248. data/vendor/libgit2/src/settings.c +140 -0
  249. data/vendor/libgit2/src/sha1_lookup.c +71 -0
  250. data/vendor/libgit2/src/sha1_lookup.h +5 -0
  251. data/vendor/libgit2/src/signature.c +38 -10
  252. data/vendor/libgit2/src/sortedcache.c +378 -0
  253. data/vendor/libgit2/src/sortedcache.h +178 -0
  254. data/vendor/libgit2/src/stash.c +98 -116
  255. data/vendor/libgit2/src/status.c +88 -60
  256. data/vendor/libgit2/src/status.h +2 -2
  257. data/vendor/libgit2/src/strmap.c +32 -0
  258. data/vendor/libgit2/src/strmap.h +14 -1
  259. data/vendor/libgit2/src/strnlen.h +23 -0
  260. data/vendor/libgit2/src/submodule.c +1073 -615
  261. data/vendor/libgit2/src/submodule.h +89 -21
  262. data/vendor/libgit2/src/sysdir.c +252 -0
  263. data/vendor/libgit2/src/sysdir.h +101 -0
  264. data/vendor/libgit2/src/tag.c +31 -20
  265. data/vendor/libgit2/src/thread-utils.h +98 -17
  266. data/vendor/libgit2/src/trace.h +0 -2
  267. data/vendor/libgit2/src/transport.c +76 -6
  268. data/vendor/libgit2/src/transports/cred.c +164 -61
  269. data/vendor/libgit2/src/transports/git.c +41 -48
  270. data/vendor/libgit2/src/transports/http.c +65 -109
  271. data/vendor/libgit2/src/transports/local.c +88 -65
  272. data/vendor/libgit2/src/transports/smart.c +91 -19
  273. data/vendor/libgit2/src/transports/smart.h +13 -5
  274. data/vendor/libgit2/src/transports/smart_pkt.c +24 -14
  275. data/vendor/libgit2/src/transports/smart_protocol.c +268 -113
  276. data/vendor/libgit2/src/transports/ssh.c +284 -186
  277. data/vendor/libgit2/src/transports/winhttp.c +279 -198
  278. data/vendor/libgit2/src/tree-cache.c +21 -23
  279. data/vendor/libgit2/src/tree-cache.h +1 -0
  280. data/vendor/libgit2/src/tree.c +109 -92
  281. data/vendor/libgit2/src/tree.h +2 -3
  282. data/vendor/libgit2/src/unix/map.c +7 -1
  283. data/vendor/libgit2/src/unix/posix.h +0 -1
  284. data/vendor/libgit2/src/userdiff.h +208 -0
  285. data/vendor/libgit2/src/util.c +16 -112
  286. data/vendor/libgit2/src/util.h +107 -3
  287. data/vendor/libgit2/src/vector.c +72 -17
  288. data/vendor/libgit2/src/vector.h +32 -5
  289. data/vendor/libgit2/src/win32/dir.c +15 -40
  290. data/vendor/libgit2/src/win32/dir.h +3 -2
  291. data/vendor/libgit2/src/win32/error.c +5 -31
  292. data/vendor/libgit2/src/win32/findfile.c +47 -66
  293. data/vendor/libgit2/src/win32/findfile.h +1 -12
  294. data/vendor/libgit2/src/win32/git2.rc +40 -0
  295. data/vendor/libgit2/src/win32/map.c +7 -1
  296. data/vendor/libgit2/src/win32/mingw-compat.h +3 -0
  297. data/vendor/libgit2/src/win32/posix.h +17 -11
  298. data/vendor/libgit2/src/win32/posix_w32.c +424 -292
  299. data/vendor/libgit2/src/win32/precompiled.h +6 -2
  300. data/vendor/libgit2/src/win32/pthread.c +141 -18
  301. data/vendor/libgit2/src/win32/pthread.h +50 -8
  302. data/vendor/libgit2/src/win32/reparse.h +57 -0
  303. data/vendor/libgit2/src/win32/utf-conv.c +117 -60
  304. data/vendor/libgit2/src/win32/utf-conv.h +81 -6
  305. data/vendor/libgit2/src/win32/version.h +21 -4
  306. data/vendor/libgit2/src/win32/w32_util.c +139 -0
  307. data/vendor/libgit2/src/win32/w32_util.h +54 -0
  308. data/vendor/libgit2/src/zstream.c +156 -0
  309. data/vendor/libgit2/src/zstream.h +39 -0
  310. metadata +84 -167
  311. data/Rakefile +0 -61
  312. data/ext/rugged/rugged_diff_patch.c +0 -169
  313. data/lib/rugged/diff/patch.rb +0 -28
  314. data/test/blob_test.rb +0 -341
  315. data/test/branch_test.rb +0 -199
  316. data/test/commit_test.rb +0 -104
  317. data/test/config_test.rb +0 -45
  318. data/test/coverage/cover.rb +0 -133
  319. data/test/diff_test.rb +0 -777
  320. data/test/errors_test.rb +0 -34
  321. data/test/fixtures/alternate/objects/14/6ae76773c91e3b1d00cf7a338ec55ae58297e2 +0 -0
  322. data/test/fixtures/alternate/objects/14/9c32d47e99d0a3572ff1e70a2e0051bbf347a9 +0 -0
  323. data/test/fixtures/alternate/objects/14/fb3108588f9421bf764041e5e3ac305eb6277f +0 -0
  324. data/test/fixtures/archive.tar.gz +0 -0
  325. data/test/fixtures/attr/attr0 +0 -1
  326. data/test/fixtures/attr/attr1 +0 -29
  327. data/test/fixtures/attr/attr2 +0 -21
  328. data/test/fixtures/attr/attr3 +0 -4
  329. data/test/fixtures/attr/binfile +0 -1
  330. data/test/fixtures/attr/dir/file +0 -0
  331. data/test/fixtures/attr/file +0 -1
  332. data/test/fixtures/attr/gitattributes +0 -29
  333. data/test/fixtures/attr/gitignore +0 -2
  334. data/test/fixtures/attr/ign +0 -1
  335. data/test/fixtures/attr/macro_bad +0 -1
  336. data/test/fixtures/attr/macro_test +0 -1
  337. data/test/fixtures/attr/root_test1 +0 -1
  338. data/test/fixtures/attr/root_test2 +0 -6
  339. data/test/fixtures/attr/root_test3 +0 -19
  340. data/test/fixtures/attr/root_test4.txt +0 -14
  341. data/test/fixtures/attr/sub/abc +0 -37
  342. data/test/fixtures/attr/sub/dir/file +0 -0
  343. data/test/fixtures/attr/sub/file +0 -1
  344. data/test/fixtures/attr/sub/ign/file +0 -1
  345. data/test/fixtures/attr/sub/ign/sub/file +0 -1
  346. data/test/fixtures/attr/sub/sub/dir +0 -0
  347. data/test/fixtures/attr/sub/sub/file +0 -1
  348. data/test/fixtures/attr/sub/sub/subsub.txt +0 -1
  349. data/test/fixtures/attr/sub/subdir_test1 +0 -2
  350. data/test/fixtures/attr/sub/subdir_test2.txt +0 -1
  351. data/test/fixtures/diff/another.txt +0 -38
  352. data/test/fixtures/diff/readme.txt +0 -36
  353. data/test/fixtures/mergedrepo/conflicts-one.txt +0 -5
  354. data/test/fixtures/mergedrepo/conflicts-two.txt +0 -5
  355. data/test/fixtures/mergedrepo/one.txt +0 -10
  356. data/test/fixtures/mergedrepo/two.txt +0 -12
  357. data/test/fixtures/status/current_file +0 -1
  358. data/test/fixtures/status/ignored_file +0 -1
  359. data/test/fixtures/status/modified_file +0 -2
  360. data/test/fixtures/status/new_file +0 -1
  361. data/test/fixtures/status/staged_changes +0 -2
  362. data/test/fixtures/status/staged_changes_modified_file +0 -3
  363. data/test/fixtures/status/staged_delete_modified_file +0 -1
  364. data/test/fixtures/status/staged_new_file +0 -1
  365. data/test/fixtures/status/staged_new_file_modified_file +0 -2
  366. data/test/fixtures/status/subdir/current_file +0 -1
  367. data/test/fixtures/status/subdir/modified_file +0 -2
  368. data/test/fixtures/status/subdir/new_file +0 -1
  369. data/test/fixtures/status/subdir.txt +0 -2
  370. data/test/fixtures/status//350/277/231 +0 -1
  371. data/test/fixtures/testrepo.git/HEAD +0 -1
  372. data/test/fixtures/testrepo.git/config +0 -13
  373. data/test/fixtures/testrepo.git/description +0 -1
  374. data/test/fixtures/testrepo.git/index +0 -0
  375. data/test/fixtures/testrepo.git/info/exclude +0 -6
  376. data/test/fixtures/testrepo.git/logs/HEAD +0 -3
  377. data/test/fixtures/testrepo.git/logs/refs/heads/master +0 -3
  378. data/test/fixtures/testrepo.git/logs/refs/notes/commits +0 -1
  379. data/test/fixtures/testrepo.git/objects/0c/37a5391bbff43c37f0d0371823a5509eed5b1d +0 -0
  380. data/test/fixtures/testrepo.git/objects/13/85f264afb75a56a5bec74243be9b367ba4ca08 +0 -0
  381. data/test/fixtures/testrepo.git/objects/18/1037049a54a1eb5fab404658a3a250b44335d7 +0 -0
  382. data/test/fixtures/testrepo.git/objects/18/10dff58d8a660512d4832e740f692884338ccd +0 -0
  383. data/test/fixtures/testrepo.git/objects/2d/2eff63372b08adf0a9eb84109ccf7d19e2f3a2 +0 -0
  384. data/test/fixtures/testrepo.git/objects/36/060c58702ed4c2a40832c51758d5344201d89a +0 -2
  385. data/test/fixtures/testrepo.git/objects/44/1034f860c1d5d90e4188d11ae0d325176869a8 +0 -1
  386. data/test/fixtures/testrepo.git/objects/45/b983be36b73c0788dc9cbcb76cbb80fc7bb057 +0 -0
  387. data/test/fixtures/testrepo.git/objects/4a/202b346bb0fb0db7eff3cffeb3c70babbd2045 +0 -2
  388. data/test/fixtures/testrepo.git/objects/5b/5b025afb0b4c913b4c338a42934a3863bf3644 +0 -2
  389. data/test/fixtures/testrepo.git/objects/60/d415052a33de2150bf68757f6461df4f563ae4 +0 -0
  390. data/test/fixtures/testrepo.git/objects/61/9f9935957e010c419cb9d15621916ddfcc0b96 +0 -0
  391. data/test/fixtures/testrepo.git/objects/68/8a8f4ef7496901d15322972f96e212a9e466cc +0 -1
  392. data/test/fixtures/testrepo.git/objects/75/057dd4114e74cca1d750d0aee1647c903cb60a +0 -0
  393. data/test/fixtures/testrepo.git/objects/77/71329dfa3002caf8c61a0ceb62a31d09023f37 +0 -0
  394. data/test/fixtures/testrepo.git/objects/81/4889a078c031f61ed08ab5fa863aea9314344d +0 -0
  395. data/test/fixtures/testrepo.git/objects/84/96071c1b46c854b31185ea97743be6a8774479 +0 -0
  396. data/test/fixtures/testrepo.git/objects/94/eca2de348d5f672faf56b0decafa5937e3235e +0 -0
  397. data/test/fixtures/testrepo.git/objects/9b/7384fe1676186192842f5d3e129457b62db9e3 +0 -0
  398. data/test/fixtures/testrepo.git/objects/9f/d738e8f7967c078dceed8190330fc8648ee56a +0 -3
  399. data/test/fixtures/testrepo.git/objects/a4/a7dce85cf63874e984719f4fdd239f5145052f +0 -2
  400. data/test/fixtures/testrepo.git/objects/a7/1586c1dfe8a71c6cbf6c129f404c5642ff31bd +0 -0
  401. data/test/fixtures/testrepo.git/objects/a8/233120f6ad708f843d861ce2b7228ec4e3dec6 +0 -0
  402. data/test/fixtures/testrepo.git/objects/b7/4713326bc972cc15751ed504dca6f6f3b91f7a +0 -3
  403. data/test/fixtures/testrepo.git/objects/be/3563ae3f795b2b4353bcce3a527ad0a4f7f644 +0 -3
  404. data/test/fixtures/testrepo.git/objects/c4/7800c7266a2be04c571c04d5a6614691ea99bd +0 -3
  405. data/test/fixtures/testrepo.git/objects/c4/dc1555e4d4fa0e0c9c3fc46734c7c35b3ce90b +0 -0
  406. data/test/fixtures/testrepo.git/objects/e6/9de29bb2d1d6434b8b29ae775ad8c2e48c5391 +0 -0
  407. data/test/fixtures/testrepo.git/objects/f6/0079018b664e4e79329a7ef9559c8d9e0378d1 +0 -0
  408. data/test/fixtures/testrepo.git/objects/fa/49b077972391ad58037050f2a75f74e3671e92 +0 -0
  409. data/test/fixtures/testrepo.git/objects/fd/093bff70906175335656e6ce6ae05783708765 +0 -0
  410. data/test/fixtures/testrepo.git/objects/pack/pack-d7c6adf9f61318f041845b01440d09aa7a91e1b5.idx +0 -0
  411. data/test/fixtures/testrepo.git/objects/pack/pack-d7c6adf9f61318f041845b01440d09aa7a91e1b5.pack +0 -0
  412. data/test/fixtures/testrepo.git/packed-refs +0 -2
  413. data/test/fixtures/testrepo.git/refs/heads/master +0 -1
  414. data/test/fixtures/testrepo.git/refs/notes/commits +0 -1
  415. data/test/fixtures/testrepo.git/refs/tags/v0.9 +0 -1
  416. data/test/fixtures/testrepo.git/refs/tags/v1.0 +0 -1
  417. data/test/fixtures/text_file.md +0 -464
  418. data/test/fixtures/unsymlinked.git/HEAD +0 -1
  419. data/test/fixtures/unsymlinked.git/config +0 -6
  420. data/test/fixtures/unsymlinked.git/description +0 -1
  421. data/test/fixtures/unsymlinked.git/info/exclude +0 -2
  422. data/test/fixtures/unsymlinked.git/objects/08/8b64704e0d6b8bd061dea879418cb5442a3fbf +0 -0
  423. data/test/fixtures/unsymlinked.git/objects/13/a5e939bca25940c069fd2169d993dba328e30b +0 -0
  424. data/test/fixtures/unsymlinked.git/objects/19/bf568e59e3a0b363cafb4106226e62d4a4c41c +0 -0
  425. data/test/fixtures/unsymlinked.git/objects/58/1fadd35b4cf320d102a152f918729011604773 +0 -0
  426. data/test/fixtures/unsymlinked.git/objects/5c/87b6791e8b13da658a14d1ef7e09b5dc3bac8c +0 -0
  427. data/test/fixtures/unsymlinked.git/objects/6f/e5f5398af85fb3de8a6aba0339b6d3bfa26a27 +0 -0
  428. data/test/fixtures/unsymlinked.git/objects/7f/ccd75616ec188b8f1b23d67506a334cc34a49d +0 -0
  429. data/test/fixtures/unsymlinked.git/objects/80/6999882bf91d24241e4077906b9017605eb1f3 +0 -0
  430. data/test/fixtures/unsymlinked.git/objects/83/7d176303c5005505ec1e4a30231c40930c0230 +0 -0
  431. data/test/fixtures/unsymlinked.git/objects/a8/595ccca04f40818ae0155c8f9c77a230e597b6 +0 -2
  432. data/test/fixtures/unsymlinked.git/objects/cf/8f1cf5cce859c438d6cc067284cb5e161206e7 +0 -0
  433. data/test/fixtures/unsymlinked.git/objects/d5/278d05c8607ec420bfee4cf219fbc0eeebfd6a +0 -0
  434. data/test/fixtures/unsymlinked.git/objects/f4/e16fb76536591a41454194058d048d8e4dd2e9 +0 -0
  435. data/test/fixtures/unsymlinked.git/objects/f9/e65619d93fdf2673882e0a261c5e93b1a84006 +0 -0
  436. data/test/fixtures/unsymlinked.git/refs/heads/exe-file +0 -1
  437. data/test/fixtures/unsymlinked.git/refs/heads/master +0 -1
  438. data/test/fixtures/unsymlinked.git/refs/heads/reg-file +0 -1
  439. data/test/index_test.rb +0 -333
  440. data/test/lib_test.rb +0 -127
  441. data/test/note_test.rb +0 -158
  442. data/test/object_test.rb +0 -43
  443. data/test/reference_test.rb +0 -207
  444. data/test/remote_test.rb +0 -324
  445. data/test/repo_pack_test.rb +0 -24
  446. data/test/repo_reset_test.rb +0 -82
  447. data/test/repo_test.rb +0 -402
  448. data/test/tag_test.rb +0 -68
  449. data/test/test_helper.rb +0 -92
  450. data/test/tree_test.rb +0 -91
  451. data/test/walker_test.rb +0 -88
  452. data/vendor/libgit2/src/amiga/map.c +0 -48
  453. data/vendor/libgit2/src/compress.c +0 -53
@@ -6,7 +6,7 @@
6
6
  */
7
7
 
8
8
  #include "common.h"
9
- #include "fileops.h"
9
+ #include "sysdir.h"
10
10
  #include "config.h"
11
11
  #include "git2/config.h"
12
12
  #include "git2/sys/config.h"
@@ -137,6 +137,38 @@ int git_config_open_ondisk(git_config **out, const char *path)
137
137
  return error;
138
138
  }
139
139
 
140
+ int git_config_snapshot(git_config **out, git_config *in)
141
+ {
142
+ int error = 0;
143
+ size_t i;
144
+ file_internal *internal;
145
+ git_config *config;
146
+
147
+ *out = NULL;
148
+
149
+ if (git_config_new(&config) < 0)
150
+ return -1;
151
+
152
+ git_vector_foreach(&in->files, i, internal) {
153
+ git_config_backend *b;
154
+
155
+ if ((error = internal->file->snapshot(&b, internal->file)) < 0)
156
+ break;
157
+
158
+ if ((error = git_config_add_backend(config, b, internal->level, 0)) < 0) {
159
+ b->free(b);
160
+ break;
161
+ }
162
+ }
163
+
164
+ if (error < 0)
165
+ git_config_free(config);
166
+ else
167
+ *out = config;
168
+
169
+ return error;
170
+ }
171
+
140
172
  static int find_internal_file_by_level(
141
173
  file_internal **internal_out,
142
174
  const git_config *cfg,
@@ -315,30 +347,239 @@ int git_config_refresh(git_config *cfg)
315
347
  * Loop over all the variables
316
348
  */
317
349
 
350
+ typedef struct {
351
+ git_config_iterator parent;
352
+ git_config_iterator *current;
353
+ const git_config *cfg;
354
+ regex_t regex;
355
+ int has_regex;
356
+ size_t i;
357
+ } all_iter;
358
+
359
+ static int find_next_backend(size_t *out, const git_config *cfg, size_t i)
360
+ {
361
+ file_internal *internal;
362
+
363
+ for (; i > 0; --i) {
364
+ internal = git_vector_get(&cfg->files, i - 1);
365
+ if (!internal || !internal->file)
366
+ continue;
367
+
368
+ *out = i;
369
+ return 0;
370
+ }
371
+
372
+ return -1;
373
+ }
374
+
375
+ static int all_iter_next(git_config_entry **entry, git_config_iterator *_iter)
376
+ {
377
+ all_iter *iter = (all_iter *) _iter;
378
+ file_internal *internal;
379
+ git_config_backend *backend;
380
+ size_t i;
381
+ int error = 0;
382
+
383
+ if (iter->current != NULL &&
384
+ (error = iter->current->next(entry, iter->current)) == 0) {
385
+ return 0;
386
+ }
387
+
388
+ if (error < 0 && error != GIT_ITEROVER)
389
+ return error;
390
+
391
+ do {
392
+ if (find_next_backend(&i, iter->cfg, iter->i) < 0)
393
+ return GIT_ITEROVER;
394
+
395
+ internal = git_vector_get(&iter->cfg->files, i - 1);
396
+ backend = internal->file;
397
+ iter->i = i - 1;
398
+
399
+ if (iter->current)
400
+ iter->current->free(iter->current);
401
+
402
+ iter->current = NULL;
403
+ error = backend->iterator(&iter->current, backend);
404
+ if (error == GIT_ENOTFOUND)
405
+ continue;
406
+
407
+ if (error < 0)
408
+ return error;
409
+
410
+ error = iter->current->next(entry, iter->current);
411
+ /* If this backend is empty, then keep going */
412
+ if (error == GIT_ITEROVER)
413
+ continue;
414
+
415
+ return error;
416
+
417
+ } while(1);
418
+
419
+ return GIT_ITEROVER;
420
+ }
421
+
422
+ static int all_iter_glob_next(git_config_entry **entry, git_config_iterator *_iter)
423
+ {
424
+ int error;
425
+ all_iter *iter = (all_iter *) _iter;
426
+
427
+ /*
428
+ * We use the "normal" function to grab the next one across
429
+ * backends and then apply the regex
430
+ */
431
+ while ((error = all_iter_next(entry, _iter)) == 0) {
432
+ /* skip non-matching keys if regexp was provided */
433
+ if (regexec(&iter->regex, (*entry)->name, 0, NULL, 0) != 0)
434
+ continue;
435
+
436
+ /* and simply return if we like the entry's name */
437
+ return 0;
438
+ }
439
+
440
+ return error;
441
+ }
442
+
443
+ static void all_iter_free(git_config_iterator *_iter)
444
+ {
445
+ all_iter *iter = (all_iter *) _iter;
446
+
447
+ if (iter->current)
448
+ iter->current->free(iter->current);
449
+
450
+ git__free(iter);
451
+ }
452
+
453
+ static void all_iter_glob_free(git_config_iterator *_iter)
454
+ {
455
+ all_iter *iter = (all_iter *) _iter;
456
+
457
+ regfree(&iter->regex);
458
+ all_iter_free(_iter);
459
+ }
460
+
461
+ int git_config_iterator_new(git_config_iterator **out, const git_config *cfg)
462
+ {
463
+ all_iter *iter;
464
+
465
+ iter = git__calloc(1, sizeof(all_iter));
466
+ GITERR_CHECK_ALLOC(iter);
467
+
468
+ iter->parent.free = all_iter_free;
469
+ iter->parent.next = all_iter_next;
470
+
471
+ iter->i = cfg->files.length;
472
+ iter->cfg = cfg;
473
+
474
+ *out = (git_config_iterator *) iter;
475
+
476
+ return 0;
477
+ }
478
+
479
+ int git_config_iterator_glob_new(git_config_iterator **out, const git_config *cfg, const char *regexp)
480
+ {
481
+ all_iter *iter;
482
+ int result;
483
+
484
+ if (regexp == NULL)
485
+ return git_config_iterator_new(out, cfg);
486
+
487
+ iter = git__calloc(1, sizeof(all_iter));
488
+ GITERR_CHECK_ALLOC(iter);
489
+
490
+ if ((result = regcomp(&iter->regex, regexp, REG_EXTENDED)) < 0) {
491
+ giterr_set_regex(&iter->regex, result);
492
+ regfree(&iter->regex);
493
+ git__free(iter);
494
+ return -1;
495
+ }
496
+
497
+ iter->parent.next = all_iter_glob_next;
498
+ iter->parent.free = all_iter_glob_free;
499
+ iter->i = cfg->files.length;
500
+ iter->cfg = cfg;
501
+
502
+ *out = (git_config_iterator *) iter;
503
+
504
+ return 0;
505
+ }
506
+
318
507
  int git_config_foreach(
319
508
  const git_config *cfg, git_config_foreach_cb cb, void *payload)
320
509
  {
321
510
  return git_config_foreach_match(cfg, NULL, cb, payload);
322
511
  }
323
512
 
513
+ int git_config_backend_foreach_match(
514
+ git_config_backend *backend,
515
+ const char *regexp,
516
+ git_config_foreach_cb cb,
517
+ void *payload)
518
+ {
519
+ git_config_entry *entry;
520
+ git_config_iterator* iter;
521
+ regex_t regex;
522
+ int error = 0;
523
+
524
+ if (regexp != NULL) {
525
+ if ((error = regcomp(&regex, regexp, REG_EXTENDED)) < 0) {
526
+ giterr_set_regex(&regex, error);
527
+ regfree(&regex);
528
+ return -1;
529
+ }
530
+ }
531
+
532
+ if ((error = backend->iterator(&iter, backend)) < 0) {
533
+ iter = NULL;
534
+ return -1;
535
+ }
536
+
537
+ while (!(iter->next(&entry, iter) < 0)) {
538
+ /* skip non-matching keys if regexp was provided */
539
+ if (regexp && regexec(&regex, entry->name, 0, NULL, 0) != 0)
540
+ continue;
541
+
542
+ /* abort iterator on non-zero return value */
543
+ if ((error = cb(entry, payload)) != 0) {
544
+ giterr_set_after_callback(error);
545
+ break;
546
+ }
547
+ }
548
+
549
+ if (regexp != NULL)
550
+ regfree(&regex);
551
+
552
+ iter->free(iter);
553
+
554
+ return error;
555
+ }
556
+
324
557
  int git_config_foreach_match(
325
558
  const git_config *cfg,
326
559
  const char *regexp,
327
560
  git_config_foreach_cb cb,
328
561
  void *payload)
329
562
  {
330
- int ret = 0;
331
- size_t i;
332
- file_internal *internal;
333
- git_config_backend *file;
563
+ int error;
564
+ git_config_iterator *iter;
565
+ git_config_entry *entry;
334
566
 
335
- for (i = 0; i < cfg->files.length && ret == 0; ++i) {
336
- internal = git_vector_get(&cfg->files, i);
337
- file = internal->file;
338
- ret = file->foreach(file, regexp, cb, payload);
567
+ if ((error = git_config_iterator_glob_new(&iter, cfg, regexp)) < 0)
568
+ return error;
569
+
570
+ while (!(error = git_config_next(&entry, iter))) {
571
+ if ((error = cb(entry, payload)) != 0) {
572
+ giterr_set_after_callback(error);
573
+ break;
574
+ }
339
575
  }
340
576
 
341
- return ret;
577
+ git_config_iterator_free(iter);
578
+
579
+ if (error == GIT_ITEROVER)
580
+ error = 0;
581
+
582
+ return error;
342
583
  }
343
584
 
344
585
  /**************
@@ -406,9 +647,112 @@ int git_config_set_string(git_config *cfg, const char *name, const char *value)
406
647
  return error;
407
648
  }
408
649
 
650
+ int git_config__update_entry(
651
+ git_config *config,
652
+ const char *key,
653
+ const char *value,
654
+ bool overwrite_existing,
655
+ bool only_if_existing)
656
+ {
657
+ int error = 0;
658
+ const git_config_entry *ce = NULL;
659
+
660
+ if ((error = git_config__lookup_entry(&ce, config, key, false)) < 0)
661
+ return error;
662
+
663
+ if (!ce && only_if_existing) /* entry doesn't exist */
664
+ return 0;
665
+ if (ce && !overwrite_existing) /* entry would be overwritten */
666
+ return 0;
667
+ if (value && ce && ce->value && !strcmp(ce->value, value)) /* no change */
668
+ return 0;
669
+ if (!value && (!ce || !ce->value)) /* asked to delete absent entry */
670
+ return 0;
671
+
672
+ if (!value)
673
+ error = git_config_delete_entry(config, key);
674
+ else
675
+ error = git_config_set_string(config, key, value);
676
+
677
+ return error;
678
+ }
679
+
409
680
  /***********
410
681
  * Getters
411
682
  ***********/
683
+
684
+ static int config_error_notfound(const char *name)
685
+ {
686
+ giterr_set(GITERR_CONFIG, "Config value '%s' was not found", name);
687
+ return GIT_ENOTFOUND;
688
+ }
689
+
690
+ enum {
691
+ GET_ALL_ERRORS = 0,
692
+ GET_NO_MISSING = 1,
693
+ GET_NO_ERRORS = 2
694
+ };
695
+
696
+ static int get_entry(
697
+ const git_config_entry **out,
698
+ const git_config *cfg,
699
+ const char *name,
700
+ bool normalize_name,
701
+ int want_errors)
702
+ {
703
+ int res = GIT_ENOTFOUND;
704
+ const char *key = name;
705
+ char *normalized = NULL;
706
+ size_t i;
707
+ file_internal *internal;
708
+
709
+ *out = NULL;
710
+
711
+ if (normalize_name) {
712
+ if ((res = git_config__normalize_name(name, &normalized)) < 0)
713
+ goto cleanup;
714
+ key = normalized;
715
+ }
716
+
717
+ res = GIT_ENOTFOUND;
718
+ git_vector_foreach(&cfg->files, i, internal) {
719
+ if (!internal || !internal->file)
720
+ continue;
721
+
722
+ res = internal->file->get(internal->file, key, out);
723
+ if (res != GIT_ENOTFOUND)
724
+ break;
725
+ }
726
+
727
+ git__free(normalized);
728
+
729
+ cleanup:
730
+ if (res == GIT_ENOTFOUND)
731
+ res = (want_errors > GET_ALL_ERRORS) ? 0 : config_error_notfound(name);
732
+ else if (res && (want_errors == GET_NO_ERRORS)) {
733
+ giterr_clear();
734
+ res = 0;
735
+ }
736
+
737
+ return res;
738
+ }
739
+
740
+ int git_config_get_entry(
741
+ const git_config_entry **out, const git_config *cfg, const char *name)
742
+ {
743
+ return get_entry(out, cfg, name, true, GET_ALL_ERRORS);
744
+ }
745
+
746
+ int git_config__lookup_entry(
747
+ const git_config_entry **out,
748
+ const git_config *cfg,
749
+ const char *key,
750
+ bool no_errors)
751
+ {
752
+ return get_entry(
753
+ out, cfg, key, false, no_errors ? GET_NO_ERRORS : GET_NO_MISSING);
754
+ }
755
+
412
756
  int git_config_get_mapped(
413
757
  int *out,
414
758
  const git_config *cfg,
@@ -416,143 +760,202 @@ int git_config_get_mapped(
416
760
  const git_cvar_map *maps,
417
761
  size_t map_n)
418
762
  {
419
- const char *value;
763
+ const git_config_entry *entry;
420
764
  int ret;
421
765
 
422
- if ((ret = git_config_get_string(&value, cfg, name)) < 0)
766
+ if ((ret = get_entry(&entry, cfg, name, true, GET_ALL_ERRORS)) < 0)
423
767
  return ret;
424
768
 
425
- return git_config_lookup_map_value(out, maps, map_n, value);
769
+ return git_config_lookup_map_value(out, maps, map_n, entry->value);
426
770
  }
427
771
 
428
772
  int git_config_get_int64(int64_t *out, const git_config *cfg, const char *name)
429
773
  {
430
- const char *value;
774
+ const git_config_entry *entry;
431
775
  int ret;
432
776
 
433
- if ((ret = git_config_get_string(&value, cfg, name)) < 0)
777
+ if ((ret = get_entry(&entry, cfg, name, true, GET_ALL_ERRORS)) < 0)
434
778
  return ret;
435
779
 
436
- return git_config_parse_int64(out, value);
780
+ return git_config_parse_int64(out, entry->value);
437
781
  }
438
782
 
439
783
  int git_config_get_int32(int32_t *out, const git_config *cfg, const char *name)
440
784
  {
441
- const char *value;
785
+ const git_config_entry *entry;
442
786
  int ret;
443
787
 
444
- if ((ret = git_config_get_string(&value, cfg, name)) < 0)
788
+ if ((ret = get_entry(&entry, cfg, name, true, GET_ALL_ERRORS)) < 0)
445
789
  return ret;
446
790
 
447
- return git_config_parse_int32(out, value);
791
+ return git_config_parse_int32(out, entry->value);
448
792
  }
449
793
 
450
- static int get_string_at_file(const char **out, const git_config_backend *file, const char *name)
794
+ int git_config_get_bool(int *out, const git_config *cfg, const char *name)
451
795
  {
452
796
  const git_config_entry *entry;
453
- int res;
797
+ int ret;
454
798
 
455
- res = file->get(file, name, &entry);
456
- if (!res)
457
- *out = entry->value;
799
+ if ((ret = get_entry(&entry, cfg, name, true, GET_ALL_ERRORS)) < 0)
800
+ return ret;
458
801
 
459
- return res;
802
+ return git_config_parse_bool(out, entry->value);
460
803
  }
461
804
 
462
- static int config_error_notfound(const char *name)
805
+ int git_config_get_string(
806
+ const char **out, const git_config *cfg, const char *name)
463
807
  {
464
- giterr_set(GITERR_CONFIG, "Config value '%s' was not found", name);
465
- return GIT_ENOTFOUND;
808
+ const git_config_entry *entry;
809
+ int ret = get_entry(&entry, cfg, name, true, GET_ALL_ERRORS);
810
+ *out = !ret ? (entry->value ? entry->value : "") : NULL;
811
+ return ret;
466
812
  }
467
813
 
468
- static int get_string(const char **out, const git_config *cfg, const char *name)
814
+ const char *git_config__get_string_force(
815
+ const git_config *cfg, const char *key, const char *fallback_value)
469
816
  {
470
- file_internal *internal;
471
- unsigned int i;
472
- int res;
817
+ const git_config_entry *entry;
818
+ get_entry(&entry, cfg, key, false, GET_NO_ERRORS);
819
+ return (entry && entry->value) ? entry->value : fallback_value;
820
+ }
473
821
 
474
- git_vector_foreach(&cfg->files, i, internal) {
475
- if (!internal || !internal->file)
476
- continue;
822
+ int git_config__get_bool_force(
823
+ const git_config *cfg, const char *key, int fallback_value)
824
+ {
825
+ int val = fallback_value;
826
+ const git_config_entry *entry;
477
827
 
478
- res = get_string_at_file(out, internal->file, name);
479
- if (res != GIT_ENOTFOUND)
480
- return res;
481
- }
828
+ get_entry(&entry, cfg, key, false, GET_NO_ERRORS);
829
+
830
+ if (entry && git_config_parse_bool(&val, entry->value) < 0)
831
+ giterr_clear();
482
832
 
483
- return config_error_notfound(name);
833
+ return val;
484
834
  }
485
835
 
486
- int git_config_get_bool(int *out, const git_config *cfg, const char *name)
836
+ int git_config__get_int_force(
837
+ const git_config *cfg, const char *key, int fallback_value)
487
838
  {
488
- const char *value = NULL;
489
- int ret;
839
+ int32_t val = (int32_t)fallback_value;
840
+ const git_config_entry *entry;
490
841
 
491
- if ((ret = get_string(&value, cfg, name)) < 0)
492
- return ret;
842
+ get_entry(&entry, cfg, key, false, GET_NO_ERRORS);
843
+
844
+ if (entry && git_config_parse_int32(&val, entry->value) < 0)
845
+ giterr_clear();
493
846
 
494
- return git_config_parse_bool(out, value);
847
+ return (int)val;
495
848
  }
496
849
 
497
- int git_config_get_string(const char **out, const git_config *cfg, const char *name)
850
+ int git_config_get_multivar_foreach(
851
+ const git_config *cfg, const char *name, const char *regexp,
852
+ git_config_foreach_cb cb, void *payload)
498
853
  {
499
- int ret;
500
- const char *str = NULL;
854
+ int err, found;
855
+ git_config_iterator *iter;
856
+ git_config_entry *entry;
501
857
 
502
- if ((ret = get_string(&str, cfg, name)) < 0)
503
- return ret;
858
+ if ((err = git_config_multivar_iterator_new(&iter, cfg, name, regexp)) < 0)
859
+ return err;
504
860
 
505
- *out = str == NULL ? "" : str;
506
- return 0;
861
+ found = 0;
862
+ while ((err = iter->next(&entry, iter)) == 0) {
863
+ found = 1;
864
+
865
+ if ((err = cb(entry, payload)) != 0) {
866
+ giterr_set_after_callback(err);
867
+ break;
868
+ }
869
+ }
870
+
871
+ iter->free(iter);
872
+ if (err == GIT_ITEROVER)
873
+ err = 0;
874
+
875
+ if (found == 0 && err == 0)
876
+ err = config_error_notfound(name);
877
+
878
+ return err;
507
879
  }
508
880
 
509
- int git_config_get_entry(const git_config_entry **out, const git_config *cfg, const char *name)
881
+ typedef struct {
882
+ git_config_iterator parent;
883
+ git_config_iterator *iter;
884
+ char *name;
885
+ regex_t regex;
886
+ int have_regex;
887
+ } multivar_iter;
888
+
889
+ static int multivar_iter_next(git_config_entry **entry, git_config_iterator *_iter)
510
890
  {
511
- file_internal *internal;
512
- unsigned int i;
513
- git_config_backend *file;
514
- int ret;
515
-
516
- *out = NULL;
891
+ multivar_iter *iter = (multivar_iter *) _iter;
892
+ int error = 0;
517
893
 
518
- git_vector_foreach(&cfg->files, i, internal) {
519
- if (!internal || !internal->file)
894
+ while ((error = iter->iter->next(entry, iter->iter)) == 0) {
895
+ if (git__strcmp(iter->name, (*entry)->name))
520
896
  continue;
521
- file = internal->file;
522
897
 
523
- ret = file->get(file, name, out);
524
- if (ret != GIT_ENOTFOUND)
525
- return ret;
898
+ if (!iter->have_regex)
899
+ return 0;
900
+
901
+ if (regexec(&iter->regex, (*entry)->value, 0, NULL, 0) == 0)
902
+ return 0;
526
903
  }
527
904
 
528
- return config_error_notfound(name);
905
+ return error;
529
906
  }
530
907
 
531
- int git_config_get_multivar(
532
- const git_config *cfg, const char *name, const char *regexp,
533
- git_config_foreach_cb cb, void *payload)
908
+ void multivar_iter_free(git_config_iterator *_iter)
534
909
  {
535
- file_internal *internal;
536
- git_config_backend *file;
537
- int ret = GIT_ENOTFOUND;
538
- size_t i;
910
+ multivar_iter *iter = (multivar_iter *) _iter;
539
911
 
540
- /*
541
- * This loop runs the "wrong" way 'round because we need to
542
- * look at every value from the most general to most specific
543
- */
544
- for (i = cfg->files.length; i > 0; --i) {
545
- internal = git_vector_get(&cfg->files, i - 1);
546
- if (!internal || !internal->file)
547
- continue;
548
- file = internal->file;
912
+ iter->iter->free(iter->iter);
913
+
914
+ git__free(iter->name);
915
+ regfree(&iter->regex);
916
+ git__free(iter);
917
+ }
918
+
919
+ int git_config_multivar_iterator_new(git_config_iterator **out, const git_config *cfg, const char *name, const char *regexp)
920
+ {
921
+ multivar_iter *iter = NULL;
922
+ git_config_iterator *inner = NULL;
923
+ int error;
924
+
925
+ if ((error = git_config_iterator_new(&inner, cfg)) < 0)
926
+ return error;
927
+
928
+ iter = git__calloc(1, sizeof(multivar_iter));
929
+ GITERR_CHECK_ALLOC(iter);
930
+
931
+ if ((error = git_config__normalize_name(name, &iter->name)) < 0)
932
+ goto on_error;
549
933
 
550
- ret = file->get_multivar(file, name, regexp, cb, payload);
551
- if (ret < 0 && ret != GIT_ENOTFOUND)
552
- return ret;
934
+ if (regexp != NULL) {
935
+ error = regcomp(&iter->regex, regexp, REG_EXTENDED);
936
+ if (error < 0) {
937
+ giterr_set_regex(&iter->regex, error);
938
+ error = -1;
939
+ regfree(&iter->regex);
940
+ goto on_error;
941
+ }
942
+
943
+ iter->have_regex = 1;
553
944
  }
554
945
 
555
- return (ret == GIT_ENOTFOUND) ? config_error_notfound(name) : 0;
946
+ iter->iter = inner;
947
+ iter->parent.free = multivar_iter_free;
948
+ iter->parent.next = multivar_iter_next;
949
+
950
+ *out = (git_config_iterator *) iter;
951
+
952
+ return 0;
953
+
954
+ on_error:
955
+
956
+ inner->free(inner);
957
+ git__free(iter);
958
+ return error;
556
959
  }
557
960
 
558
961
  int git_config_set_multivar(git_config *cfg, const char *name, const char *regexp, const char *value)
@@ -568,83 +971,70 @@ int git_config_set_multivar(git_config *cfg, const char *name, const char *regex
568
971
  return file->set_multivar(file, name, regexp, value);
569
972
  }
570
973
 
571
- static int git_config__find_file_to_path(
572
- char *out, size_t outlen, int (*find)(git_buf *buf))
974
+ int git_config_delete_multivar(git_config *cfg, const char *name, const char *regexp)
573
975
  {
574
- int error = 0;
575
- git_buf path = GIT_BUF_INIT;
576
-
577
- if ((error = find(&path)) < 0)
578
- goto done;
579
-
580
- if (path.size >= outlen) {
581
- giterr_set(GITERR_NOMEMORY, "Buffer is too short for the path");
582
- error = GIT_EBUFS;
583
- goto done;
584
- }
976
+ git_config_backend *file;
977
+ file_internal *internal;
585
978
 
586
- git_buf_copy_cstr(out, outlen, &path);
979
+ internal = git_vector_get(&cfg->files, 0);
980
+ if (!internal || !internal->file)
981
+ return config_error_nofiles(name);
982
+ file = internal->file;
587
983
 
588
- done:
589
- git_buf_free(&path);
590
- return error;
984
+ return file->del_multivar(file, name, regexp);
591
985
  }
592
986
 
593
- int git_config_find_global_r(git_buf *path)
987
+ int git_config_next(git_config_entry **entry, git_config_iterator *iter)
594
988
  {
595
- return git_futils_find_global_file(path, GIT_CONFIG_FILENAME_GLOBAL);
989
+ return iter->next(entry, iter);
596
990
  }
597
991
 
598
- int git_config_find_global(char *global_config_path, size_t length)
992
+ void git_config_iterator_free(git_config_iterator *iter)
599
993
  {
600
- return git_config__find_file_to_path(
601
- global_config_path, length, git_config_find_global_r);
602
- }
994
+ if (iter == NULL)
995
+ return;
603
996
 
604
- int git_config_find_xdg_r(git_buf *path)
605
- {
606
- return git_futils_find_xdg_file(path, GIT_CONFIG_FILENAME_XDG);
997
+ iter->free(iter);
607
998
  }
608
999
 
609
- int git_config_find_xdg(char *xdg_config_path, size_t length)
1000
+ int git_config_find_global(git_buf *path)
610
1001
  {
611
- return git_config__find_file_to_path(
612
- xdg_config_path, length, git_config_find_xdg_r);
1002
+ git_buf_sanitize(path);
1003
+ return git_sysdir_find_global_file(path, GIT_CONFIG_FILENAME_GLOBAL);
613
1004
  }
614
1005
 
615
- int git_config_find_system_r(git_buf *path)
1006
+ int git_config_find_xdg(git_buf *path)
616
1007
  {
617
- return git_futils_find_system_file(path, GIT_CONFIG_FILENAME_SYSTEM);
1008
+ git_buf_sanitize(path);
1009
+ return git_sysdir_find_xdg_file(path, GIT_CONFIG_FILENAME_XDG);
618
1010
  }
619
1011
 
620
- int git_config_find_system(char *system_config_path, size_t length)
1012
+ int git_config_find_system(git_buf *path)
621
1013
  {
622
- return git_config__find_file_to_path(
623
- system_config_path, length, git_config_find_system_r);
1014
+ git_buf_sanitize(path);
1015
+ return git_sysdir_find_system_file(path, GIT_CONFIG_FILENAME_SYSTEM);
624
1016
  }
625
1017
 
626
1018
  int git_config__global_location(git_buf *buf)
627
1019
  {
628
1020
  const git_buf *paths;
629
1021
  const char *sep, *start;
630
- size_t len;
631
1022
 
632
- if (git_futils_dirs_get(&paths, GIT_FUTILS_DIR_GLOBAL) < 0)
1023
+ if (git_sysdir_get(&paths, GIT_SYSDIR_GLOBAL) < 0)
633
1024
  return -1;
634
1025
 
635
1026
  /* no paths, so give up */
636
- if (git_buf_len(paths) == 0)
1027
+ if (!paths || !git_buf_len(paths))
637
1028
  return -1;
638
1029
 
639
- start = git_buf_cstr(paths);
640
- sep = strchr(start, GIT_PATH_LIST_SEPARATOR);
641
-
642
- if (sep)
643
- len = sep - start;
644
- else
645
- len = paths->size;
1030
+ /* find unescaped separator or end of string */
1031
+ for (sep = start = git_buf_cstr(paths); *sep; ++sep) {
1032
+ if (*sep == GIT_PATH_LIST_SEPARATOR &&
1033
+ (sep <= start || sep[-1] != '\\'))
1034
+ break;
1035
+ }
646
1036
 
647
- if (git_buf_set(buf, start, len) < 0)
1037
+ if (git_buf_set(buf, start, (size_t)(sep - start)) < 0)
648
1038
  return -1;
649
1039
 
650
1040
  return git_buf_joinpath(buf, buf->ptr, GIT_CONFIG_FILENAME_GLOBAL);
@@ -659,16 +1049,16 @@ int git_config_open_default(git_config **out)
659
1049
  if ((error = git_config_new(&cfg)) < 0)
660
1050
  return error;
661
1051
 
662
- if (!git_config_find_global_r(&buf) || !git_config__global_location(&buf)) {
1052
+ if (!git_config_find_global(&buf) || !git_config__global_location(&buf)) {
663
1053
  error = git_config_add_file_ondisk(cfg, buf.ptr,
664
1054
  GIT_CONFIG_LEVEL_GLOBAL, 0);
665
1055
  }
666
1056
 
667
- if (!error && !git_config_find_xdg_r(&buf))
1057
+ if (!error && !git_config_find_xdg(&buf))
668
1058
  error = git_config_add_file_ondisk(cfg, buf.ptr,
669
1059
  GIT_CONFIG_LEVEL_XDG, 0);
670
1060
 
671
- if (!error && !git_config_find_system_r(&buf))
1061
+ if (!error && !git_config_find_system(&buf))
672
1062
  error = git_config_add_file_ondisk(cfg, buf.ptr,
673
1063
  GIT_CONFIG_LEVEL_SYSTEM, 0);
674
1064
 
@@ -753,7 +1143,7 @@ int git_config_parse_int64(int64_t *out, const char *value)
753
1143
  const char *num_end;
754
1144
  int64_t num;
755
1145
 
756
- if (git__strtol64(&num, value, &num_end, 0) < 0)
1146
+ if (!value || git__strtol64(&num, value, &num_end, 0) < 0)
757
1147
  goto fail_parse;
758
1148
 
759
1149
  switch (*num_end) {
@@ -787,7 +1177,7 @@ int git_config_parse_int64(int64_t *out, const char *value)
787
1177
  }
788
1178
 
789
1179
  fail_parse:
790
- giterr_set(GITERR_CONFIG, "Failed to parse '%s' as an integer", value);
1180
+ giterr_set(GITERR_CONFIG, "Failed to parse '%s' as an integer", value ? value : "(null)");
791
1181
  return -1;
792
1182
  }
793
1183
 
@@ -807,15 +1197,49 @@ int git_config_parse_int32(int32_t *out, const char *value)
807
1197
  return 0;
808
1198
 
809
1199
  fail_parse:
810
- giterr_set(GITERR_CONFIG, "Failed to parse '%s' as a 32-bit integer", value);
1200
+ giterr_set(GITERR_CONFIG, "Failed to parse '%s' as a 32-bit integer", value ? value : "(null)");
811
1201
  return -1;
812
1202
  }
813
1203
 
1204
+ /* Take something the user gave us and make it nice for our hash function */
1205
+ int git_config__normalize_name(const char *in, char **out)
1206
+ {
1207
+ char *name, *fdot, *ldot;
1208
+
1209
+ assert(in && out);
1210
+
1211
+ name = git__strdup(in);
1212
+ GITERR_CHECK_ALLOC(name);
1213
+
1214
+ fdot = strchr(name, '.');
1215
+ ldot = strrchr(name, '.');
1216
+
1217
+ if (fdot == NULL || fdot == name || ldot == NULL || !ldot[1])
1218
+ goto invalid;
1219
+
1220
+ /* Validate and downcase up to first dot and after last dot */
1221
+ if (git_config_file_normalize_section(name, fdot) < 0 ||
1222
+ git_config_file_normalize_section(ldot + 1, NULL) < 0)
1223
+ goto invalid;
1224
+
1225
+ /* If there is a middle range, make sure it doesn't have newlines */
1226
+ while (fdot < ldot)
1227
+ if (*fdot++ == '\n')
1228
+ goto invalid;
1229
+
1230
+ *out = name;
1231
+ return 0;
1232
+
1233
+ invalid:
1234
+ git__free(name);
1235
+ giterr_set(GITERR_CONFIG, "Invalid config item name '%s'", in);
1236
+ return GIT_EINVALIDSPEC;
1237
+ }
1238
+
814
1239
  struct rename_data {
815
1240
  git_config *config;
816
1241
  git_buf *name;
817
1242
  size_t old_len;
818
- int actual_error;
819
1243
  };
820
1244
 
821
1245
  static int rename_config_entries_cb(
@@ -838,8 +1262,6 @@ static int rename_config_entries_cb(
838
1262
  if (!error)
839
1263
  error = git_config_delete_entry(data->config, entry->name);
840
1264
 
841
- data->actual_error = error; /* preserve actual error code */
842
-
843
1265
  return error;
844
1266
  }
845
1267
 
@@ -864,7 +1286,6 @@ int git_config_rename_section(
864
1286
  data.config = config;
865
1287
  data.name = &replace;
866
1288
  data.old_len = strlen(old_section_name) + 1;
867
- data.actual_error = 0;
868
1289
 
869
1290
  if ((error = git_buf_join(&replace, '.', new_section_name, "")) < 0)
870
1291
  goto cleanup;
@@ -881,12 +1302,16 @@ int git_config_rename_section(
881
1302
  error = git_config_foreach_match(
882
1303
  config, git_buf_cstr(&pattern), rename_config_entries_cb, &data);
883
1304
 
884
- if (error == GIT_EUSER)
885
- error = data.actual_error;
886
-
887
1305
  cleanup:
888
1306
  git_buf_free(&pattern);
889
1307
  git_buf_free(&replace);
890
1308
 
891
1309
  return error;
892
1310
  }
1311
+
1312
+ int git_config_init_backend(git_config_backend *backend, unsigned int version)
1313
+ {
1314
+ GIT_INIT_STRUCTURE_FROM_TEMPLATE(
1315
+ backend, version, git_config_backend, GIT_CONFIG_BACKEND_INIT);
1316
+ return 0;
1317
+ }