You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

BatchRefUpdateTest.java 35KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081
  1. /*
  2. * Copyright (C) 2017 Google Inc. and others
  3. *
  4. * This program and the accompanying materials are made available under the
  5. * terms of the Eclipse Distribution License v. 1.0 which is available at
  6. * https://www.eclipse.org/org/documents/edl-v10.php.
  7. *
  8. * SPDX-License-Identifier: BSD-3-Clause
  9. */
  10. package org.eclipse.jgit.internal.storage.file;
  11. import static java.nio.charset.StandardCharsets.UTF_8;
  12. import static java.util.concurrent.TimeUnit.NANOSECONDS;
  13. import static java.util.concurrent.TimeUnit.SECONDS;
  14. import static org.eclipse.jgit.internal.storage.file.BatchRefUpdateTest.Result.LOCK_FAILURE;
  15. import static org.eclipse.jgit.internal.storage.file.BatchRefUpdateTest.Result.OK;
  16. import static org.eclipse.jgit.internal.storage.file.BatchRefUpdateTest.Result.REJECTED_MISSING_OBJECT;
  17. import static org.eclipse.jgit.internal.storage.file.BatchRefUpdateTest.Result.REJECTED_NONFASTFORWARD;
  18. import static org.eclipse.jgit.internal.storage.file.BatchRefUpdateTest.Result.TRANSACTION_ABORTED;
  19. import static org.eclipse.jgit.lib.ObjectId.zeroId;
  20. import static org.eclipse.jgit.transport.ReceiveCommand.Type.CREATE;
  21. import static org.eclipse.jgit.transport.ReceiveCommand.Type.DELETE;
  22. import static org.eclipse.jgit.transport.ReceiveCommand.Type.UPDATE;
  23. import static org.eclipse.jgit.transport.ReceiveCommand.Type.UPDATE_NONFASTFORWARD;
  24. import static org.junit.Assert.assertEquals;
  25. import static org.junit.Assert.assertFalse;
  26. import static org.junit.Assert.assertNotNull;
  27. import static org.junit.Assert.assertNull;
  28. import static org.junit.Assert.assertTrue;
  29. import static org.junit.Assume.assumeFalse;
  30. import static org.junit.Assume.assumeTrue;
  31. import java.io.File;
  32. import java.io.IOException;
  33. import java.nio.file.Files;
  34. import java.util.Arrays;
  35. import java.util.Collection;
  36. import java.util.Collections;
  37. import java.util.LinkedHashMap;
  38. import java.util.List;
  39. import java.util.Map;
  40. import java.util.concurrent.locks.ReentrantLock;
  41. import java.util.function.Predicate;
  42. import org.eclipse.jgit.events.ListenerHandle;
  43. import org.eclipse.jgit.events.RefsChangedListener;
  44. import org.eclipse.jgit.junit.LocalDiskRepositoryTestCase;
  45. import org.eclipse.jgit.junit.StrictWorkMonitor;
  46. import org.eclipse.jgit.junit.TestRepository;
  47. import org.eclipse.jgit.lib.AnyObjectId;
  48. import org.eclipse.jgit.lib.BatchRefUpdate;
  49. import org.eclipse.jgit.lib.CheckoutEntry;
  50. import org.eclipse.jgit.lib.ConfigConstants;
  51. import org.eclipse.jgit.lib.Constants;
  52. import org.eclipse.jgit.lib.NullProgressMonitor;
  53. import org.eclipse.jgit.lib.ObjectId;
  54. import org.eclipse.jgit.lib.PersonIdent;
  55. import org.eclipse.jgit.lib.Ref;
  56. import org.eclipse.jgit.lib.RefDatabase;
  57. import org.eclipse.jgit.lib.RefUpdate;
  58. import org.eclipse.jgit.lib.ReflogEntry;
  59. import org.eclipse.jgit.lib.ReflogReader;
  60. import org.eclipse.jgit.lib.Repository;
  61. import org.eclipse.jgit.lib.StoredConfig;
  62. import org.eclipse.jgit.revwalk.RevCommit;
  63. import org.eclipse.jgit.revwalk.RevWalk;
  64. import org.eclipse.jgit.transport.ReceiveCommand;
  65. import org.junit.After;
  66. import org.junit.Before;
  67. import org.junit.Test;
  68. import org.junit.runner.RunWith;
  69. import org.junit.runners.Parameterized;
  70. import org.junit.runners.Parameterized.Parameter;
  71. import org.junit.runners.Parameterized.Parameters;
  72. @SuppressWarnings("boxing")
  73. @RunWith(Parameterized.class)
  74. public class BatchRefUpdateTest extends LocalDiskRepositoryTestCase {
  75. @Parameter(0)
  76. public boolean atomic;
  77. @Parameter(1)
  78. public boolean useReftable;
  79. @Parameters(name = "atomic={0} reftable={1}")
  80. public static Collection<Object[]> data() {
  81. return Arrays.asList(new Object[][] { { Boolean.FALSE, Boolean.FALSE },
  82. { Boolean.TRUE, Boolean.FALSE },
  83. { Boolean.FALSE, Boolean.TRUE },
  84. { Boolean.TRUE, Boolean.TRUE }, });
  85. }
  86. private Repository diskRepo;
  87. private TestRepository<Repository> repo;
  88. private RefDirectory refdir;
  89. private RevCommit A;
  90. private RevCommit B; // B descends from A.
  91. /**
  92. * When asserting the number of RefsChangedEvents you must account for one
  93. * additional event due to the initial ref setup via a number of calls to
  94. * {@link #writeLooseRef(String, AnyObjectId)} (will be fired in execute()
  95. * when it is detected that the on-disk loose refs have changed), or for one
  96. * additional event per {@link #writeRef(String, AnyObjectId)}.
  97. */
  98. private int refsChangedEvents;
  99. private ListenerHandle handle;
  100. private RefsChangedListener refsChangedListener = event -> {
  101. refsChangedEvents++;
  102. };
  103. @Override
  104. @Before
  105. public void setUp() throws Exception {
  106. super.setUp();
  107. FileRepository fileRepo = createBareRepository();
  108. if (useReftable) {
  109. fileRepo.convertToReftable(false, false);
  110. }
  111. diskRepo = fileRepo;
  112. setLogAllRefUpdates(true);
  113. if (!useReftable) {
  114. refdir = (RefDirectory) diskRepo.getRefDatabase();
  115. refdir.setRetrySleepMs(Arrays.asList(0, 0));
  116. }
  117. repo = new TestRepository<>(diskRepo);
  118. A = repo.commit().create();
  119. B = repo.commit(repo.getRevWalk().parseCommit(A));
  120. refsChangedEvents = 0;
  121. handle = diskRepo.getListenerList()
  122. .addRefsChangedListener(refsChangedListener);
  123. }
  124. @After
  125. public void removeListener() {
  126. handle.remove();
  127. refsChangedEvents = 0;
  128. }
  129. @Test
  130. public void packedRefsFileIsSorted() throws IOException {
  131. assumeTrue(atomic);
  132. assumeFalse(useReftable);
  133. for (int i = 0; i < 2; i++) {
  134. BatchRefUpdate bu = diskRepo.getRefDatabase().newBatchUpdate();
  135. String b1 = String.format("refs/heads/a%d", i);
  136. String b2 = String.format("refs/heads/b%d", i);
  137. bu.setAtomic(atomic);
  138. ReceiveCommand c1 = new ReceiveCommand(ObjectId.zeroId(), A, b1);
  139. ReceiveCommand c2 = new ReceiveCommand(ObjectId.zeroId(), B, b2);
  140. bu.addCommand(c1, c2);
  141. try (RevWalk rw = new RevWalk(diskRepo)) {
  142. bu.execute(rw, NullProgressMonitor.INSTANCE);
  143. }
  144. assertEquals(c1.getResult(), ReceiveCommand.Result.OK);
  145. assertEquals(c2.getResult(), ReceiveCommand.Result.OK);
  146. }
  147. File packed = new File(diskRepo.getDirectory(), "packed-refs");
  148. String packedStr = new String(Files.readAllBytes(packed.toPath()),
  149. UTF_8);
  150. int a2 = packedStr.indexOf("refs/heads/a1");
  151. int b1 = packedStr.indexOf("refs/heads/b0");
  152. assertTrue(a2 < b1);
  153. }
  154. @Test
  155. public void simpleNoForce() throws IOException {
  156. writeLooseRefs("refs/heads/master", A, "refs/heads/masters", B);
  157. List<ReceiveCommand> cmds = Arrays.asList(
  158. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  159. new ReceiveCommand(B, A, "refs/heads/masters",
  160. UPDATE_NONFASTFORWARD));
  161. execute(newBatchUpdate(cmds));
  162. if (atomic) {
  163. assertResults(cmds, TRANSACTION_ABORTED, REJECTED_NONFASTFORWARD);
  164. assertRefs("refs/heads/master", A, "refs/heads/masters", B);
  165. assertEquals(1, refsChangedEvents);
  166. } else {
  167. assertResults(cmds, OK, REJECTED_NONFASTFORWARD);
  168. assertRefs("refs/heads/master", B, "refs/heads/masters", B);
  169. assertEquals(2, refsChangedEvents);
  170. }
  171. }
  172. @Test
  173. public void simpleForce() throws IOException {
  174. writeLooseRefs("refs/heads/master", A, "refs/heads/masters", B);
  175. List<ReceiveCommand> cmds = Arrays.asList(
  176. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  177. new ReceiveCommand(B, A, "refs/heads/masters",
  178. UPDATE_NONFASTFORWARD));
  179. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  180. assertResults(cmds, OK, OK);
  181. assertRefs("refs/heads/master", B, "refs/heads/masters", A);
  182. assertEquals(batchesRefUpdates() ? 2 : 3, refsChangedEvents);
  183. }
  184. @Test
  185. public void nonFastForwardDoesNotDoExpensiveMergeCheck()
  186. throws IOException {
  187. writeLooseRef("refs/heads/master", B);
  188. List<ReceiveCommand> cmds = Arrays.asList(new ReceiveCommand(B, A,
  189. "refs/heads/master", UPDATE_NONFASTFORWARD));
  190. try (RevWalk rw = new RevWalk(diskRepo) {
  191. @Override
  192. public boolean isMergedInto(RevCommit base, RevCommit tip) {
  193. throw new AssertionError("isMergedInto() should not be called");
  194. }
  195. }) {
  196. newBatchUpdate(cmds).setAllowNonFastForwards(true).execute(rw,
  197. new StrictWorkMonitor());
  198. }
  199. assertResults(cmds, OK);
  200. assertRefs("refs/heads/master", A);
  201. assertEquals(2, refsChangedEvents);
  202. }
  203. @Test
  204. public void fileDirectoryConflict() throws IOException {
  205. writeLooseRefs("refs/heads/master", A, "refs/heads/masters", B);
  206. List<ReceiveCommand> cmds = Arrays.asList(
  207. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  208. new ReceiveCommand(zeroId(), A, "refs/heads/master/x", CREATE),
  209. new ReceiveCommand(zeroId(), A, "refs/heads", CREATE));
  210. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true), false);
  211. if (atomic) {
  212. // Atomic update sees that master and master/x are conflicting, then
  213. // marks the first one in the list as LOCK_FAILURE and aborts the rest.
  214. assertResults(cmds, LOCK_FAILURE, TRANSACTION_ABORTED,
  215. TRANSACTION_ABORTED);
  216. assertRefs("refs/heads/master", A, "refs/heads/masters", B);
  217. assertEquals(1, refsChangedEvents);
  218. } else {
  219. // Non-atomic updates are applied in order: master succeeds, then
  220. // master/x fails due to conflict.
  221. assertResults(cmds, OK, LOCK_FAILURE, LOCK_FAILURE);
  222. assertRefs("refs/heads/master", B, "refs/heads/masters", B);
  223. assertEquals(2, refsChangedEvents);
  224. }
  225. }
  226. @Test
  227. public void conflictThanksToDelete() throws IOException {
  228. writeLooseRefs("refs/heads/master", A, "refs/heads/masters", B);
  229. List<ReceiveCommand> cmds = Arrays.asList(
  230. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  231. new ReceiveCommand(zeroId(), A, "refs/heads/masters/x", CREATE),
  232. new ReceiveCommand(B, zeroId(), "refs/heads/masters", DELETE));
  233. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  234. assertResults(cmds, OK, OK, OK);
  235. assertRefs("refs/heads/master", B, "refs/heads/masters/x", A);
  236. if (atomic) {
  237. assertEquals(2, refsChangedEvents);
  238. } else if (!useReftable) {
  239. // The non-atomic case actually produces 5 events, but that's an
  240. // implementation detail. We expect at least 4 events, one for the
  241. // initial read due to writeLooseRef(), and then one for each
  242. // successful ref update.
  243. assertTrue(refsChangedEvents >= 4);
  244. }
  245. }
  246. @Test
  247. public void updateToMissingObject() throws IOException {
  248. writeLooseRef("refs/heads/master", A);
  249. ObjectId bad = ObjectId
  250. .fromString("deadbeefdeadbeefdeadbeefdeadbeefdeadbeef");
  251. List<ReceiveCommand> cmds = Arrays.asList(
  252. new ReceiveCommand(A, bad, "refs/heads/master", UPDATE),
  253. new ReceiveCommand(zeroId(), B, "refs/heads/foo2", CREATE));
  254. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true), false);
  255. if (atomic) {
  256. assertResults(cmds, REJECTED_MISSING_OBJECT, TRANSACTION_ABORTED);
  257. assertRefs("refs/heads/master", A);
  258. assertEquals(1, refsChangedEvents);
  259. } else {
  260. assertResults(cmds, REJECTED_MISSING_OBJECT, OK);
  261. assertRefs("refs/heads/master", A, "refs/heads/foo2", B);
  262. assertEquals(2, refsChangedEvents);
  263. }
  264. }
  265. @Test
  266. public void addMissingObject() throws IOException {
  267. writeLooseRef("refs/heads/master", A);
  268. ObjectId bad = ObjectId
  269. .fromString("deadbeefdeadbeefdeadbeefdeadbeefdeadbeef");
  270. List<ReceiveCommand> cmds = Arrays.asList(
  271. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  272. new ReceiveCommand(zeroId(), bad, "refs/heads/foo2", CREATE));
  273. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true), false);
  274. if (atomic) {
  275. assertResults(cmds, TRANSACTION_ABORTED, REJECTED_MISSING_OBJECT);
  276. assertRefs("refs/heads/master", A);
  277. assertEquals(1, refsChangedEvents);
  278. } else {
  279. assertResults(cmds, OK, REJECTED_MISSING_OBJECT);
  280. assertRefs("refs/heads/master", B);
  281. assertEquals(2, refsChangedEvents);
  282. }
  283. }
  284. @Test
  285. public void oneNonExistentRef() throws IOException {
  286. List<ReceiveCommand> cmds = Arrays.asList(
  287. new ReceiveCommand(A, B, "refs/heads/foo1", UPDATE),
  288. new ReceiveCommand(zeroId(), B, "refs/heads/foo2", CREATE));
  289. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  290. if (atomic) {
  291. assertResults(cmds, LOCK_FAILURE, TRANSACTION_ABORTED);
  292. assertRefs();
  293. assertEquals(0, refsChangedEvents);
  294. } else {
  295. assertResults(cmds, LOCK_FAILURE, OK);
  296. assertRefs("refs/heads/foo2", B);
  297. assertEquals(1, refsChangedEvents);
  298. }
  299. }
  300. @Test
  301. public void oneRefWrongOldValue() throws IOException {
  302. writeLooseRef("refs/heads/master", A);
  303. List<ReceiveCommand> cmds = Arrays.asList(
  304. new ReceiveCommand(B, B, "refs/heads/master", UPDATE),
  305. new ReceiveCommand(zeroId(), B, "refs/heads/foo2", CREATE));
  306. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  307. if (atomic) {
  308. assertResults(cmds, LOCK_FAILURE, TRANSACTION_ABORTED);
  309. assertRefs("refs/heads/master", A);
  310. assertEquals(1, refsChangedEvents);
  311. } else {
  312. assertResults(cmds, LOCK_FAILURE, OK);
  313. assertRefs("refs/heads/master", A, "refs/heads/foo2", B);
  314. assertEquals(2, refsChangedEvents);
  315. }
  316. }
  317. @Test
  318. public void nonExistentRef() throws IOException {
  319. writeLooseRef("refs/heads/master", A);
  320. List<ReceiveCommand> cmds = Arrays.asList(
  321. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  322. new ReceiveCommand(A, zeroId(), "refs/heads/foo2", DELETE));
  323. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  324. if (atomic) {
  325. assertResults(cmds, TRANSACTION_ABORTED, LOCK_FAILURE);
  326. assertRefs("refs/heads/master", A);
  327. assertEquals(1, refsChangedEvents);
  328. } else {
  329. assertResults(cmds, OK, LOCK_FAILURE);
  330. assertRefs("refs/heads/master", B);
  331. assertEquals(2, refsChangedEvents);
  332. }
  333. }
  334. @Test
  335. public void noRefLog() throws IOException {
  336. writeRef("refs/heads/master", A);
  337. Map<String, ReflogEntry> oldLogs = getLastReflogs("refs/heads/master",
  338. "refs/heads/branch");
  339. assertEquals(Collections.singleton("refs/heads/master"),
  340. oldLogs.keySet());
  341. List<ReceiveCommand> cmds = Arrays.asList(
  342. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  343. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  344. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  345. assertResults(cmds, OK, OK);
  346. assertRefs("refs/heads/master", B, "refs/heads/branch", B);
  347. assertEquals(batchesRefUpdates() ? 2 : 3, refsChangedEvents);
  348. assertReflogUnchanged(oldLogs, "refs/heads/master");
  349. assertReflogUnchanged(oldLogs, "refs/heads/branch");
  350. }
  351. @Test
  352. public void reflogDefaultIdent() throws IOException {
  353. writeRef("refs/heads/master", A);
  354. writeRef("refs/heads/branch2", A);
  355. Map<String, ReflogEntry> oldLogs = getLastReflogs("refs/heads/master",
  356. "refs/heads/branch1", "refs/heads/branch2");
  357. List<ReceiveCommand> cmds = Arrays.asList(
  358. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  359. new ReceiveCommand(zeroId(), B, "refs/heads/branch1", CREATE));
  360. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true)
  361. .setRefLogMessage("a reflog", false));
  362. assertResults(cmds, OK, OK);
  363. assertRefs("refs/heads/master", B, "refs/heads/branch1", B,
  364. "refs/heads/branch2", A);
  365. assertEquals(batchesRefUpdates() ? 3 : 4, refsChangedEvents);
  366. assertReflogEquals(reflog(A, B, new PersonIdent(diskRepo), "a reflog"),
  367. getLastReflog("refs/heads/master"));
  368. assertReflogEquals(
  369. reflog(zeroId(), B, new PersonIdent(diskRepo), "a reflog"),
  370. getLastReflog("refs/heads/branch1"));
  371. assertReflogUnchanged(oldLogs, "refs/heads/branch2");
  372. }
  373. @Test
  374. public void reflogAppendStatusNoMessage() throws IOException {
  375. writeRef("refs/heads/master", A);
  376. writeRef("refs/heads/branch1", B);
  377. List<ReceiveCommand> cmds = Arrays.asList(
  378. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  379. new ReceiveCommand(B, A, "refs/heads/branch1",
  380. UPDATE_NONFASTFORWARD),
  381. new ReceiveCommand(zeroId(), A, "refs/heads/branch2", CREATE));
  382. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true)
  383. .setRefLogMessage(null, true));
  384. assertResults(cmds, OK, OK, OK);
  385. assertRefs("refs/heads/master", B, "refs/heads/branch1", A,
  386. "refs/heads/branch2", A);
  387. assertEquals(batchesRefUpdates() ? 3 : 5, refsChangedEvents);
  388. assertReflogEquals(
  389. // Always forced; setAllowNonFastForwards(true) bypasses the
  390. // check.
  391. reflog(A, B, new PersonIdent(diskRepo), "forced-update"),
  392. getLastReflog("refs/heads/master"));
  393. assertReflogEquals(
  394. reflog(B, A, new PersonIdent(diskRepo), "forced-update"),
  395. getLastReflog("refs/heads/branch1"));
  396. assertReflogEquals(
  397. reflog(zeroId(), A, new PersonIdent(diskRepo), "created"),
  398. getLastReflog("refs/heads/branch2"));
  399. }
  400. @Test
  401. public void reflogAppendStatusFastForward() throws IOException {
  402. writeRef("refs/heads/master", A);
  403. List<ReceiveCommand> cmds = Arrays
  404. .asList(new ReceiveCommand(A, B, "refs/heads/master", UPDATE));
  405. execute(newBatchUpdate(cmds).setRefLogMessage(null, true));
  406. assertResults(cmds, OK);
  407. assertRefs("refs/heads/master", B);
  408. assertEquals(2, refsChangedEvents);
  409. assertReflogEquals(
  410. reflog(A, B, new PersonIdent(diskRepo), "fast-forward"),
  411. getLastReflog("refs/heads/master"));
  412. }
  413. @Test
  414. public void reflogAppendStatusWithMessage() throws IOException {
  415. writeRef("refs/heads/master", A);
  416. List<ReceiveCommand> cmds = Arrays.asList(
  417. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  418. new ReceiveCommand(zeroId(), A, "refs/heads/branch", CREATE));
  419. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", true));
  420. assertResults(cmds, OK, OK);
  421. assertRefs("refs/heads/master", B, "refs/heads/branch", A);
  422. assertEquals(batchesRefUpdates() ? 2 : 3, refsChangedEvents);
  423. assertReflogEquals(
  424. reflog(A, B, new PersonIdent(diskRepo),
  425. "a reflog: fast-forward"),
  426. getLastReflog("refs/heads/master"));
  427. assertReflogEquals(
  428. reflog(zeroId(), A, new PersonIdent(diskRepo),
  429. "a reflog: created"),
  430. getLastReflog("refs/heads/branch"));
  431. }
  432. @Test
  433. public void reflogCustomIdent() throws IOException {
  434. writeRef("refs/heads/master", A);
  435. List<ReceiveCommand> cmds = Arrays.asList(
  436. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  437. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  438. PersonIdent ident = new PersonIdent("A Reflog User",
  439. "reflog@example.com");
  440. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", false)
  441. .setRefLogIdent(ident));
  442. assertResults(cmds, OK, OK);
  443. assertEquals(batchesRefUpdates() ? 2 : 3, refsChangedEvents);
  444. assertRefs("refs/heads/master", B, "refs/heads/branch", B);
  445. assertReflogEquals(reflog(A, B, ident, "a reflog"),
  446. getLastReflog("refs/heads/master"), true);
  447. assertReflogEquals(reflog(zeroId(), B, ident, "a reflog"),
  448. getLastReflog("refs/heads/branch"), true);
  449. }
  450. @Test
  451. public void reflogDelete() throws IOException {
  452. writeRef("refs/heads/master", A);
  453. writeRef("refs/heads/branch", A);
  454. assertEquals(2, getLastReflogs("refs/heads/master", "refs/heads/branch")
  455. .size());
  456. List<ReceiveCommand> cmds = Arrays.asList(
  457. new ReceiveCommand(A, zeroId(), "refs/heads/master", DELETE),
  458. new ReceiveCommand(A, B, "refs/heads/branch", UPDATE));
  459. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", false));
  460. assertResults(cmds, OK, OK);
  461. assertRefs("refs/heads/branch", B);
  462. assertEquals(batchesRefUpdates() ? 3 : 4, refsChangedEvents);
  463. if (useReftable) {
  464. // reftable retains reflog entries for deleted branches.
  465. assertReflogEquals(
  466. reflog(A, zeroId(), new PersonIdent(diskRepo), "a reflog"),
  467. getLastReflog("refs/heads/master"));
  468. } else {
  469. assertNull(getLastReflog("refs/heads/master"));
  470. }
  471. assertReflogEquals(reflog(A, B, new PersonIdent(diskRepo), "a reflog"),
  472. getLastReflog("refs/heads/branch"));
  473. }
  474. @Test
  475. public void reflogFileDirectoryConflict() throws IOException {
  476. writeRef("refs/heads/master", A);
  477. List<ReceiveCommand> cmds = Arrays.asList(
  478. new ReceiveCommand(A, zeroId(), "refs/heads/master", DELETE),
  479. new ReceiveCommand(zeroId(), A, "refs/heads/master/x", CREATE));
  480. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", false));
  481. assertResults(cmds, OK, OK);
  482. assertRefs("refs/heads/master/x", A);
  483. assertEquals(batchesRefUpdates() ? 2 : 3, refsChangedEvents);
  484. if (!useReftable) {
  485. // reftable retains reflog entries for deleted branches.
  486. assertNull(getLastReflog("refs/heads/master"));
  487. }
  488. assertReflogEquals(
  489. reflog(zeroId(), A, new PersonIdent(diskRepo), "a reflog"),
  490. getLastReflog("refs/heads/master/x"));
  491. }
  492. @Test
  493. public void reflogOnLockFailure() throws IOException {
  494. writeRef("refs/heads/master", A);
  495. Map<String, ReflogEntry> oldLogs = getLastReflogs("refs/heads/master",
  496. "refs/heads/branch");
  497. List<ReceiveCommand> cmds = Arrays.asList(
  498. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  499. new ReceiveCommand(A, B, "refs/heads/branch", UPDATE));
  500. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", false));
  501. if (atomic) {
  502. assertResults(cmds, TRANSACTION_ABORTED, LOCK_FAILURE);
  503. assertEquals(1, refsChangedEvents);
  504. assertReflogUnchanged(oldLogs, "refs/heads/master");
  505. assertReflogUnchanged(oldLogs, "refs/heads/branch");
  506. } else {
  507. assertResults(cmds, OK, LOCK_FAILURE);
  508. assertEquals(2, refsChangedEvents);
  509. assertReflogEquals(
  510. reflog(A, B, new PersonIdent(diskRepo), "a reflog"),
  511. getLastReflog("refs/heads/master"));
  512. assertReflogUnchanged(oldLogs, "refs/heads/branch");
  513. }
  514. }
  515. @Test
  516. public void overrideRefLogMessage() throws Exception {
  517. writeRef("refs/heads/master", A);
  518. List<ReceiveCommand> cmds = Arrays.asList(
  519. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  520. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  521. cmds.get(0).setRefLogMessage("custom log", false);
  522. PersonIdent ident = new PersonIdent(diskRepo);
  523. execute(newBatchUpdate(cmds).setRefLogIdent(ident)
  524. .setRefLogMessage("a reflog", true));
  525. assertResults(cmds, OK, OK);
  526. assertEquals(batchesRefUpdates() ? 2 : 3, refsChangedEvents);
  527. assertReflogEquals(reflog(A, B, ident, "custom log"),
  528. getLastReflog("refs/heads/master"), true);
  529. assertReflogEquals(reflog(zeroId(), B, ident, "a reflog: created"),
  530. getLastReflog("refs/heads/branch"), true);
  531. }
  532. @Test
  533. public void overrideDisableRefLog() throws Exception {
  534. writeRef("refs/heads/master", A);
  535. Map<String, ReflogEntry> oldLogs = getLastReflogs("refs/heads/master",
  536. "refs/heads/branch");
  537. List<ReceiveCommand> cmds = Arrays.asList(
  538. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  539. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  540. cmds.get(0).disableRefLog();
  541. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", true));
  542. assertResults(cmds, OK, OK);
  543. assertEquals(batchesRefUpdates() ? 2 : 3, refsChangedEvents);
  544. assertReflogUnchanged(oldLogs, "refs/heads/master");
  545. assertReflogEquals(
  546. reflog(zeroId(), B, new PersonIdent(diskRepo),
  547. "a reflog: created"),
  548. getLastReflog("refs/heads/branch"));
  549. }
  550. @Test
  551. public void refLogNotWrittenWithoutConfigOption() throws Exception {
  552. assumeFalse(useReftable);
  553. setLogAllRefUpdates(false);
  554. writeRef("refs/heads/master", A);
  555. Map<String, ReflogEntry> oldLogs = getLastReflogs("refs/heads/master",
  556. "refs/heads/branch");
  557. assertTrue(oldLogs.isEmpty());
  558. List<ReceiveCommand> cmds = Arrays.asList(
  559. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  560. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  561. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", false));
  562. assertResults(cmds, OK, OK);
  563. assertReflogUnchanged(oldLogs, "refs/heads/master");
  564. assertReflogUnchanged(oldLogs, "refs/heads/branch");
  565. }
  566. @Test
  567. public void forceRefLogInUpdate() throws Exception {
  568. assumeFalse(useReftable);
  569. setLogAllRefUpdates(false);
  570. writeRef("refs/heads/master", A);
  571. assertTrue(getLastReflogs("refs/heads/master", "refs/heads/branch")
  572. .isEmpty());
  573. List<ReceiveCommand> cmds = Arrays.asList(
  574. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  575. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  576. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", false)
  577. .setForceRefLog(true));
  578. assertResults(cmds, OK, OK);
  579. assertReflogEquals(reflog(A, B, new PersonIdent(diskRepo), "a reflog"),
  580. getLastReflog("refs/heads/master"));
  581. assertReflogEquals(
  582. reflog(zeroId(), B, new PersonIdent(diskRepo), "a reflog"),
  583. getLastReflog("refs/heads/branch"));
  584. }
  585. @Test
  586. public void forceRefLogInCommand() throws Exception {
  587. assumeFalse(useReftable);
  588. setLogAllRefUpdates(false);
  589. writeRef("refs/heads/master", A);
  590. Map<String, ReflogEntry> oldLogs = getLastReflogs("refs/heads/master",
  591. "refs/heads/branch");
  592. assertTrue(oldLogs.isEmpty());
  593. List<ReceiveCommand> cmds = Arrays.asList(
  594. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  595. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  596. cmds.get(1).setForceRefLog(true);
  597. execute(newBatchUpdate(cmds).setRefLogMessage("a reflog", false));
  598. assertResults(cmds, OK, OK);
  599. assertReflogUnchanged(oldLogs, "refs/heads/master");
  600. assertReflogEquals(
  601. reflog(zeroId(), B, new PersonIdent(diskRepo), "a reflog"),
  602. getLastReflog("refs/heads/branch"));
  603. }
  604. @Test
  605. public void packedRefsLockFailure() throws Exception {
  606. assumeFalse(useReftable);
  607. writeLooseRef("refs/heads/master", A);
  608. List<ReceiveCommand> cmds = Arrays.asList(
  609. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  610. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  611. LockFile myLock = refdir.lockPackedRefs();
  612. try {
  613. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  614. assertFalse(getLockFile("refs/heads/master").exists());
  615. assertFalse(getLockFile("refs/heads/branch").exists());
  616. if (atomic) {
  617. assertResults(cmds, LOCK_FAILURE, TRANSACTION_ABORTED);
  618. assertRefs("refs/heads/master", A);
  619. assertEquals(1, refsChangedEvents);
  620. } else {
  621. // Only operates on loose refs, doesn't care that packed-refs is
  622. // locked.
  623. assertResults(cmds, OK, OK);
  624. assertRefs("refs/heads/master", B, "refs/heads/branch", B);
  625. assertEquals(3, refsChangedEvents);
  626. }
  627. } finally {
  628. myLock.unlock();
  629. }
  630. }
  631. @Test
  632. public void oneRefLockFailure() throws Exception {
  633. assumeFalse(useReftable);
  634. writeLooseRef("refs/heads/master", A);
  635. List<ReceiveCommand> cmds = Arrays.asList(
  636. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE),
  637. new ReceiveCommand(A, B, "refs/heads/master", UPDATE));
  638. LockFile myLock = new LockFile(refdir.fileFor("refs/heads/master"));
  639. assertTrue(myLock.lock());
  640. try {
  641. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  642. assertFalse(LockFile.getLockFile(refdir.packedRefsFile).exists());
  643. assertFalse(getLockFile("refs/heads/branch").exists());
  644. if (atomic) {
  645. assertResults(cmds, TRANSACTION_ABORTED, LOCK_FAILURE);
  646. assertRefs("refs/heads/master", A);
  647. assertEquals(1, refsChangedEvents);
  648. } else {
  649. assertResults(cmds, OK, LOCK_FAILURE);
  650. assertRefs("refs/heads/branch", B, "refs/heads/master", A);
  651. assertEquals(2, refsChangedEvents);
  652. }
  653. } finally {
  654. myLock.unlock();
  655. }
  656. }
  657. @Test
  658. public void singleRefUpdateDoesNotRequirePackedRefsLock() throws Exception {
  659. assumeFalse(useReftable);
  660. writeLooseRef("refs/heads/master", A);
  661. List<ReceiveCommand> cmds = Arrays
  662. .asList(new ReceiveCommand(A, B, "refs/heads/master", UPDATE));
  663. LockFile myLock = refdir.lockPackedRefs();
  664. try {
  665. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  666. assertFalse(getLockFile("refs/heads/master").exists());
  667. assertResults(cmds, OK);
  668. assertEquals(2, refsChangedEvents);
  669. assertRefs("refs/heads/master", B);
  670. } finally {
  671. myLock.unlock();
  672. }
  673. }
  674. @Test
  675. public void atomicUpdateRespectsInProcessLock() throws Exception {
  676. assumeTrue(atomic);
  677. assumeFalse(useReftable);
  678. writeLooseRef("refs/heads/master", A);
  679. List<ReceiveCommand> cmds = Arrays.asList(
  680. new ReceiveCommand(A, B, "refs/heads/master", UPDATE),
  681. new ReceiveCommand(zeroId(), B, "refs/heads/branch", CREATE));
  682. Thread t = new Thread(() -> {
  683. try {
  684. execute(newBatchUpdate(cmds).setAllowNonFastForwards(true));
  685. } catch (Exception e) {
  686. throw new RuntimeException(e);
  687. }
  688. });
  689. ReentrantLock l = refdir.inProcessPackedRefsLock;
  690. l.lock();
  691. try {
  692. t.start();
  693. long timeoutSecs = 10;
  694. long startNanos = System.nanoTime();
  695. // Hold onto the lock until we observe the worker thread has
  696. // attempted to
  697. // acquire it.
  698. while (l.getQueueLength() == 0) {
  699. long elapsedNanos = System.nanoTime() - startNanos;
  700. assertTrue(
  701. "timed out waiting for work thread to attempt to acquire lock",
  702. NANOSECONDS.toSeconds(elapsedNanos) < timeoutSecs);
  703. Thread.sleep(3);
  704. }
  705. // Once we unlock, the worker thread should finish the update
  706. // promptly.
  707. l.unlock();
  708. t.join(SECONDS.toMillis(timeoutSecs));
  709. assertFalse(t.isAlive());
  710. } finally {
  711. if (l.isHeldByCurrentThread()) {
  712. l.unlock();
  713. }
  714. }
  715. assertResults(cmds, OK, OK);
  716. assertEquals(2, refsChangedEvents);
  717. assertRefs("refs/heads/master", B, "refs/heads/branch", B);
  718. }
  719. private void setLogAllRefUpdates(boolean enable) throws Exception {
  720. StoredConfig cfg = diskRepo.getConfig();
  721. cfg.load();
  722. cfg.setBoolean(ConfigConstants.CONFIG_CORE_SECTION, null,
  723. ConfigConstants.CONFIG_KEY_LOGALLREFUPDATES, enable);
  724. cfg.save();
  725. }
  726. private void writeLooseRef(String name, AnyObjectId id) throws IOException {
  727. if (useReftable) {
  728. writeRef(name, id);
  729. } else {
  730. write(new File(diskRepo.getDirectory(), name), id.name() + "\n");
  731. }
  732. }
  733. private void writeLooseRefs(String name1, AnyObjectId id1, String name2,
  734. AnyObjectId id2) throws IOException {
  735. if (useReftable) {
  736. BatchRefUpdate bru = diskRepo.getRefDatabase().newBatchUpdate();
  737. Ref r1 = diskRepo.exactRef(name1);
  738. ReceiveCommand c1 = new ReceiveCommand(
  739. r1 != null ? r1.getObjectId() : ObjectId.zeroId(),
  740. id1.toObjectId(), name1, r1 == null ? CREATE : UPDATE);
  741. Ref r2 = diskRepo.exactRef(name2);
  742. ReceiveCommand c2 = new ReceiveCommand(
  743. r2 != null ? r2.getObjectId() : ObjectId.zeroId(),
  744. id2.toObjectId(), name2, r2 == null ? CREATE : UPDATE);
  745. bru.addCommand(c1, c2);
  746. try (RevWalk rw = new RevWalk(diskRepo)) {
  747. bru.execute(rw, NullProgressMonitor.INSTANCE);
  748. }
  749. assertEquals(c2.getResult(), ReceiveCommand.Result.OK);
  750. assertEquals(c1.getResult(), ReceiveCommand.Result.OK);
  751. } else {
  752. writeLooseRef(name1, id1);
  753. writeLooseRef(name2, id2);
  754. }
  755. }
  756. private void writeRef(String name, AnyObjectId id) throws IOException {
  757. RefUpdate u = diskRepo.updateRef(name);
  758. u.setRefLogMessage(getClass().getSimpleName(), false);
  759. u.setForceUpdate(true);
  760. u.setNewObjectId(id);
  761. RefUpdate.Result r = u.update();
  762. switch (r) {
  763. case NEW:
  764. case FORCED:
  765. return;
  766. default:
  767. throw new IOException("Got " + r + " while updating " + name);
  768. }
  769. }
  770. private BatchRefUpdate newBatchUpdate(List<ReceiveCommand> cmds) {
  771. BatchRefUpdate u = diskRepo.getRefDatabase().newBatchUpdate();
  772. if (atomic) {
  773. assertTrue(u.isAtomic());
  774. } else {
  775. u.setAtomic(false);
  776. }
  777. u.addCommand(cmds);
  778. return u;
  779. }
  780. private void execute(BatchRefUpdate u) throws IOException {
  781. execute(u, false);
  782. }
  783. private void execute(BatchRefUpdate u, boolean strictWork)
  784. throws IOException {
  785. try (RevWalk rw = new RevWalk(diskRepo)) {
  786. u.execute(rw, strictWork ? new StrictWorkMonitor()
  787. : NullProgressMonitor.INSTANCE);
  788. }
  789. }
  790. private void assertRefs(Object... args) throws IOException {
  791. if (args.length % 2 != 0) {
  792. throw new IllegalArgumentException(
  793. "expected even number of args: " + Arrays.toString(args));
  794. }
  795. Map<String, AnyObjectId> expected = new LinkedHashMap<>();
  796. for (int i = 0; i < args.length; i += 2) {
  797. expected.put((String) args[i], (AnyObjectId) args[i + 1]);
  798. }
  799. Map<String, Ref> refs = diskRepo.getRefDatabase()
  800. .getRefs(RefDatabase.ALL);
  801. Ref actualHead = refs.remove(Constants.HEAD);
  802. if (actualHead != null) {
  803. String actualLeafName = actualHead.getLeaf().getName();
  804. assertEquals(
  805. "expected HEAD to point to refs/heads/master, got: "
  806. + actualLeafName,
  807. "refs/heads/master", actualLeafName);
  808. AnyObjectId expectedMaster = expected.get("refs/heads/master");
  809. assertNotNull("expected master ref since HEAD exists",
  810. expectedMaster);
  811. assertEquals(expectedMaster, actualHead.getObjectId());
  812. }
  813. Map<String, AnyObjectId> actual = new LinkedHashMap<>();
  814. refs.forEach((n, r) -> actual.put(n, r.getObjectId()));
  815. assertEquals(expected.keySet(), actual.keySet());
  816. actual.forEach((n, a) -> assertEquals(n, expected.get(n), a));
  817. }
  818. enum Result {
  819. OK(ReceiveCommand.Result.OK), LOCK_FAILURE(
  820. ReceiveCommand.Result.LOCK_FAILURE), REJECTED_NONFASTFORWARD(
  821. ReceiveCommand.Result.REJECTED_NONFASTFORWARD), REJECTED_MISSING_OBJECT(
  822. ReceiveCommand.Result.REJECTED_MISSING_OBJECT), TRANSACTION_ABORTED(
  823. ReceiveCommand::isTransactionAborted);
  824. @SuppressWarnings("ImmutableEnumChecker")
  825. final Predicate<? super ReceiveCommand> p;
  826. private Result(Predicate<? super ReceiveCommand> p) {
  827. this.p = p;
  828. }
  829. private Result(ReceiveCommand.Result result) {
  830. this(c -> c.getResult() == result);
  831. }
  832. }
  833. private void assertResults(List<ReceiveCommand> cmds, Result... expected) {
  834. if (expected.length != cmds.size()) {
  835. throw new IllegalArgumentException(
  836. "expected " + cmds.size() + " result args");
  837. }
  838. for (int i = 0; i < cmds.size(); i++) {
  839. ReceiveCommand c = cmds.get(i);
  840. Result r = expected[i];
  841. assertTrue(String.format(
  842. "result of command (%d) should be %s, got %s %s%s",
  843. Integer.valueOf(i), r, c, c.getResult(),
  844. c.getMessage() != null ? " (" + c.getMessage() + ")" : ""),
  845. r.p.test(c));
  846. }
  847. }
  848. private Map<String, ReflogEntry> getLastReflogs(String... names)
  849. throws IOException {
  850. Map<String, ReflogEntry> result = new LinkedHashMap<>();
  851. for (String name : names) {
  852. ReflogEntry e = getLastReflog(name);
  853. if (e != null) {
  854. result.put(name, e);
  855. }
  856. }
  857. return result;
  858. }
  859. private ReflogEntry getLastReflog(String name) throws IOException {
  860. ReflogReader r = diskRepo.getReflogReader(name);
  861. if (r == null) {
  862. return null;
  863. }
  864. return r.getLastEntry();
  865. }
  866. private File getLockFile(String refName) {
  867. return LockFile.getLockFile(refdir.fileFor(refName));
  868. }
  869. private void assertReflogUnchanged(Map<String, ReflogEntry> old,
  870. String name) throws IOException {
  871. assertReflogEquals(old.get(name), getLastReflog(name), true);
  872. }
  873. private static void assertReflogEquals(ReflogEntry expected,
  874. ReflogEntry actual) {
  875. assertReflogEquals(expected, actual, false);
  876. }
  877. private static void assertReflogEquals(ReflogEntry expected,
  878. ReflogEntry actual, boolean strictTime) {
  879. if (expected == null) {
  880. assertNull(actual);
  881. return;
  882. }
  883. assertNotNull(actual);
  884. assertEquals(expected.getOldId(), actual.getOldId());
  885. assertEquals(expected.getNewId(), actual.getNewId());
  886. if (strictTime) {
  887. assertEquals(expected.getWho(), actual.getWho());
  888. } else {
  889. assertEquals(expected.getWho().getName(),
  890. actual.getWho().getName());
  891. assertEquals(expected.getWho().getEmailAddress(),
  892. actual.getWho().getEmailAddress());
  893. }
  894. assertEquals(expected.getComment(), actual.getComment());
  895. }
  896. private static ReflogEntry reflog(ObjectId oldId, ObjectId newId,
  897. PersonIdent who, String comment) {
  898. return new ReflogEntry() {
  899. @Override
  900. public ObjectId getOldId() {
  901. return oldId;
  902. }
  903. @Override
  904. public ObjectId getNewId() {
  905. return newId;
  906. }
  907. @Override
  908. public PersonIdent getWho() {
  909. return who;
  910. }
  911. @Override
  912. public String getComment() {
  913. return comment;
  914. }
  915. @Override
  916. public CheckoutEntry parseCheckout() {
  917. throw new UnsupportedOperationException();
  918. }
  919. };
  920. }
  921. private boolean batchesRefUpdates() {
  922. return atomic || useReftable;
  923. }
  924. }