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.

FileReftableTest.java 19KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568
  1. /*
  2. * Copyright (C) 2019, Google Inc.
  3. * and other copyright owners as documented in the project's IP log.
  4. *
  5. * This program and the accompanying materials are made available
  6. * under the terms of the Eclipse Distribution License v1.0 which
  7. * accompanies this distribution, is reproduced below, and is
  8. * available at http://www.eclipse.org/org/documents/edl-v10.php
  9. *
  10. * All rights reserved.
  11. *
  12. * Redistribution and use in source and binary forms, with or
  13. * without modification, are permitted provided that the following
  14. * conditions are met:
  15. *
  16. * - Redistributions of source code must retain the above copyright
  17. * notice, this list of conditions and the following disclaimer.
  18. *
  19. * - Redistributions in binary form must reproduce the above
  20. * copyright notice, this list of conditions and the following
  21. * disclaimer in the documentation and/or other materials provided
  22. * with the distribution.
  23. *
  24. * - Neither the name of the Eclipse Foundation, Inc. nor the
  25. * names of its contributors may be used to endorse or promote
  26. * products derived from this software without specific prior
  27. * written permission.
  28. *
  29. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
  30. * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
  31. * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  32. * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  33. * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
  34. * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  35. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
  36. * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  37. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  38. * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
  39. * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  40. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
  41. * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  42. */
  43. package org.eclipse.jgit.internal.storage.file;
  44. import static org.eclipse.jgit.lib.RefUpdate.Result.FAST_FORWARD;
  45. import static org.eclipse.jgit.lib.RefUpdate.Result.FORCED;
  46. import static org.eclipse.jgit.lib.RefUpdate.Result.IO_FAILURE;
  47. import static org.eclipse.jgit.lib.RefUpdate.Result.LOCK_FAILURE;
  48. import static org.junit.Assert.assertEquals;
  49. import static org.junit.Assert.assertFalse;
  50. import static org.junit.Assert.assertNotEquals;
  51. import static org.junit.Assert.assertNotNull;
  52. import static org.junit.Assert.assertNotSame;
  53. import static org.junit.Assert.assertNull;
  54. import static org.junit.Assert.assertSame;
  55. import static org.junit.Assert.assertTrue;
  56. import static org.junit.Assert.fail;
  57. import java.io.File;
  58. import java.io.IOException;
  59. import java.util.ArrayList;
  60. import java.util.List;
  61. import org.eclipse.jgit.lib.AnyObjectId;
  62. import org.eclipse.jgit.lib.Constants;
  63. import org.eclipse.jgit.lib.NullProgressMonitor;
  64. import org.eclipse.jgit.lib.ObjectId;
  65. import org.eclipse.jgit.lib.PersonIdent;
  66. import org.eclipse.jgit.lib.Ref;
  67. import org.eclipse.jgit.lib.RefRename;
  68. import org.eclipse.jgit.lib.RefUpdate;
  69. import org.eclipse.jgit.lib.RefUpdate.Result;
  70. import org.eclipse.jgit.lib.ReflogEntry;
  71. import org.eclipse.jgit.lib.ReflogReader;
  72. import org.eclipse.jgit.lib.RepositoryCache;
  73. import org.eclipse.jgit.revwalk.RevWalk;
  74. import org.eclipse.jgit.test.resources.SampleDataRepositoryTestCase;
  75. import org.eclipse.jgit.transport.ReceiveCommand;
  76. import org.junit.Test;
  77. public class FileReftableTest extends SampleDataRepositoryTestCase {
  78. String bCommit;
  79. @Override
  80. public void setUp() throws Exception {
  81. super.setUp();
  82. Ref b = db.exactRef("refs/heads/b");
  83. bCommit = b.getObjectId().getName();
  84. db.convertToReftable(false, false);
  85. }
  86. @SuppressWarnings("boxing")
  87. @Test
  88. public void testRacyReload() throws Exception {
  89. ObjectId id = db.resolve("master");
  90. int retry = 0;
  91. try (FileRepository repo1 = new FileRepository(db.getDirectory());
  92. FileRepository repo2 = new FileRepository(db.getDirectory())) {
  93. FileRepository repos[] = { repo1, repo2 };
  94. for (int i = 0; i < 10; i++) {
  95. for (int j = 0; j < 2; j++) {
  96. FileRepository repo = repos[j];
  97. RefUpdate u = repo.getRefDatabase().newUpdate(
  98. String.format("branch%d", i * 10 + j), false);
  99. u.setNewObjectId(id);
  100. RefUpdate.Result r = u.update();
  101. if (!r.equals(Result.NEW)) {
  102. retry++;
  103. u = repo.getRefDatabase().newUpdate(
  104. String.format("branch%d", i * 10 + j), false);
  105. u.setNewObjectId(id);
  106. r = u.update();
  107. assertEquals(r, Result.NEW);
  108. }
  109. }
  110. }
  111. // only the first one succeeds
  112. assertEquals(retry, 19);
  113. }
  114. }
  115. @Test
  116. public void additionalRefsAreRemoved() {
  117. assertFalse(new File(db.getDirectory(), Constants.HEAD).exists());
  118. }
  119. @Test
  120. public void testCompactFully() throws Exception {
  121. ObjectId c1 = db.resolve("master^^");
  122. ObjectId c2 = db.resolve("master^");
  123. for (int i = 0; i < 5; i++) {
  124. RefUpdate u = db.updateRef("refs/heads/master");
  125. u.setForceUpdate(true);
  126. u.setNewObjectId((i%2) == 0 ? c1 : c2);
  127. assertEquals(u.update(), FORCED);
  128. }
  129. File tableDir = new File(db.getDirectory(), Constants.REFTABLE);
  130. assertTrue(tableDir.listFiles().length > 1);
  131. ((FileReftableDatabase)db.getRefDatabase()).compactFully();
  132. assertEquals(tableDir.listFiles().length,1);
  133. }
  134. @Test
  135. public void testConvert() throws Exception {
  136. Ref h = db.exactRef("HEAD");
  137. assertTrue(h.isSymbolic());
  138. assertEquals("refs/heads/master", h.getTarget().getName());
  139. Ref b = db.exactRef("refs/heads/b");
  140. assertFalse(b.isSymbolic());
  141. assertTrue(b.isPeeled());
  142. assertEquals(bCommit, b.getObjectId().name());
  143. assertTrue(db.getRefDatabase().hasFastTipsWithSha1());
  144. }
  145. @Test
  146. public void testConvertToRefdir() throws Exception {
  147. db.convertToPackedRefs(false);
  148. assertTrue(db.getRefDatabase() instanceof RefDirectory);
  149. Ref h = db.exactRef("HEAD");
  150. assertTrue(h.isSymbolic());
  151. assertEquals("refs/heads/master", h.getTarget().getName());
  152. Ref b = db.exactRef("refs/heads/b");
  153. assertFalse(b.isSymbolic());
  154. assertTrue(b.isPeeled());
  155. assertEquals(bCommit, b.getObjectId().name());
  156. assertFalse(db.getRefDatabase().hasFastTipsWithSha1());
  157. }
  158. @Test
  159. public void testBatchrefUpdate() throws Exception {
  160. ObjectId cur = db.resolve("master");
  161. ObjectId prev = db.resolve("master^");
  162. PersonIdent person = new PersonIdent("name", "mail@example.com");
  163. ReceiveCommand rc1 = new ReceiveCommand(ObjectId.zeroId(), cur, "refs/heads/batch1");
  164. ReceiveCommand rc2 = new ReceiveCommand(ObjectId.zeroId(), prev, "refs/heads/batch2");
  165. String msg = "message";
  166. try (RevWalk rw = new RevWalk(db)) {
  167. db.getRefDatabase().newBatchUpdate()
  168. .addCommand(rc1, rc2)
  169. .setAtomic(true)
  170. .setRefLogIdent(person)
  171. .setRefLogMessage(msg, false)
  172. .execute(rw, NullProgressMonitor.INSTANCE);
  173. }
  174. assertEquals(rc1.getResult(), ReceiveCommand.Result.OK);
  175. assertEquals(rc2.getResult(), ReceiveCommand.Result.OK);
  176. ReflogEntry e = db.getReflogReader("refs/heads/batch1").getLastEntry();
  177. assertEquals(msg, e.getComment());
  178. assertEquals(person, e.getWho());
  179. assertEquals(cur, e.getNewId());
  180. e = db.getReflogReader("refs/heads/batch2").getLastEntry();
  181. assertEquals(msg, e.getComment());
  182. assertEquals(person, e.getWho());
  183. assertEquals(prev, e.getNewId());
  184. assertEquals(cur, db.exactRef("refs/heads/batch1").getObjectId());
  185. assertEquals(prev, db.exactRef("refs/heads/batch2").getObjectId());
  186. }
  187. @Test
  188. public void testFastforwardStatus() throws Exception {
  189. ObjectId cur = db.resolve("master");
  190. ObjectId prev = db.resolve("master^");
  191. RefUpdate u = db.updateRef("refs/heads/master");
  192. u.setNewObjectId(prev);
  193. u.setForceUpdate(true);
  194. assertEquals(FORCED, u.update());
  195. RefUpdate u2 = db.updateRef("refs/heads/master");
  196. u2.setNewObjectId(cur);
  197. assertEquals(FAST_FORWARD, u2.update());
  198. }
  199. @Test
  200. public void testUpdateChecksOldValue() throws Exception {
  201. ObjectId cur = db.resolve("master");
  202. ObjectId prev = db.resolve("master^");
  203. RefUpdate u1 = db.updateRef("refs/heads/master");
  204. RefUpdate u2 = db.updateRef("refs/heads/master");
  205. u1.setExpectedOldObjectId(cur);
  206. u1.setNewObjectId(prev);
  207. u1.setForceUpdate(true);
  208. u2.setExpectedOldObjectId(cur);
  209. u2.setNewObjectId(prev);
  210. u2.setForceUpdate(true);
  211. assertEquals(FORCED, u1.update());
  212. assertEquals(LOCK_FAILURE, u2.update());
  213. }
  214. @Test
  215. public void testWritesymref() throws Exception {
  216. writeSymref(Constants.HEAD, "refs/heads/a");
  217. assertNotNull(db.exactRef("refs/heads/b"));
  218. }
  219. @Test
  220. public void testFastforwardStatus2() throws Exception {
  221. writeSymref(Constants.HEAD, "refs/heads/a");
  222. ObjectId bId = db.exactRef("refs/heads/b").getObjectId();
  223. RefUpdate u = db.updateRef("refs/heads/a");
  224. u.setNewObjectId(bId);
  225. u.setRefLogMessage("Setup", false);
  226. assertEquals(FAST_FORWARD, u.update());
  227. }
  228. @Test
  229. public void testDelete() throws Exception {
  230. RefUpdate up = db.getRefDatabase().newUpdate("refs/heads/a", false);
  231. up.setForceUpdate(true);
  232. RefUpdate.Result res = up.delete();
  233. assertEquals(res, FORCED);
  234. assertNull(db.exactRef("refs/heads/a"));
  235. }
  236. @Test
  237. public void testDeleteWithoutHead() throws IOException {
  238. // Prepare repository without HEAD
  239. RefUpdate refUpdate = db.updateRef(Constants.HEAD, true);
  240. refUpdate.setForceUpdate(true);
  241. refUpdate.setNewObjectId(ObjectId.zeroId());
  242. RefUpdate.Result updateResult = refUpdate.update();
  243. assertEquals(FORCED, updateResult);
  244. Ref r = db.exactRef("HEAD");
  245. assertEquals(ObjectId.zeroId(), r.getObjectId());
  246. RefUpdate.Result deleteHeadResult = db.updateRef(Constants.HEAD)
  247. .delete();
  248. // why does doDelete say NEW ?
  249. assertEquals(RefUpdate.Result.NO_CHANGE, deleteHeadResult);
  250. // Any result is ok as long as it's not an NPE
  251. db.updateRef(Constants.R_HEADS + "master").delete();
  252. }
  253. @Test
  254. public void testUpdateRefDetached() throws Exception {
  255. ObjectId pid = db.resolve("refs/heads/master");
  256. ObjectId ppid = db.resolve("refs/heads/master^");
  257. RefUpdate updateRef = db.updateRef("HEAD", true);
  258. updateRef.setForceUpdate(true);
  259. updateRef.setNewObjectId(ppid);
  260. RefUpdate.Result update = updateRef.update();
  261. assertEquals(FORCED, update);
  262. assertEquals(ppid, db.resolve("HEAD"));
  263. Ref ref = db.exactRef("HEAD");
  264. assertEquals("HEAD", ref.getName());
  265. assertTrue("is detached", !ref.isSymbolic());
  266. // the branch HEAD referred to is left untouched
  267. assertEquals(pid, db.resolve("refs/heads/master"));
  268. ReflogReader reflogReader = db.getReflogReader("HEAD");
  269. ReflogEntry e = reflogReader.getReverseEntries().get(0);
  270. assertEquals(ppid, e.getNewId());
  271. assertEquals("GIT_COMMITTER_EMAIL", e.getWho().getEmailAddress());
  272. assertEquals("GIT_COMMITTER_NAME", e.getWho().getName());
  273. assertEquals(1250379778000L, e.getWho().getWhen().getTime());
  274. assertEquals(pid, e.getOldId());
  275. }
  276. @Test
  277. public void testWriteReflog() throws Exception {
  278. ObjectId pid = db.resolve("refs/heads/master^");
  279. RefUpdate updateRef = db.updateRef("refs/heads/master");
  280. updateRef.setNewObjectId(pid);
  281. String msg = "REFLOG!";
  282. updateRef.setRefLogMessage(msg, true);
  283. PersonIdent person = new PersonIdent("name", "mail@example.com");
  284. updateRef.setRefLogIdent(person);
  285. updateRef.setForceUpdate(true);
  286. RefUpdate.Result update = updateRef.update();
  287. assertEquals(FORCED, update); // internal
  288. ReflogReader r = db.getReflogReader("refs/heads/master");
  289. ReflogEntry e = r.getLastEntry();
  290. assertEquals(e.getNewId(), pid);
  291. assertEquals(e.getComment(), "REFLOG!: FORCED");
  292. assertEquals(e.getWho(), person);
  293. }
  294. @Test
  295. public void testLooseDelete() throws IOException {
  296. final String newRef = "refs/heads/abc";
  297. assertNull(db.exactRef(newRef));
  298. RefUpdate ref = db.updateRef(newRef);
  299. ObjectId nonZero = db.resolve(Constants.HEAD);
  300. assertNotEquals(nonZero, ObjectId.zeroId());
  301. ref.setNewObjectId(nonZero);
  302. assertEquals(RefUpdate.Result.NEW, ref.update());
  303. ref = db.updateRef(newRef);
  304. ref.setNewObjectId(db.resolve(Constants.HEAD));
  305. assertEquals(ref.delete(), RefUpdate.Result.NO_CHANGE);
  306. // Differs from RefupdateTest. Deleting a loose ref leaves reflog trail.
  307. ReflogReader reader = db.getReflogReader("refs/heads/abc");
  308. assertEquals(ObjectId.zeroId(), reader.getReverseEntry(1).getOldId());
  309. assertEquals(nonZero, reader.getReverseEntry(1).getNewId());
  310. assertEquals(nonZero, reader.getReverseEntry(0).getOldId());
  311. assertEquals(ObjectId.zeroId(), reader.getReverseEntry(0).getNewId());
  312. }
  313. private static class SubclassedId extends ObjectId {
  314. SubclassedId(AnyObjectId src) {
  315. super(src);
  316. }
  317. }
  318. @Test
  319. public void testNoCacheObjectIdSubclass() throws IOException {
  320. final String newRef = "refs/heads/abc";
  321. final RefUpdate ru = updateRef(newRef);
  322. final SubclassedId newid = new SubclassedId(ru.getNewObjectId());
  323. ru.setNewObjectId(newid);
  324. RefUpdate.Result update = ru.update();
  325. assertEquals(RefUpdate.Result.NEW, update);
  326. Ref r = db.exactRef(newRef);
  327. assertEquals(newRef, r.getName());
  328. assertNotNull(r.getObjectId());
  329. assertNotSame(newid, r.getObjectId());
  330. assertSame(ObjectId.class, r.getObjectId().getClass());
  331. assertEquals(newid, r.getObjectId());
  332. List<ReflogEntry> reverseEntries1 = db.getReflogReader("refs/heads/abc")
  333. .getReverseEntries();
  334. ReflogEntry entry1 = reverseEntries1.get(0);
  335. assertEquals(1, reverseEntries1.size());
  336. assertEquals(ObjectId.zeroId(), entry1.getOldId());
  337. assertEquals(r.getObjectId(), entry1.getNewId());
  338. assertEquals(new PersonIdent(db).toString(),
  339. entry1.getWho().toString());
  340. assertEquals("", entry1.getComment());
  341. List<ReflogEntry> reverseEntries2 = db.getReflogReader("HEAD")
  342. .getReverseEntries();
  343. assertEquals(0, reverseEntries2.size());
  344. }
  345. @Test
  346. public void testDeleteSymref() throws IOException {
  347. RefUpdate dst = updateRef("refs/heads/abc");
  348. assertEquals(RefUpdate.Result.NEW, dst.update());
  349. ObjectId id = dst.getNewObjectId();
  350. RefUpdate u = db.updateRef("refs/symref");
  351. assertEquals(RefUpdate.Result.NEW, u.link(dst.getName()));
  352. Ref ref = db.exactRef(u.getName());
  353. assertNotNull(ref);
  354. assertTrue(ref.isSymbolic());
  355. assertEquals(dst.getName(), ref.getLeaf().getName());
  356. assertEquals(id, ref.getLeaf().getObjectId());
  357. u = db.updateRef(u.getName());
  358. u.setDetachingSymbolicRef();
  359. u.setForceUpdate(true);
  360. assertEquals(FORCED, u.delete());
  361. assertNull(db.exactRef(u.getName()));
  362. ref = db.exactRef(dst.getName());
  363. assertNotNull(ref);
  364. assertFalse(ref.isSymbolic());
  365. assertEquals(id, ref.getObjectId());
  366. }
  367. @Test
  368. public void writeUnbornHead() throws Exception {
  369. RefUpdate.Result r = db.updateRef("HEAD").link("refs/heads/unborn");
  370. assertEquals(FORCED, r);
  371. Ref head = db.exactRef("HEAD");
  372. assertTrue(head.isSymbolic());
  373. assertEquals(head.getTarget().getName(), "refs/heads/unborn");
  374. }
  375. /**
  376. * Update the HEAD ref when the referenced branch is unborn
  377. *
  378. * @throws Exception
  379. */
  380. @Test
  381. public void testUpdateRefDetachedUnbornHead() throws Exception {
  382. ObjectId ppid = db.resolve("refs/heads/master^");
  383. writeSymref("HEAD", "refs/heads/unborn");
  384. RefUpdate updateRef = db.updateRef("HEAD", true);
  385. updateRef.setForceUpdate(true);
  386. updateRef.setNewObjectId(ppid);
  387. RefUpdate.Result update = updateRef.update();
  388. assertEquals(RefUpdate.Result.NEW, update);
  389. assertEquals(ppid, db.resolve("HEAD"));
  390. Ref ref = db.exactRef("HEAD");
  391. assertEquals("HEAD", ref.getName());
  392. assertTrue("is detached", !ref.isSymbolic());
  393. // the branch HEAD referred to is left untouched
  394. assertNull(db.resolve("refs/heads/unborn"));
  395. ReflogReader reflogReader = db.getReflogReader("HEAD");
  396. ReflogEntry e = reflogReader.getReverseEntries().get(0);
  397. assertEquals(ObjectId.zeroId(), e.getOldId());
  398. assertEquals(ppid, e.getNewId());
  399. assertEquals("GIT_COMMITTER_EMAIL", e.getWho().getEmailAddress());
  400. assertEquals("GIT_COMMITTER_NAME", e.getWho().getName());
  401. assertEquals(1250379778000L, e.getWho().getWhen().getTime());
  402. }
  403. @Test
  404. public void testDeleteNotFound() throws IOException {
  405. RefUpdate ref = updateRef("refs/heads/doesnotexist");
  406. assertNull(db.exactRef(ref.getName()));
  407. assertEquals(RefUpdate.Result.NEW, ref.delete());
  408. assertNull(db.exactRef(ref.getName()));
  409. }
  410. @Test
  411. public void testRenameSymref() throws IOException {
  412. db.resolve("HEAD");
  413. RefRename r = db.renameRef("HEAD", "KOPF");
  414. assertEquals(IO_FAILURE, r.rename());
  415. }
  416. @Test
  417. public void testRenameCurrentBranch() throws IOException {
  418. ObjectId rb = db.resolve("refs/heads/b");
  419. writeSymref(Constants.HEAD, "refs/heads/b");
  420. ObjectId oldHead = db.resolve(Constants.HEAD);
  421. assertEquals("internal test condition, b == HEAD", oldHead, rb);
  422. RefRename renameRef = db.renameRef("refs/heads/b",
  423. "refs/heads/new/name");
  424. RefUpdate.Result result = renameRef.rename();
  425. assertEquals(RefUpdate.Result.RENAMED, result);
  426. assertEquals(rb, db.resolve("refs/heads/new/name"));
  427. assertNull(db.resolve("refs/heads/b"));
  428. assertEquals(rb, db.resolve(Constants.HEAD));
  429. List<String> names = new ArrayList<>();
  430. names.add("HEAD");
  431. names.add("refs/heads/b");
  432. names.add("refs/heads/new/name");
  433. for (String nm : names) {
  434. ReflogReader rd = db.getReflogReader(nm);
  435. assertNotNull(rd);
  436. ReflogEntry last = rd.getLastEntry();
  437. ObjectId id = last.getNewId();
  438. assertTrue(ObjectId.zeroId().equals(id) || rb.equals(id));
  439. id = last.getNewId();
  440. assertTrue(ObjectId.zeroId().equals(id) || rb.equals(id));
  441. String want = "Branch: renamed b to new/name";
  442. assertEquals(want, last.getComment());
  443. }
  444. }
  445. @Test
  446. public void isGitRepository() {
  447. assertTrue(RepositoryCache.FileKey.isGitRepository(db.getDirectory(), db.getFS()));
  448. }
  449. @Test
  450. public void testRenameDestExists() throws IOException {
  451. ObjectId rb = db.resolve("refs/heads/b");
  452. writeSymref(Constants.HEAD, "refs/heads/b");
  453. ObjectId oldHead = db.resolve(Constants.HEAD);
  454. assertEquals("internal test condition, b == HEAD", oldHead, rb);
  455. RefRename renameRef = db.renameRef("refs/heads/b", "refs/heads/a");
  456. RefUpdate.Result result = renameRef.rename();
  457. assertEquals(RefUpdate.Result.LOCK_FAILURE, result);
  458. }
  459. @Test
  460. public void testRenameAtomic() throws IOException {
  461. ObjectId prevId = db.resolve("refs/heads/master^");
  462. RefRename rename = db.renameRef("refs/heads/master",
  463. "refs/heads/newmaster");
  464. RefUpdate updateRef = db.updateRef("refs/heads/master");
  465. updateRef.setNewObjectId(prevId);
  466. updateRef.setForceUpdate(true);
  467. assertEquals(FORCED, updateRef.update());
  468. assertEquals(RefUpdate.Result.LOCK_FAILURE, rename.rename());
  469. }
  470. @Test
  471. public void reftableRefsStorageClass() throws IOException {
  472. Ref b = db.exactRef("refs/heads/b");
  473. assertEquals(Ref.Storage.PACKED, b.getStorage());
  474. }
  475. private RefUpdate updateRef(String name) throws IOException {
  476. final RefUpdate ref = db.updateRef(name);
  477. ref.setNewObjectId(db.resolve(Constants.HEAD));
  478. return ref;
  479. }
  480. private void writeSymref(String src, String dst) throws IOException {
  481. RefUpdate u = db.updateRef(src);
  482. switch (u.link(dst)) {
  483. case NEW:
  484. case FORCED:
  485. case NO_CHANGE:
  486. break;
  487. default:
  488. fail("link " + src + " to " + dst);
  489. }
  490. }
  491. }