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.

DatabaseTest.java 34KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091
  1. /*
  2. Copyright (c) 2007 Health Market Science, Inc.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package com.healthmarketscience.jackcess;
  14. import java.io.File;
  15. import java.io.FileNotFoundException;
  16. import java.io.IOException;
  17. import java.math.BigDecimal;
  18. import java.text.DateFormat;
  19. import java.text.SimpleDateFormat;
  20. import java.time.ZoneId;
  21. import java.util.ArrayList;
  22. import java.util.Arrays;
  23. import java.util.Calendar;
  24. import java.util.Date;
  25. import java.util.HashSet;
  26. import java.util.LinkedHashMap;
  27. import java.util.List;
  28. import java.util.Map;
  29. import java.util.Set;
  30. import java.util.TimeZone;
  31. import java.util.TreeSet;
  32. import java.util.UUID;
  33. import static com.healthmarketscience.jackcess.Database.*;
  34. import com.healthmarketscience.jackcess.impl.ColumnImpl;
  35. import com.healthmarketscience.jackcess.impl.DatabaseImpl;
  36. import static com.healthmarketscience.jackcess.impl.JetFormatTest.*;
  37. import com.healthmarketscience.jackcess.impl.RowIdImpl;
  38. import com.healthmarketscience.jackcess.impl.RowImpl;
  39. import com.healthmarketscience.jackcess.impl.TableImpl;
  40. import com.healthmarketscience.jackcess.util.LinkResolver;
  41. import com.healthmarketscience.jackcess.util.RowFilterTest;
  42. import junit.framework.TestCase;
  43. import static com.healthmarketscience.jackcess.TestUtil.*;
  44. /**
  45. * @author Tim McCune
  46. */
  47. @SuppressWarnings("deprecation")
  48. public class DatabaseTest extends TestCase
  49. {
  50. public DatabaseTest(String name) throws Exception {
  51. super(name);
  52. }
  53. public void testInvalidTableDefs() throws Exception {
  54. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  55. Database db = create(fileFormat);
  56. try {
  57. new TableBuilder("test").toTable(db);
  58. fail("created table with no columns?");
  59. } catch(IllegalArgumentException e) {
  60. // success
  61. }
  62. try {
  63. new TableBuilder("test")
  64. .addColumn(new ColumnBuilder("A", DataType.TEXT))
  65. .addColumn(new ColumnBuilder("a", DataType.MEMO))
  66. .toTable(db);
  67. fail("created table with duplicate column names?");
  68. } catch(IllegalArgumentException e) {
  69. // success
  70. }
  71. try {
  72. new TableBuilder("test")
  73. .addColumn(new ColumnBuilder("A", DataType.TEXT)
  74. .setLengthInUnits(352))
  75. .toTable(db);
  76. fail("created table with invalid column length?");
  77. } catch(IllegalArgumentException e) {
  78. // success
  79. }
  80. try {
  81. new TableBuilder("test")
  82. .addColumn(new ColumnBuilder("A_" + createString(70), DataType.TEXT))
  83. .toTable(db);
  84. fail("created table with too long column name?");
  85. } catch(IllegalArgumentException e) {
  86. // success
  87. }
  88. new TableBuilder("test")
  89. .addColumn(new ColumnBuilder("A", DataType.TEXT))
  90. .toTable(db);
  91. try {
  92. new TableBuilder("Test")
  93. .addColumn(new ColumnBuilder("A", DataType.TEXT))
  94. .toTable(db);
  95. fail("create duplicate tables?");
  96. } catch(IllegalArgumentException e) {
  97. // success
  98. }
  99. db.close();
  100. }
  101. }
  102. public void testReadDeletedRows() throws Exception {
  103. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.DEL, true)) {
  104. Table table = open(testDB).getTable("Table");
  105. int rows = 0;
  106. while (table.getNextRow() != null) {
  107. rows++;
  108. }
  109. assertEquals(2, rows);
  110. table.getDatabase().close();
  111. }
  112. }
  113. public void testGetColumns() throws Exception {
  114. for (final TestDB testDB : SUPPORTED_DBS_TEST_FOR_READ) {
  115. List<? extends Column> columns = open(testDB).getTable("Table1").getColumns();
  116. assertEquals(9, columns.size());
  117. checkColumn(columns, 0, "A", DataType.TEXT);
  118. checkColumn(columns, 1, "B", DataType.TEXT);
  119. checkColumn(columns, 2, "C", DataType.BYTE);
  120. checkColumn(columns, 3, "D", DataType.INT);
  121. checkColumn(columns, 4, "E", DataType.LONG);
  122. checkColumn(columns, 5, "F", DataType.DOUBLE);
  123. checkColumn(columns, 6, "G", DataType.SHORT_DATE_TIME);
  124. checkColumn(columns, 7, "H", DataType.MONEY);
  125. checkColumn(columns, 8, "I", DataType.BOOLEAN);
  126. }
  127. }
  128. private static void checkColumn(
  129. List<? extends Column> columns, int columnNumber, String name,
  130. DataType dataType)
  131. throws Exception
  132. {
  133. Column column = columns.get(columnNumber);
  134. assertEquals(name, column.getName());
  135. assertEquals(dataType, column.getType());
  136. }
  137. public void testGetNextRow() throws Exception {
  138. for (final TestDB testDB : SUPPORTED_DBS_TEST_FOR_READ) {
  139. final Database db = open(testDB);
  140. assertEquals(4, db.getTableNames().size());
  141. final Table table = db.getTable("Table1");
  142. Row row1 = table.getNextRow();
  143. Row row2 = table.getNextRow();
  144. if(!"abcdefg".equals(row1.get("A"))) {
  145. Row tmpRow = row1;
  146. row1 = row2;
  147. row2 = tmpRow;
  148. }
  149. checkTestDBTable1RowABCDEFG(testDB, table, row1);
  150. checkTestDBTable1RowA(testDB, table, row2);
  151. db.close();
  152. }
  153. }
  154. public void testCreate() throws Exception {
  155. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  156. Database db = create(fileFormat);
  157. assertEquals(0, db.getTableNames().size());
  158. db.close();
  159. }
  160. }
  161. public void testDeleteCurrentRow() throws Exception {
  162. // make sure correct row is deleted
  163. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  164. Database db = createMem(fileFormat);
  165. createTestTable(db);
  166. Map<String,Object> row1 = createTestRowMap("Tim1");
  167. Map<String,Object> row2 = createTestRowMap("Tim2");
  168. Map<String,Object> row3 = createTestRowMap("Tim3");
  169. Table table = db.getTable("Test");
  170. @SuppressWarnings("unchecked")
  171. List<Map<String,Object>> rows = Arrays.asList(row1, row2, row3);
  172. table.addRowsFromMaps(rows);
  173. assertRowCount(3, table);
  174. table.reset();
  175. table.getNextRow();
  176. table.getNextRow();
  177. table.getDefaultCursor().deleteCurrentRow();
  178. table.reset();
  179. Map<String, Object> outRow = table.getNextRow();
  180. assertEquals("Tim1", outRow.get("A"));
  181. outRow = table.getNextRow();
  182. assertEquals("Tim3", outRow.get("A"));
  183. assertRowCount(2, table);
  184. db.close();
  185. // test multi row delete/add
  186. db = createMem(fileFormat);
  187. createTestTable(db);
  188. Object[] row = createTestRow();
  189. table = db.getTable("Test");
  190. for (int i = 0; i < 10; i++) {
  191. row[3] = i;
  192. table.addRow(row);
  193. }
  194. row[3] = 1974;
  195. assertRowCount(10, table);
  196. table.reset();
  197. table.getNextRow();
  198. table.getDefaultCursor().deleteCurrentRow();
  199. assertRowCount(9, table);
  200. table.reset();
  201. table.getNextRow();
  202. table.getDefaultCursor().deleteCurrentRow();
  203. assertRowCount(8, table);
  204. table.reset();
  205. for (int i = 0; i < 8; i++) {
  206. table.getNextRow();
  207. }
  208. table.getDefaultCursor().deleteCurrentRow();
  209. assertRowCount(7, table);
  210. table.addRow(row);
  211. assertRowCount(8, table);
  212. table.reset();
  213. for (int i = 0; i < 3; i++) {
  214. table.getNextRow();
  215. }
  216. table.getDefaultCursor().deleteCurrentRow();
  217. assertRowCount(7, table);
  218. table.reset();
  219. assertEquals(2, table.getNextRow().get("D"));
  220. db.close();
  221. }
  222. }
  223. public void testDeleteRow() throws Exception {
  224. // make sure correct row is deleted
  225. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  226. Database db = createMem(fileFormat);
  227. createTestTable(db);
  228. Table table = db.getTable("Test");
  229. for(int i = 0; i < 10; ++i) {
  230. table.addRowFromMap(createTestRowMap("Tim" + i));
  231. }
  232. assertRowCount(10, table);
  233. table.reset();
  234. List<Row> rows = RowFilterTest.toList(table);
  235. Row r1 = rows.remove(7);
  236. Row r2 = rows.remove(3);
  237. assertEquals(8, rows.size());
  238. assertSame(r2, table.deleteRow(r2));
  239. assertSame(r1, table.deleteRow(r1));
  240. assertTable(rows, table);
  241. table.deleteRow(r2);
  242. table.deleteRow(r1);
  243. assertTable(rows, table);
  244. }
  245. }
  246. public void testMissingFile() throws Exception {
  247. File bogusFile = new File("fooby-dooby.mdb");
  248. assertTrue(!bogusFile.exists());
  249. try {
  250. new DatabaseBuilder(bogusFile).setReadOnly(true).
  251. setAutoSync(getTestAutoSync()).open();
  252. fail("FileNotFoundException should have been thrown");
  253. } catch(FileNotFoundException e) {
  254. }
  255. assertTrue(!bogusFile.exists());
  256. }
  257. public void testReadWithDeletedCols() throws Exception {
  258. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.DEL_COL, true)) {
  259. Table table = open(testDB).getTable("Table1");
  260. Map<String, Object> expectedRow0 = new LinkedHashMap<String, Object>();
  261. expectedRow0.put("id", 0);
  262. expectedRow0.put("id2", 2);
  263. expectedRow0.put("data", "foo");
  264. expectedRow0.put("data2", "foo2");
  265. Map<String, Object> expectedRow1 = new LinkedHashMap<String, Object>();
  266. expectedRow1.put("id", 3);
  267. expectedRow1.put("id2", 5);
  268. expectedRow1.put("data", "bar");
  269. expectedRow1.put("data2", "bar2");
  270. int rowNum = 0;
  271. Map<String, Object> row = null;
  272. while ((row = table.getNextRow()) != null) {
  273. if(rowNum == 0) {
  274. assertEquals(expectedRow0, row);
  275. } else if(rowNum == 1) {
  276. assertEquals(expectedRow1, row);
  277. } else if(rowNum >= 2) {
  278. fail("should only have 2 rows");
  279. }
  280. rowNum++;
  281. }
  282. table.getDatabase().close();
  283. }
  284. }
  285. public void testCurrency() throws Exception {
  286. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  287. Database db = create(fileFormat);
  288. Table table = new TableBuilder("test")
  289. .addColumn(new ColumnBuilder("A", DataType.MONEY))
  290. .toTable(db);
  291. table.addRow(new BigDecimal("-2341234.03450"));
  292. table.addRow(37L);
  293. table.addRow("10000.45");
  294. table.reset();
  295. List<Object> foundValues = new ArrayList<Object>();
  296. Map<String, Object> row = null;
  297. while((row = table.getNextRow()) != null) {
  298. foundValues.add(row.get("A"));
  299. }
  300. assertEquals(Arrays.asList(
  301. new BigDecimal("-2341234.0345"),
  302. new BigDecimal("37.0000"),
  303. new BigDecimal("10000.4500")),
  304. foundValues);
  305. try {
  306. table.addRow(new BigDecimal("342523234145343543.3453"));
  307. fail("IOException should have been thrown");
  308. } catch(IOException e) {
  309. // ignored
  310. }
  311. db.close();
  312. }
  313. }
  314. public void testGUID() throws Exception
  315. {
  316. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  317. Database db = create(fileFormat);
  318. Table table = new TableBuilder("test")
  319. .addColumn(new ColumnBuilder("A", DataType.GUID))
  320. .toTable(db);
  321. table.addRow("{32A59F01-AA34-3E29-453F-4523453CD2E6}");
  322. table.addRow("{32a59f01-aa34-3e29-453f-4523453cd2e6}");
  323. table.addRow("{11111111-1111-1111-1111-111111111111}");
  324. table.addRow(" {FFFFFFFF-FFFF-FFFF-FFFF-FFFFFFFFFFFF} ");
  325. table.addRow(UUID.fromString("32a59f01-1234-3e29-4aaf-4523453cd2e6"));
  326. table.reset();
  327. List<Object> foundValues = new ArrayList<Object>();
  328. Map<String, Object> row = null;
  329. while((row = table.getNextRow()) != null) {
  330. foundValues.add(row.get("A"));
  331. }
  332. assertEquals(Arrays.asList(
  333. "{32A59F01-AA34-3E29-453F-4523453CD2E6}",
  334. "{32A59F01-AA34-3E29-453F-4523453CD2E6}",
  335. "{11111111-1111-1111-1111-111111111111}",
  336. "{FFFFFFFF-FFFF-FFFF-FFFF-FFFFFFFFFFFF}",
  337. "{32A59F01-1234-3E29-4AAF-4523453CD2E6}"),
  338. foundValues);
  339. try {
  340. table.addRow("3245234");
  341. fail("IOException should have been thrown");
  342. } catch(IOException e) {
  343. // ignored
  344. }
  345. db.close();
  346. }
  347. }
  348. public void testNumeric() throws Exception
  349. {
  350. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  351. Database db = create(fileFormat);
  352. ColumnBuilder col = new ColumnBuilder("A", DataType.NUMERIC)
  353. .setScale(4).setPrecision(8).toColumn();
  354. assertTrue(col.getType().isVariableLength());
  355. Table table = new TableBuilder("test")
  356. .addColumn(col)
  357. .addColumn(new ColumnBuilder("B", DataType.NUMERIC)
  358. .setScale(8).setPrecision(28))
  359. .toTable(db);
  360. table.addRow(new BigDecimal("-1234.03450"),
  361. new BigDecimal("23923434453436.36234219"));
  362. table.addRow(37L, 37L);
  363. table.addRow("1000.45", "-3452345321000");
  364. table.reset();
  365. List<Object> foundSmallValues = new ArrayList<Object>();
  366. List<Object> foundBigValues = new ArrayList<Object>();
  367. Map<String, Object> row = null;
  368. while((row = table.getNextRow()) != null) {
  369. foundSmallValues.add(row.get("A"));
  370. foundBigValues.add(row.get("B"));
  371. }
  372. assertEquals(Arrays.asList(
  373. new BigDecimal("-1234.0345"),
  374. new BigDecimal("37.0000"),
  375. new BigDecimal("1000.4500")),
  376. foundSmallValues);
  377. assertEquals(Arrays.asList(
  378. new BigDecimal("23923434453436.36234219"),
  379. new BigDecimal("37.00000000"),
  380. new BigDecimal("-3452345321000.00000000")),
  381. foundBigValues);
  382. try {
  383. table.addRow(new BigDecimal("3245234.234"),
  384. new BigDecimal("3245234.234"));
  385. fail("IOException should have been thrown");
  386. } catch(IOException e) {
  387. // ignored
  388. }
  389. db.close();
  390. }
  391. }
  392. public void testFixedNumeric() throws Exception
  393. {
  394. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.FIXED_NUMERIC)) {
  395. Database db = openCopy(testDB);
  396. Table t = db.getTable("test");
  397. boolean first = true;
  398. for(Column col : t.getColumns()) {
  399. if(first) {
  400. assertTrue(col.isVariableLength());
  401. assertEquals(DataType.MEMO, col.getType());
  402. first = false;
  403. } else {
  404. assertFalse(col.isVariableLength());
  405. assertEquals(DataType.NUMERIC, col.getType());
  406. }
  407. }
  408. Map<String, Object> row = t.getNextRow();
  409. assertEquals("some data", row.get("col1"));
  410. assertEquals(new BigDecimal("1"), row.get("col2"));
  411. assertEquals(new BigDecimal("0"), row.get("col3"));
  412. assertEquals(new BigDecimal("0"), row.get("col4"));
  413. assertEquals(new BigDecimal("4"), row.get("col5"));
  414. assertEquals(new BigDecimal("-1"), row.get("col6"));
  415. assertEquals(new BigDecimal("1"), row.get("col7"));
  416. Object[] tmpRow = new Object[]{
  417. "foo", new BigDecimal("1"), new BigDecimal(3), new BigDecimal("13"),
  418. new BigDecimal("-17"), new BigDecimal("0"), new BigDecimal("8734")};
  419. t.addRow(tmpRow);
  420. t.reset();
  421. t.getNextRow();
  422. row = t.getNextRow();
  423. assertEquals(tmpRow[0], row.get("col1"));
  424. assertEquals(tmpRow[1], row.get("col2"));
  425. assertEquals(tmpRow[2], row.get("col3"));
  426. assertEquals(tmpRow[3], row.get("col4"));
  427. assertEquals(tmpRow[4], row.get("col5"));
  428. assertEquals(tmpRow[5], row.get("col6"));
  429. assertEquals(tmpRow[6], row.get("col7"));
  430. db.close();
  431. }
  432. }
  433. public void testMultiPageTableDef() throws Exception
  434. {
  435. for (final TestDB testDB : SUPPORTED_DBS_TEST_FOR_READ) {
  436. List<? extends Column> columns = open(testDB).getTable("Table2").getColumns();
  437. assertEquals(89, columns.size());
  438. }
  439. }
  440. public void testOverflow() throws Exception
  441. {
  442. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.OVERFLOW, true)) {
  443. Database mdb = open(testDB);
  444. Table table = mdb.getTable("Table1");
  445. // 7 rows, 3 and 5 are overflow
  446. table.getNextRow();
  447. table.getNextRow();
  448. Map<String, Object> row = table.getNextRow();
  449. assertEquals(Arrays.<Object>asList(
  450. null, "row3col3", null, null, null, null, null,
  451. "row3col9", null),
  452. new ArrayList<Object>(row.values()));
  453. table.getNextRow();
  454. row = table.getNextRow();
  455. assertEquals(Arrays.<Object>asList(
  456. null, "row5col2", null, null, null, null, null, null,
  457. null),
  458. new ArrayList<Object>(row.values()));
  459. table.reset();
  460. assertRowCount(7, table);
  461. mdb.close();
  462. }
  463. }
  464. public void testUsageMapPromotion() throws Exception {
  465. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.PROMOTION)) {
  466. Database db = openMem(testDB);
  467. Table t = db.getTable("jobDB1");
  468. assertTrue(((TableImpl)t).getOwnedPagesCursor().getUsageMap().toString()
  469. .startsWith("InlineHandler"));
  470. String lval = createNonAsciiString(255); // "--255 chars long text--";
  471. ((DatabaseImpl)db).getPageChannel().startWrite();
  472. try {
  473. for(int i = 0; i < 1000; ++i) {
  474. t.addRow(i, 13, 57, lval, lval, lval, lval, lval, lval, 47.0d);
  475. }
  476. } finally {
  477. ((DatabaseImpl)db).getPageChannel().finishWrite();
  478. }
  479. Set<Integer> ids = new HashSet<Integer>();
  480. for(Row row : t) {
  481. ids.add(row.getInt("ID"));
  482. }
  483. assertEquals(1000, ids.size());
  484. assertTrue(((TableImpl)t).getOwnedPagesCursor().getUsageMap().toString()
  485. .startsWith("ReferenceHandler"));
  486. db.close();
  487. }
  488. }
  489. public void testLargeTableDef() throws Exception {
  490. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  491. Database db = create(fileFormat);
  492. final int numColumns = 90;
  493. List<ColumnBuilder> columns = new ArrayList<ColumnBuilder>();
  494. List<String> colNames = new ArrayList<String>();
  495. for(int i = 0; i < numColumns; ++i) {
  496. String colName = "MyColumnName" + i;
  497. colNames.add(colName);
  498. columns.add(new ColumnBuilder(colName, DataType.TEXT).toColumn());
  499. }
  500. Table t = new TableBuilder("test")
  501. .addColumns(columns)
  502. .toTable(db);
  503. List<String> row = new ArrayList<String>();
  504. Map<String,Object> expectedRowData = new LinkedHashMap<String, Object>();
  505. for(int i = 0; i < numColumns; ++i) {
  506. String value = "" + i + " some row data";
  507. row.add(value);
  508. expectedRowData.put(colNames.get(i), value);
  509. }
  510. t.addRow(row.toArray());
  511. t.reset();
  512. assertEquals(expectedRowData, t.getNextRow());
  513. db.close();
  514. }
  515. }
  516. public void testWriteAndReadDate() throws Exception {
  517. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  518. Database db = createMem(fileFormat);
  519. Table table = new TableBuilder("test")
  520. .addColumn(new ColumnBuilder("name", DataType.TEXT))
  521. .addColumn(new ColumnBuilder("date", DataType.SHORT_DATE_TIME))
  522. .toTable(db);
  523. // since jackcess does not really store millis, shave them off before
  524. // storing the current date/time
  525. long curTimeNoMillis = (System.currentTimeMillis() / 1000L);
  526. curTimeNoMillis *= 1000L;
  527. DateFormat df = new SimpleDateFormat("yyyyMMdd HH:mm:ss");
  528. List<Date> dates =
  529. new ArrayList<Date>(
  530. Arrays.asList(
  531. df.parse("19801231 00:00:00"),
  532. df.parse("19930513 14:43:27"),
  533. null,
  534. df.parse("20210102 02:37:00"),
  535. new Date(curTimeNoMillis)));
  536. Calendar c = Calendar.getInstance();
  537. for(int year = 1801; year < 2050; year +=3) {
  538. for(int month = 0; month <= 12; ++month) {
  539. for(int day = 1; day < 29; day += 3) {
  540. c.clear();
  541. c.set(Calendar.YEAR, year);
  542. c.set(Calendar.MONTH, month);
  543. c.set(Calendar.DAY_OF_MONTH, day);
  544. dates.add(c.getTime());
  545. }
  546. }
  547. }
  548. ((DatabaseImpl)db).getPageChannel().startWrite();
  549. try {
  550. for(Date d : dates) {
  551. table.addRow("row " + d, d);
  552. }
  553. } finally {
  554. ((DatabaseImpl)db).getPageChannel().finishWrite();
  555. }
  556. List<Date> foundDates = new ArrayList<Date>();
  557. for(Row row : table) {
  558. foundDates.add(row.getDate("date"));
  559. }
  560. assertEquals(dates.size(), foundDates.size());
  561. for(int i = 0; i < dates.size(); ++i) {
  562. Date expected = dates.get(i);
  563. Date found = foundDates.get(i);
  564. assertSameDate(expected, found);
  565. }
  566. db.close();
  567. }
  568. }
  569. public void testAncientDates() throws Exception
  570. {
  571. TimeZone tz = TimeZone.getTimeZone("America/New_York");
  572. SimpleDateFormat sdf = DatabaseBuilder.createDateFormat("yyyy-MM-dd");
  573. sdf.getCalendar().setTimeZone(tz);
  574. List<String> dates = Arrays.asList("1582-10-15", "1582-10-14",
  575. "1492-01-10", "1392-01-10");
  576. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  577. Database db = createMem(fileFormat);
  578. db.setTimeZone(tz);
  579. Table table = new TableBuilder("test")
  580. .addColumn(new ColumnBuilder("name", DataType.TEXT))
  581. .addColumn(new ColumnBuilder("date", DataType.SHORT_DATE_TIME))
  582. .toTable(db);
  583. for(String dateStr : dates) {
  584. Date d = sdf.parse(dateStr);
  585. table.addRow("row " + dateStr, d);
  586. }
  587. List<String> foundDates = new ArrayList<String>();
  588. for(Row row : table) {
  589. foundDates.add(sdf.format(row.getDate("date")));
  590. }
  591. assertEquals(dates, foundDates);
  592. db.close();
  593. }
  594. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.OLD_DATES)) {
  595. Database db = openCopy(testDB);
  596. Table t = db.getTable("Table1");
  597. List<String> foundDates = new ArrayList<String>();
  598. for(Row row : t) {
  599. foundDates.add(sdf.format(row.getDate("DateField")));
  600. }
  601. assertEquals(dates, foundDates);
  602. db.close();
  603. }
  604. }
  605. public void testSystemTable() throws Exception
  606. {
  607. for (final FileFormat fileFormat : SUPPORTED_FILEFORMATS) {
  608. Database db = create(fileFormat);
  609. Set<String> sysTables = new TreeSet<String>(
  610. String.CASE_INSENSITIVE_ORDER);
  611. sysTables.addAll(
  612. Arrays.asList("MSysObjects", "MSysQueries", "MSysACES",
  613. "MSysRelationships"));
  614. if (fileFormat == FileFormat.GENERIC_JET4) {
  615. assertNull("file format: " + fileFormat, db.getSystemTable("MSysAccessObjects"));
  616. } else if (fileFormat.ordinal() < FileFormat.V2003.ordinal()) {
  617. assertNotNull("file format: " + fileFormat, db.getSystemTable("MSysAccessObjects"));
  618. sysTables.add("MSysAccessObjects");
  619. } else {
  620. // v2003+ template files have no "MSysAccessObjects" table
  621. assertNull("file format: " + fileFormat, db.getSystemTable("MSysAccessObjects"));
  622. sysTables.addAll(
  623. Arrays.asList("MSysNavPaneGroupCategories",
  624. "MSysNavPaneGroups", "MSysNavPaneGroupToObjects",
  625. "MSysNavPaneObjectIDs", "MSysAccessStorage"));
  626. if(fileFormat.ordinal() >= FileFormat.V2007.ordinal()) {
  627. sysTables.addAll(
  628. Arrays.asList(
  629. "MSysComplexColumns", "MSysComplexType_Attachment",
  630. "MSysComplexType_Decimal", "MSysComplexType_GUID",
  631. "MSysComplexType_IEEEDouble", "MSysComplexType_IEEESingle",
  632. "MSysComplexType_Long", "MSysComplexType_Short",
  633. "MSysComplexType_Text", "MSysComplexType_UnsignedByte"));
  634. }
  635. if(fileFormat.ordinal() >= FileFormat.V2010.ordinal()) {
  636. sysTables.add("f_12D7448B56564D8AAE333BCC9B3718E5_Data");
  637. sysTables.add("MSysResources");
  638. }
  639. }
  640. assertEquals(sysTables, db.getSystemTableNames());
  641. assertNotNull(db.getSystemTable("MSysObjects"));
  642. assertNotNull(db.getSystemTable("MSysQueries"));
  643. assertNotNull(db.getSystemTable("MSysACES"));
  644. assertNotNull(db.getSystemTable("MSysRelationships"));
  645. assertNull(db.getSystemTable("MSysBogus"));
  646. TableMetaData tmd = db.getTableMetaData("MSysObjects");
  647. assertEquals("MSysObjects", tmd.getName());
  648. assertFalse(tmd.isLinked());
  649. assertTrue(tmd.isSystem());
  650. db.close();
  651. }
  652. }
  653. public void testFixedText() throws Exception
  654. {
  655. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.FIXED_TEXT)) {
  656. Database db = openCopy(testDB);
  657. Table t = db.getTable("users");
  658. Column c = t.getColumn("c_flag_");
  659. assertEquals(DataType.TEXT, c.getType());
  660. assertEquals(false, c.isVariableLength());
  661. assertEquals(2, c.getLength());
  662. Map<String,Object> row = t.getNextRow();
  663. assertEquals("N", row.get("c_flag_"));
  664. t.addRow(3, "testFixedText", "boo", "foo", "bob", 3, 5, 9, "Y",
  665. new Date());
  666. t.getNextRow();
  667. row = t.getNextRow();
  668. assertEquals("testFixedText", row.get("c_user_login"));
  669. assertEquals("Y", row.get("c_flag_"));
  670. db.close();
  671. }
  672. }
  673. public void testDbSortOrder() throws Exception {
  674. for (final TestDB testDB : SUPPORTED_DBS_TEST_FOR_READ) {
  675. Database db = open(testDB);
  676. assertEquals(((DatabaseImpl)db).getFormat().DEFAULT_SORT_ORDER,
  677. ((DatabaseImpl)db).getDefaultSortOrder());
  678. db.close();
  679. }
  680. }
  681. public void testUnsupportedColumns() throws Exception {
  682. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.UNSUPPORTED)) {
  683. Database db = open(testDB);
  684. Table t = db.getTable("Test");
  685. Column varCol = t.getColumn("UnknownVar");
  686. assertEquals(DataType.UNSUPPORTED_VARLEN, varCol.getType());
  687. Column fixCol = t.getColumn("UnknownFix");
  688. assertEquals(DataType.UNSUPPORTED_FIXEDLEN, fixCol.getType());
  689. List<String> varVals = Arrays.asList(
  690. "RawData[(10) FF FE 73 6F 6D 65 64 61 74 61]",
  691. "RawData[(12) FF FE 6F 74 68 65 72 20 64 61 74 61]",
  692. null);
  693. List<String> fixVals = Arrays.asList("RawData[(4) 37 00 00 00]",
  694. "RawData[(4) F3 FF FF FF]",
  695. "RawData[(4) 02 00 00 00]");
  696. int idx = 0;
  697. for(Map<String,Object> row : t) {
  698. checkRawValue(varVals.get(idx), varCol.getRowValue(row));
  699. checkRawValue(fixVals.get(idx), fixCol.getRowValue(row));
  700. ++idx;
  701. }
  702. db.close();
  703. }
  704. }
  705. public void testLinkedTables() throws Exception {
  706. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.LINKED)) {
  707. Database db = openCopy(testDB);
  708. try {
  709. db.getTable("Table2");
  710. fail("FileNotFoundException should have been thrown");
  711. } catch(FileNotFoundException e) {
  712. // success
  713. }
  714. TableMetaData tmd = db.getTableMetaData("Table2");
  715. assertEquals("Table2", tmd.getName());
  716. assertTrue(tmd.isLinked());
  717. assertFalse(tmd.isSystem());
  718. assertEquals("Table1", tmd.getLinkedTableName());
  719. assertEquals("Z:\\jackcess_test\\linkeeTest.accdb", tmd.getLinkedDbName());
  720. tmd = db.getTableMetaData("FooTable");
  721. assertNull(tmd);
  722. assertTrue(db.getLinkedDatabases().isEmpty());
  723. final String linkeeDbName = "Z:\\jackcess_test\\linkeeTest.accdb";
  724. final File linkeeFile = new File("src/test/data/linkeeTest.accdb");
  725. db.setLinkResolver(new LinkResolver() {
  726. public Database resolveLinkedDatabase(Database linkerdb, String dbName)
  727. throws IOException {
  728. assertEquals(linkeeDbName, dbName);
  729. return DatabaseBuilder.open(linkeeFile);
  730. }
  731. });
  732. Table t2 = db.getTable("Table2");
  733. assertEquals(1, db.getLinkedDatabases().size());
  734. Database linkeeDb = db.getLinkedDatabases().get(linkeeDbName);
  735. assertNotNull(linkeeDb);
  736. assertEquals(linkeeFile, linkeeDb.getFile());
  737. assertEquals("linkeeTest.accdb", ((DatabaseImpl)linkeeDb).getName());
  738. List<? extends Map<String, Object>> expectedRows =
  739. createExpectedTable(
  740. createExpectedRow(
  741. "ID", 1,
  742. "Field1", "bar"));
  743. assertTable(expectedRows, t2);
  744. db.createLinkedTable("FooTable", linkeeDbName, "Table2");
  745. tmd = db.getTableMetaData("FooTable");
  746. assertEquals("FooTable", tmd.getName());
  747. assertTrue(tmd.isLinked());
  748. assertFalse(tmd.isSystem());
  749. assertEquals("Table2", tmd.getLinkedTableName());
  750. assertEquals("Z:\\jackcess_test\\linkeeTest.accdb", tmd.getLinkedDbName());
  751. Table t3 = db.getTable("FooTable");
  752. assertEquals(1, db.getLinkedDatabases().size());
  753. expectedRows =
  754. createExpectedTable(
  755. createExpectedRow(
  756. "ID", 1,
  757. "Field1", "buzz"));
  758. assertTable(expectedRows, t3);
  759. tmd = db.getTableMetaData("Table1");
  760. assertEquals("Table1", tmd.getName());
  761. assertFalse(tmd.isLinked());
  762. assertFalse(tmd.isSystem());
  763. assertNull(tmd.getLinkedTableName());
  764. assertNull(tmd.getLinkedDbName());
  765. Table t1 = tmd.open(db);
  766. assertFalse(db.isLinkedTable(null));
  767. assertTrue(db.isLinkedTable(t2));
  768. assertTrue(db.isLinkedTable(t3));
  769. assertFalse(db.isLinkedTable(t1));
  770. List<Table> tables = getTables(db.newIterable());
  771. assertEquals(3, tables.size());
  772. assertTrue(tables.contains(t1));
  773. assertTrue(tables.contains(t2));
  774. assertTrue(tables.contains(t3));
  775. assertFalse(tables.contains(((DatabaseImpl)db).getSystemCatalog()));
  776. tables = getTables(db.newIterable().setIncludeNormalTables(false));
  777. assertEquals(2, tables.size());
  778. assertFalse(tables.contains(t1));
  779. assertTrue(tables.contains(t2));
  780. assertTrue(tables.contains(t3));
  781. assertFalse(tables.contains(((DatabaseImpl)db).getSystemCatalog()));
  782. tables = getTables(db.newIterable().withLocalUserTablesOnly());
  783. assertEquals(1, tables.size());
  784. assertTrue(tables.contains(t1));
  785. assertFalse(tables.contains(t2));
  786. assertFalse(tables.contains(t3));
  787. assertFalse(tables.contains(((DatabaseImpl)db).getSystemCatalog()));
  788. tables = getTables(db.newIterable().withSystemTablesOnly());
  789. assertTrue(tables.size() > 5);
  790. assertFalse(tables.contains(t1));
  791. assertFalse(tables.contains(t2));
  792. assertFalse(tables.contains(t3));
  793. assertTrue(tables.contains(((DatabaseImpl)db).getSystemCatalog()));
  794. db.close();
  795. }
  796. }
  797. private static List<Table> getTables(Iterable<Table> tableIter)
  798. {
  799. List<Table> tableList = new ArrayList<Table>();
  800. for(Table t : tableIter) {
  801. tableList.add(t);
  802. }
  803. return tableList;
  804. }
  805. public void testTimeZone() throws Exception
  806. {
  807. TimeZone tz = TimeZone.getTimeZone("America/New_York");
  808. doTestTimeZone(tz);
  809. tz = TimeZone.getTimeZone("Australia/Sydney");
  810. doTestTimeZone(tz);
  811. }
  812. private static void doTestTimeZone(final TimeZone tz) throws Exception
  813. {
  814. ColumnImpl col = new ColumnImpl(null, null, DataType.SHORT_DATE_TIME, 0, 0, 0) {
  815. @Override
  816. public TimeZone getTimeZone() { return tz; }
  817. @Override
  818. public ZoneId getZoneId() { return null; }
  819. @Override
  820. public ColumnImpl.DateTimeFactory getDateTimeFactory() {
  821. return getDateTimeFactory(DateTimeType.DATE);
  822. }
  823. };
  824. SimpleDateFormat df = new SimpleDateFormat("yyyy.MM.dd");
  825. df.setTimeZone(tz);
  826. long startDate = df.parse("2012.01.01").getTime();
  827. long endDate = df.parse("2013.01.01").getTime();
  828. Calendar curCal = Calendar.getInstance(tz);
  829. curCal.setTimeInMillis(startDate);
  830. SimpleDateFormat sdf = new SimpleDateFormat("yyyy.MM.dd HH:mm:ss");
  831. sdf.setTimeZone(tz);
  832. while(curCal.getTimeInMillis() < endDate) {
  833. Date curDate = curCal.getTime();
  834. Date newDate = new Date(col.fromDateDouble(col.toDateDouble(curDate)));
  835. if(curDate.getTime() != newDate.getTime()) {
  836. assertEquals(sdf.format(curDate), sdf.format(newDate));
  837. }
  838. curCal.add(Calendar.MINUTE, 30);
  839. }
  840. }
  841. public void testToString()
  842. {
  843. RowImpl row = new RowImpl(new RowIdImpl(1, 1));
  844. row.put("id", 37);
  845. row.put("data", null);
  846. assertEquals("Row[1:1][{id=37,data=<null>}]", row.toString());
  847. }
  848. public void testIterateTableNames() throws Exception {
  849. for (final TestDB testDB : SUPPORTED_DBS_TEST_FOR_READ) {
  850. final Database db = open(testDB);
  851. Set<String> names = new HashSet<>();
  852. int sysCount = 0;
  853. for(TableMetaData tmd : db.newTableMetaDataIterable()) {
  854. if(tmd.isSystem()) {
  855. ++sysCount;
  856. continue;
  857. }
  858. assertFalse(tmd.isLinked());
  859. assertNull(tmd.getLinkedTableName());
  860. assertNull(tmd.getLinkedDbName());
  861. names.add(tmd.getName());
  862. }
  863. assertTrue(sysCount > 4);
  864. assertEquals(new HashSet<>(Arrays.asList("Table1", "Table2", "Table3",
  865. "Table4")),
  866. names);
  867. }
  868. for (final TestDB testDB : TestDB.getSupportedForBasename(Basename.LINKED)) {
  869. final Database db = open(testDB);
  870. Set<String> names = new HashSet<>();
  871. for(TableMetaData tmd : db.newTableMetaDataIterable()) {
  872. if(tmd.isSystem()) {
  873. continue;
  874. }
  875. if("Table1".equals(tmd.getName())) {
  876. assertFalse(tmd.isLinked());
  877. assertNull(tmd.getLinkedTableName());
  878. assertNull(tmd.getLinkedDbName());
  879. } else {
  880. assertTrue(tmd.isLinked());
  881. assertEquals("Table1", tmd.getLinkedTableName());
  882. assertEquals("Z:\\jackcess_test\\linkeeTest.accdb", tmd.getLinkedDbName());
  883. }
  884. names.add(tmd.getName());
  885. }
  886. assertEquals(new HashSet<>(Arrays.asList("Table1", "Table2")),
  887. names);
  888. }
  889. }
  890. private static void checkRawValue(String expected, Object val)
  891. {
  892. if(expected != null) {
  893. assertTrue(ColumnImpl.isRawData(val));
  894. assertEquals(expected, val.toString());
  895. } else {
  896. assertNull(val);
  897. }
  898. }
  899. }