summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--src/java/com/healthmarketscience/jackcess/ByteUtil.java26
-rw-r--r--src/java/com/healthmarketscience/jackcess/Column.java70
-rw-r--r--src/java/com/healthmarketscience/jackcess/Cursor.java9
-rw-r--r--src/java/com/healthmarketscience/jackcess/Index.java10
-rw-r--r--src/java/com/healthmarketscience/jackcess/Table.java320
-rw-r--r--test/src/java/com/healthmarketscience/jackcess/DatabaseTest.java61
-rw-r--r--test/src/java/com/healthmarketscience/jackcess/TableTest.java2
7 files changed, 413 insertions, 85 deletions
diff --git a/src/java/com/healthmarketscience/jackcess/ByteUtil.java b/src/java/com/healthmarketscience/jackcess/ByteUtil.java
index 95c2d8e..199926f 100644
--- a/src/java/com/healthmarketscience/jackcess/ByteUtil.java
+++ b/src/java/com/healthmarketscience/jackcess/ByteUtil.java
@@ -253,6 +253,18 @@ public final class ByteUtil {
}
/**
+ * Sets all bits in the given remaining byte range to 0.
+ */
+ public static void clearRemaining(ByteBuffer buffer)
+ {
+ if(!buffer.hasRemaining()) {
+ return;
+ }
+ int pos = buffer.position();
+ clearRange(buffer, pos, pos + buffer.remaining());
+ }
+
+ /**
* Sets all bits in the given byte range to 0.
*/
public static void clearRange(ByteBuffer buffer, int start,
@@ -445,5 +457,17 @@ public final class ByteUtil {
bytes[offset + 0] = bytes[offset + 1];
bytes[offset + 1] = b;
}
-
+
+ /**
+ * Moves the position of the given buffer the given count from the current
+ * position.
+ * @return the new buffer position
+ */
+ public static int forward(ByteBuffer buffer, int count)
+ {
+ int newPos = buffer.position() + count;
+ buffer.position(newPos);
+ return newPos;
+ }
+
}
diff --git a/src/java/com/healthmarketscience/jackcess/Column.java b/src/java/com/healthmarketscience/jackcess/Column.java
index 9f689b3..a41b450 100644
--- a/src/java/com/healthmarketscience/jackcess/Column.java
+++ b/src/java/com/healthmarketscience/jackcess/Column.java
@@ -65,6 +65,12 @@ public class Column implements Comparable<Column> {
public static final Object AUTO_NUMBER = "<AUTO_NUMBER>";
/**
+ * Meaningless placeholder object for updating rows which indicates that a
+ * given column should keep its existing value.
+ */
+ public static final Object KEEP_VALUE = "<KEEP_VALUE>";
+
+ /**
* Access stores numeric dates in days. Java stores them in milliseconds.
*/
private static final double MILLISECONDS_PER_DAY =
@@ -351,6 +357,10 @@ public class Column implements Comparable<Column> {
}
}
+ /**
+ * Returns the AutoNumberGenerator for this column if this is an autonumber
+ * column, {@code null} otherwise.
+ */
public AutoNumberGenerator getAutoNumberGenerator() {
return _autoNumberGenerator;
}
@@ -872,7 +882,7 @@ public class Column implements Comparable<Column> {
lvalPage = getLongValuePage(value.length, lvalBufferH);
firstLvalPageNum = lvalBufferH.getPageNumber();
firstLvalRow = (byte)Table.addDataPageRow(lvalPage, value.length,
- getFormat());
+ getFormat(), 0);
lvalPage.put(value);
getPageChannel().writePage(lvalPage, firstLvalPageNum);
break;
@@ -910,7 +920,7 @@ public class Column implements Comparable<Column> {
// add row to this page
byte lvalRow = (byte)Table.addDataPageRow(lvalPage, chunkLength + 4,
- getFormat());
+ getFormat(), 0);
// write next page info (we'll always be writing into row 0 for
// newly created pages)
@@ -1015,6 +1025,11 @@ public class Column implements Comparable<Column> {
public ByteBuffer write(Object obj, int remainingRowLength, ByteOrder order)
throws IOException
{
+ if(isRawData(obj)) {
+ // just slap it right in (not for the faint of heart!)
+ return ByteBuffer.wrap(((RawData)obj).getBytes());
+ }
+
if(!isVariableLength()) {
return writeFixedLengthField(obj, order);
}
@@ -1434,6 +1449,23 @@ public class Column implements Comparable<Column> {
}
/**
+ * Returns a wrapper for raw column data that can be written without
+ * understanding the data. Useful for wrapping unparseable data for
+ * re-writing.
+ */
+ static RawData rawDataWrapper(byte[] bytes) {
+ return new RawData(bytes);
+ }
+
+ /**
+ * Returs {@code true} if the given value is "raw" column data,
+ * {@code false} otherwise.
+ */
+ static boolean isRawData(Object value) {
+ return(value instanceof RawData);
+ }
+
+ /**
* Date subclass which stashes the original date bits, in case we attempt to
* re-write the value (will not lose precision).
*/
@@ -1461,16 +1493,50 @@ public class Column implements Comparable<Column> {
}
/**
+ * Wrapper for raw column data which can be re-written.
+ */
+ private static class RawData
+ {
+ private final byte[] _bytes;
+
+ private RawData(byte[] bytes) {
+ _bytes = bytes;
+ }
+
+ private byte[] getBytes() {
+ return _bytes;
+ }
+
+ @Override
+ public String toString() {
+ return "RawData: " + ByteUtil.toHexString(_bytes);
+ }
+ }
+
+ /**
* Base class for the supported autonumber types.
*/
public abstract class AutoNumberGenerator
{
protected AutoNumberGenerator() {}
+ /**
+ * Returns the last autonumber generated by this generator. Only valid
+ * after a call to {@link Table#addRow}, otherwise undefined.
+ */
public abstract Object getLast();
+ /**
+ * Returns the next autonumber for this generator.
+ * <p>
+ * <i>Warning, calling this externally will result in this value being
+ * "lost" for the table.</i>
+ */
public abstract Object getNext();
+ /**
+ * Returns the flags used when writing this column.
+ */
public abstract int getColumnFlags();
}
diff --git a/src/java/com/healthmarketscience/jackcess/Cursor.java b/src/java/com/healthmarketscience/jackcess/Cursor.java
index 5804eea..8e7f61f 100644
--- a/src/java/com/healthmarketscience/jackcess/Cursor.java
+++ b/src/java/com/healthmarketscience/jackcess/Cursor.java
@@ -511,6 +511,15 @@ public abstract class Cursor implements Iterable<Map<String, Object>>
}
/**
+ * Update the current row.
+ * @throws IllegalStateException if the current row is not valid (at
+ * beginning or end of table), or deleted.
+ */
+ public void updateCurrentRow(Object... row) throws IOException {
+ _table.updateRow(_rowState, _curPos.getRowId(), row);
+ }
+
+ /**
* Moves to the next row in the table and returns it.
* @return The next row in this table (Column name -> Column value), or
* {@code null} if no next row is found
diff --git a/src/java/com/healthmarketscience/jackcess/Index.java b/src/java/com/healthmarketscience/jackcess/Index.java
index 97315d6..5300fe6 100644
--- a/src/java/com/healthmarketscience/jackcess/Index.java
+++ b/src/java/com/healthmarketscience/jackcess/Index.java
@@ -357,7 +357,7 @@ public abstract class Index implements Comparable<Index> {
_rootPageNumber = tableBuffer.getInt();
tableBuffer.getInt(); //Forward past Unknown
_indexFlags = tableBuffer.get();
- tableBuffer.position(tableBuffer.position() + 5); //Forward past other stuff
+ ByteUtil.forward(tableBuffer, 5); //Forward past other stuff
}
/**
@@ -942,11 +942,13 @@ public abstract class Index implements Comparable<Index> {
ByteArrayOutputStream bout = new ByteArrayOutputStream();
- // annoyingly, the values array could come from different sources, one
- // of which will make it a different size than the other. we need to
- // handle both situations.
for(ColumnDescriptor col : _columns) {
Object value = values[col.getColumnIndex()];
+ if(Column.isRawData(value)) {
+ // ignore it, we could not parse it
+ continue;
+ }
+
col.writeValue(value, bout);
}
diff --git a/src/java/com/healthmarketscience/jackcess/Table.java b/src/java/com/healthmarketscience/jackcess/Table.java
index 585a619..efc9a1a 100644
--- a/src/java/com/healthmarketscience/jackcess/Table.java
+++ b/src/java/com/healthmarketscience/jackcess/Table.java
@@ -403,16 +403,8 @@ public class Table
ByteBuffer rowBuffer = positionAtRowData(rowState, rowId);
requireNonDeletedRow(rowState, rowId);
- Object value = getRowColumn(rowBuffer, getRowNullMask(rowBuffer), column,
- rowState);
-
- // cache the row values in order to be able to update the index on row
- // deletion. note, most of the returned values are immutable, except
- // for binary data (returned as byte[]), but binary data shouldn't be
- // indexed anyway.
- rowState.setRowValue(column.getColumnIndex(), value);
-
- return value;
+ return getRowColumn(rowBuffer, getRowNullMask(rowBuffer), column,
+ rowState);
}
/**
@@ -451,14 +443,8 @@ public class Table
if((columnNames == null) || (columnNames.contains(column.getName()))) {
// Add the value to the row data
- Object value = getRowColumn(rowBuffer, nullMask, column, rowState);
- rtn.put(column.getName(), value);
-
- // cache the row values in order to be able to update the index on row
- // deletion. note, most of the returned values are immutable, except
- // for binary data (returned as byte[]), but binary data shouldn't be
- // indexed anyway.
- rowState.setRowValue(column.getColumnIndex(), value);
+ rtn.put(column.getName(),
+ getRowColumn(rowBuffer, nullMask, column, rowState));
}
}
return rtn;
@@ -466,6 +452,7 @@ public class Table
/**
* Reads the column data from the given row buffer. Leaves limit unchanged.
+ * Caches the returned value in the rowState.
*/
private static Object getRowColumn(ByteBuffer rowBuffer,
NullMask nullMask,
@@ -478,9 +465,12 @@ public class Table
boolean isNull = nullMask.isNull(column);
if(column.getType() == DataType.BOOLEAN) {
- return Boolean.valueOf(!isNull); //Boolean values are stored in the null mask
+ // Boolean values are stored in the null mask. see note about
+ // caching below
+ return rowState.setRowValue(column.getColumnIndex(),
+ Boolean.valueOf(!isNull));
} else if(isNull) {
- // well, that's easy!
+ // well, that's easy! (no need to update cache w/ null)
return null;
}
@@ -516,10 +506,19 @@ public class Table
rowBuffer.position(colDataPos);
rowBuffer.get(columnData);
- // parse the column data
- return column.read(columnData);
+ // parse the column data. we cache the row values in order to be able
+ // to update the index on row deletion. note, most of the returned
+ // values are immutable, except for binary data (returned as byte[]),
+ // but binary data shouldn't be indexed anyway.
+ return rowState.setRowValue(column.getColumnIndex(),
+ column.read(columnData));
} catch(Exception e) {
+
+ // cache "raw" row value. see note about caching above
+ rowState.setRowValue(column.getColumnIndex(),
+ Column.rawDataWrapper(columnData));
+
return rowState.handleRowError(column, columnData, e);
}
}
@@ -755,7 +754,7 @@ public class Table
int curTdefPageNumber = nextTdefPageNumber;
int writeLen = Math.min(partialTdef.remaining(), buffer.remaining());
partialTdef.put(buffer.array(), buffer.position(), writeLen);
- buffer.position(buffer.position() + writeLen);
+ ByteUtil.forward(buffer, writeLen);
if(buffer.hasRemaining()) {
// need a next page
@@ -1046,9 +1045,9 @@ public class Table
tableBuffer.getInt(); //Forward past Unknown
tableBuffer.getInt(); //Forward past alternate index number
int indexNumber = tableBuffer.getInt();
- tableBuffer.position(tableBuffer.position() + 11);
+ ByteUtil.forward(tableBuffer, 11);
byte indexType = tableBuffer.get();
- tableBuffer.position(tableBuffer.position() + 4);
+ ByteUtil.forward(tableBuffer, 4);
if(i < firstRealIdx) {
// ignore this info
@@ -1134,7 +1133,7 @@ public class Table
*/
private void skipName(ByteBuffer buffer) {
int nameLength = ByteUtil.getUnsignedShort(buffer);
- buffer.position(buffer.position() + nameLength);
+ ByteUtil.forward(buffer, nameLength);
}
/**
@@ -1193,6 +1192,10 @@ public class Table
TempBufferHolder writeRowBufferH)
throws IOException
{
+ if(inRows.isEmpty()) {
+ return;
+ }
+
// copy the input rows to a modifiable list so we can update the elements
List<Object[]> rows = new ArrayList<Object[]>(inRows);
ByteBuffer[] rowData = new ByteBuffer[rows.size()];
@@ -1211,64 +1214,36 @@ public class Table
// write the row of data to a temporary buffer
rowData[i] = createRow(row, getFormat().MAX_ROW_SIZE,
- writeRowBufferH.getPageBuffer(getPageChannel()));
+ writeRowBufferH.getPageBuffer(getPageChannel()),
+ false, 0);
if (rowData[i].limit() > getFormat().MAX_ROW_SIZE) {
throw new IOException("Row size " + rowData[i].limit() +
" is too large");
}
}
-
+
ByteBuffer dataPage = null;
int pageNumber = PageChannel.INVALID_PAGE_NUMBER;
-
- // find last data page (Not bothering to check other pages for free
- // space.)
- UsageMap.PageCursor revPageCursor = _ownedPages.cursor();
- revPageCursor.afterLast();
- while(true) {
- int tmpPageNumber = revPageCursor.getPreviousPage();
- if(tmpPageNumber < 0) {
- break;
- }
- dataPage = _addRowBufferH.setPage(getPageChannel(), tmpPageNumber);
- if(dataPage.get() == PageTypes.DATA) {
- // found last data page, only use if actually listed in free space
- // pages
- if(_freeSpacePages.containsPageNumber(tmpPageNumber)) {
- pageNumber = tmpPageNumber;
- }
- break;
- }
- }
-
- if(pageNumber == PageChannel.INVALID_PAGE_NUMBER) {
- // No data pages exist (with free space). Create a new one.
- dataPage = newDataPage();
- pageNumber = _addRowBufferH.getPageNumber();
- }
for (int i = 0; i < rowData.length; i++) {
int rowSize = rowData[i].remaining();
- if(!rowFitsOnDataPage(rowSize, dataPage, getFormat())) {
- // Last data page is full. Create a new one.
- writeDataPage(dataPage, pageNumber);
- _freeSpacePages.removePageNumber(pageNumber);
-
- dataPage = newDataPage();
- pageNumber = _addRowBufferH.getPageNumber();
- }
+ // get page with space
+ dataPage = findFreeRowSpace(rowSize, dataPage, pageNumber);
+ pageNumber = _addRowBufferH.getPageNumber();
// write out the row data
- int rowNum = addDataPageRow(dataPage, rowSize, getFormat());
+ int rowNum = addDataPageRow(dataPage, rowSize, getFormat(), 0);
dataPage.put(rowData[i]);
// update the indexes
+ RowId rowId = new RowId(pageNumber, rowNum);
for(Index index : _indexes) {
- index.addRow(rows.get(i), new RowId(pageNumber, rowNum));
+ index.addRow(rows.get(i), rowId);
}
}
+
writeDataPage(dataPage, pageNumber);
// Update tdef page
@@ -1276,6 +1251,173 @@ public class Table
}
/**
+ * Updates the current row to the new values.
+ * <p>
+ * Note, if this table has an auto-number column(s), the existing value(s)
+ * will be maintained, unchanged.
+ *
+ * @param row new row values for the current row.
+ */
+ public void updateCurrentRow(Object... row) throws IOException {
+ _cursor.updateCurrentRow(row);
+ }
+
+ /**
+ * Update the row on which the given rowState is currently positioned.
+ */
+ public void updateRow(RowState rowState, RowId rowId, Object... row)
+ throws IOException
+ {
+ requireValidRowId(rowId);
+
+ // ensure that the relevant row state is up-to-date
+ ByteBuffer rowBuffer = positionAtRowData(rowState, rowId);
+ int oldRowSize = rowBuffer.remaining();
+
+ requireNonDeletedRow(rowState, rowId);
+
+ // we need to make sure the row is the right length (fill with null).
+ if(row.length < _columns.size()) {
+ row = dupeRow(row, _columns.size());
+ }
+
+ // fill in any auto-numbers (we don't allow autonumber values to be
+ // modified) or "keep value" fields
+ NullMask nullMask = getRowNullMask(rowBuffer);
+ for(Column column : _columns) {
+ if(column.isAutoNumber() ||
+ (row[column.getColumnIndex()] == Column.KEEP_VALUE)) {
+ row[column.getColumnIndex()] = getRowColumn(rowBuffer, nullMask,
+ column, rowState);
+ }
+ }
+
+ // generate new row bytes
+ ByteBuffer newRowData = createRow(
+ row, getFormat().MAX_ROW_SIZE,
+ _singleRowBufferH.getPageBuffer(getPageChannel()), true, oldRowSize);
+
+ if (newRowData.limit() > getFormat().MAX_ROW_SIZE) {
+ throw new IOException("Row size " + newRowData.limit() +
+ " is too large");
+ }
+
+ Object[] oldRowValues = (!_indexes.isEmpty() ?
+ rowState.getRowValues() : null);
+
+ // delete old values from indexes
+ for(Index index : _indexes) {
+ index.deleteRow(oldRowValues, rowId);
+ }
+
+ // see if we can squeeze the new row data into the existing row
+ rowBuffer.reset();
+ int rowSize = newRowData.remaining();
+
+ ByteBuffer dataPage = null;
+ int pageNumber = PageChannel.INVALID_PAGE_NUMBER;
+
+ if(oldRowSize >= rowSize) {
+
+ // awesome, slap it in!
+ rowBuffer.put(newRowData);
+
+ // grab the page we just updated
+ dataPage = rowState.getFinalPage();
+ pageNumber = rowState.getFinalRowId().getPageNumber();
+
+ } else {
+
+ // bummer, need to find a new page for the data
+ dataPage = findFreeRowSpace(rowSize, null,
+ PageChannel.INVALID_PAGE_NUMBER);
+ pageNumber = _addRowBufferH.getPageNumber();
+
+ ByteBuffer oldDataPage = rowState.getFinalPage();
+ int oldPageNumber = rowState.getFinalRowId().getPageNumber();
+ if(pageNumber == oldPageNumber) {
+ // new row is on the same page as current row, share page
+ dataPage = oldDataPage;
+ }
+
+ // write out the new row data (set the deleted flag on the new data row)
+ int rowNum = addDataPageRow(dataPage, rowSize, getFormat(),
+ DELETED_ROW_MASK);
+ dataPage.put(newRowData);
+
+ // write the overflow info into the old row and clear out the remaining
+ // old data
+ rowBuffer.put((byte)rowNum);
+ ByteUtil.put3ByteInt(rowBuffer, pageNumber);
+ ByteUtil.clearRemaining(rowBuffer);
+
+ // set the overflow flag on the old row
+ int oldRowNumber = rowState.getFinalRowId().getRowNumber();
+ int oldRowIndex = getRowStartOffset(oldRowNumber, getFormat());
+ oldDataPage.putShort(oldRowIndex,
+ (short)(oldDataPage.getShort(oldRowIndex)
+ | OVERFLOW_ROW_MASK));
+ if(pageNumber != oldPageNumber) {
+ writeDataPage(oldDataPage, oldPageNumber);
+ }
+ }
+
+ // update the indexes
+ for(Index index : _indexes) {
+ index.addRow(row, rowId);
+ }
+
+ writeDataPage(dataPage, pageNumber);
+
+ updateTableDefinition(0);
+ }
+
+ private ByteBuffer findFreeRowSpace(int rowSize, ByteBuffer dataPage,
+ int pageNumber)
+ throws IOException
+ {
+ if(dataPage == null) {
+
+ // find last data page (Not bothering to check other pages for free
+ // space.)
+ UsageMap.PageCursor revPageCursor = _ownedPages.cursor();
+ revPageCursor.afterLast();
+ while(true) {
+ int tmpPageNumber = revPageCursor.getPreviousPage();
+ if(tmpPageNumber < 0) {
+ break;
+ }
+ dataPage = _addRowBufferH.setPage(getPageChannel(), tmpPageNumber);
+ if(dataPage.get() == PageTypes.DATA) {
+ // found last data page, only use if actually listed in free space
+ // pages
+ if(_freeSpacePages.containsPageNumber(tmpPageNumber)) {
+ pageNumber = tmpPageNumber;
+ }
+ break;
+ }
+ }
+
+ if(pageNumber == PageChannel.INVALID_PAGE_NUMBER) {
+ // No data pages exist (with free space). Create a new one.
+ return newDataPage();
+ }
+
+ }
+
+ if(!rowFitsOnDataPage(rowSize, dataPage, getFormat())) {
+
+ // Last data page is full. Create a new one.
+ writeDataPage(dataPage, pageNumber);
+ _freeSpacePages.removePageNumber(pageNumber);
+
+ dataPage = newDataPage();
+ }
+
+ return dataPage;
+ }
+
+ /**
* Updates the table definition after rows are modified.
*/
private void updateTableDefinition(int rowCountInc) throws IOException
@@ -1290,9 +1432,7 @@ public class Table
tdefPage.putInt(getFormat().OFFSET_NEXT_AUTO_NUMBER, _lastLongAutoNumber);
// write any index changes
- Iterator<Index> indIter = _indexes.iterator();
- for (int i = 0; i < _indexes.size(); i++) {
- Index index = indIter.next();
+ for (Index index : _indexes) {
// write the unique entry count for the index to the table definition
// page
tdefPage.putInt(index.getUniqueEntryCountOffset(),
@@ -1338,7 +1478,8 @@ public class Table
* @param buffer buffer to which to write the row data
* @return the given buffer, filled with the row data
*/
- ByteBuffer createRow(Object[] rowArray, int maxRowSize, ByteBuffer buffer)
+ ByteBuffer createRow(Object[] rowArray, int maxRowSize, ByteBuffer buffer,
+ boolean isUpdate, int minRowSize)
throws IOException
{
buffer.putShort(_maxColumnCount);
@@ -1362,7 +1503,7 @@ public class Table
} else {
- if(col.isAutoNumber()) {
+ if(col.isAutoNumber() && !isUpdate) {
// ignore given row value, use next autonumber
rowValue = col.getAutoNumberGenerator().getNext();
@@ -1400,7 +1541,8 @@ public class Table
// account for already written space
maxRowSize -= buffer.position();
// now, account for trailer space
- maxRowSize -= (nullMask.byteSize() + 4 + (_maxVarColumnCount * 2));
+ int trailerSize = (nullMask.byteSize() + 4 + (_maxVarColumnCount * 2));
+ maxRowSize -= trailerSize;
// for each non-null long value column we need to reserve a small
// amount of space so that we don't end up running out of row space
@@ -1443,13 +1585,25 @@ public class Table
varColumnOffsets[varColumnOffsetsIndex++] = (short) buffer.position();
}
- buffer.putShort((short) buffer.position()); //EOD marker
+ // record where we stopped writing
+ int eod = buffer.position();
+
+ // insert padding if necessary
+ padRowBuffer(buffer, minRowSize, trailerSize);
+
+ buffer.putShort((short) eod); //EOD marker
+
//Now write out variable length offsets
//Offsets are stored in reverse order
for (int i = _maxVarColumnCount - 1; i >= 0; i--) {
buffer.putShort(varColumnOffsets[i]);
}
buffer.putShort(_maxVarColumnCount); //Number of var length columns
+
+ } else {
+
+ // insert padding for row w/ no var cols
+ padRowBuffer(buffer, minRowSize, nullMask.byteSize());
}
nullMask.write(buffer); //Null mask
@@ -1460,6 +1614,17 @@ public class Table
return buffer;
}
+ private void padRowBuffer(ByteBuffer buffer, int minRowSize, int trailerSize)
+ {
+ int pos = buffer.position();
+ if((pos + trailerSize) < minRowSize) {
+ // pad the row to get to the min byte size
+ int padSize = minRowSize - (pos + trailerSize);
+ ByteUtil.clearRange(buffer, pos, pos + padSize);
+ ByteUtil.forward(buffer, padSize);
+ }
+ }
+
public int getRowCount() {
return _rowCount;
}
@@ -1549,7 +1714,8 @@ public class Table
*/
public static int addDataPageRow(ByteBuffer dataPage,
int rowSize,
- JetFormat format)
+ JetFormat format,
+ int rowFlags)
{
int rowSpaceUsage = getRowSpaceUsage(rowSize, format);
@@ -1568,7 +1734,8 @@ public class Table
rowLocation -= rowSize;
// write row position
- dataPage.putShort(getRowStartOffset(rowCount, format), rowLocation);
+ dataPage.putShort(getRowStartOffset(rowCount, format),
+ (short)(rowLocation | rowFlags));
// set position for row data
dataPage.position(rowLocation);
@@ -1692,7 +1859,7 @@ public class Table
System.arraycopy(row, 0, copy, 0, row.length);
return copy;
}
-
+
/** various statuses for the row data */
private enum RowStatus {
INIT, INVALID_PAGE, INVALID_ROW, VALID, DELETED, NORMAL, OVERFLOW;
@@ -1702,7 +1869,7 @@ public class Table
private enum RowStateStatus {
INIT, AT_HEADER, AT_FINAL;
}
-
+
/**
* Maintains the state of reading a row of data.
*/
@@ -1835,9 +2002,10 @@ public class Table
return(_status.ordinal() >= RowStateStatus.AT_FINAL.ordinal());
}
- private void setRowValue(int idx, Object value) {
+ private Object setRowValue(int idx, Object value) {
_haveRowValues = true;
_rowValues[idx] = value;
+ return value;
}
public Object[] getRowValues() {
diff --git a/test/src/java/com/healthmarketscience/jackcess/DatabaseTest.java b/test/src/java/com/healthmarketscience/jackcess/DatabaseTest.java
index 9055ed8..fac88f3 100644
--- a/test/src/java/com/healthmarketscience/jackcess/DatabaseTest.java
+++ b/test/src/java/com/healthmarketscience/jackcess/DatabaseTest.java
@@ -184,7 +184,7 @@ public class DatabaseTest extends TestCase {
public void testGetNextRow() throws Exception {
Database db = open();
- assertEquals(3, db.getTableNames().size());
+ assertEquals(4, db.getTableNames().size());
Table table = db.getTable("Table1");
Map<String, Object> row = table.getNextRow();
@@ -913,6 +913,65 @@ public class DatabaseTest extends TestCase {
db.close();
}
+
+ public void testUpdateRow() throws Exception
+ {
+ Database db = create();
+
+ Table t = new TableBuilder("test")
+ .addColumn(new ColumnBuilder("name", DataType.TEXT).toColumn())
+ .addColumn(new ColumnBuilder("id", DataType.LONG)
+ .setAutoNumber(true).toColumn())
+ .addColumn(new ColumnBuilder("data", DataType.TEXT)
+ .setLength(JetFormat.TEXT_FIELD_MAX_LENGTH).toColumn())
+ .toTable(db);
+
+ for(int i = 0; i < 10; ++i) {
+ t.addRow("row" + i, Column.AUTO_NUMBER, "initial data");
+ }
+
+ t.reset();
+ t.getNextRow();
+ Map<String,Object> row = t.getNextRow();
+
+ assertEquals(createExpectedRow("name", "row1",
+ "id", 2,
+ "data", "initial data"),
+ row);
+
+ t.updateCurrentRow(Column.KEEP_VALUE, Column.AUTO_NUMBER, "new data");
+
+ t.getNextRow();
+ t.getNextRow();
+ row = t.getNextRow();
+
+ assertEquals(createExpectedRow("name", "row4",
+ "id", 5,
+ "data", "initial data"),
+ row);
+
+ t.updateCurrentRow(Column.KEEP_VALUE, Column.AUTO_NUMBER, "a larger amount of new data");
+
+ t.reset();
+ t.getNextRow();
+ row = t.getNextRow();
+
+ assertEquals(createExpectedRow("name", "row1",
+ "id", 2,
+ "data", "new data"),
+ row);
+
+ t.getNextRow();
+ t.getNextRow();
+ row = t.getNextRow();
+
+ assertEquals(createExpectedRow("name", "row4",
+ "id", 5,
+ "data", "a larger amount of new data"),
+ row);
+
+ db.close();
+ }
static Object[] createTestRow(String col1Val) {
return new Object[] {col1Val, "R", "McCune", 1234, (byte) 0xad, 555.66d,
diff --git a/test/src/java/com/healthmarketscience/jackcess/TableTest.java b/test/src/java/com/healthmarketscience/jackcess/TableTest.java
index 3190846..54c819a 100644
--- a/test/src/java/com/healthmarketscience/jackcess/TableTest.java
+++ b/test/src/java/com/healthmarketscience/jackcess/TableTest.java
@@ -119,7 +119,7 @@ public class TableTest extends TestCase {
{
return _testTable.createRow(
row, _testTable.getFormat().MAX_ROW_SIZE,
- _testTable.getPageChannel().createPageBuffer());
+ _testTable.getPageChannel().createPageBuffer(), false, 0);
}
private ByteBuffer[] encodeColumns(Object... row)