Acknowledgments are now returned, received, and the message (Plaintext object) updated
-> no logic to resend messages yet
This commit is contained in:
		| @@ -31,8 +31,6 @@ import org.slf4j.LoggerFactory; | |||||||
|  |  | ||||||
| import java.net.InetAddress; | import java.net.InetAddress; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Timer; |  | ||||||
| import java.util.TimerTask; |  | ||||||
| import java.util.concurrent.CancellationException; | import java.util.concurrent.CancellationException; | ||||||
| import java.util.concurrent.ExecutionException; | import java.util.concurrent.ExecutionException; | ||||||
| import java.util.concurrent.Future; | import java.util.concurrent.Future; | ||||||
| @@ -71,16 +69,10 @@ public class BitmessageContext { | |||||||
|     private BitmessageContext(Builder builder) { |     private BitmessageContext(Builder builder) { | ||||||
|         ctx = new InternalContext(builder); |         ctx = new InternalContext(builder); | ||||||
|         labeler = builder.labeler; |         labeler = builder.labeler; | ||||||
|  |         ctx.getProofOfWorkService().doMissingProofOfWork(); | ||||||
|  |  | ||||||
|         networkListener = new DefaultMessageListener(ctx, labeler, builder.listener); |         networkListener = new DefaultMessageListener(ctx, labeler, builder.listener); | ||||||
|  |  | ||||||
|         sendPubkeyOnIdentityCreation = builder.sendPubkeyOnIdentityCreation; |         sendPubkeyOnIdentityCreation = builder.sendPubkeyOnIdentityCreation; | ||||||
|  |  | ||||||
|         new Timer().schedule(new TimerTask() { |  | ||||||
|             @Override |  | ||||||
|             public void run() { |  | ||||||
|                 ctx.getProofOfWorkService().doMissingProofOfWork(); |  | ||||||
|             } |  | ||||||
|         }, 30_000); // After 30 seconds |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public AddressRepository addresses() { |     public AddressRepository addresses() { | ||||||
|   | |||||||
| @@ -48,9 +48,12 @@ class DefaultMessageListener implements NetworkHandler.MessageListener { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|  |     @SuppressWarnings("ConstantConditions") | ||||||
|     public void receive(ObjectMessage object) throws IOException { |     public void receive(ObjectMessage object) throws IOException { | ||||||
|         ObjectPayload payload = object.getPayload(); |         ObjectPayload payload = object.getPayload(); | ||||||
|         if (payload.getType() == null) return; |         if (payload.getType() == null && payload instanceof GenericPayload) { | ||||||
|  |             receive((GenericPayload) payload); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         switch (payload.getType()) { |         switch (payload.getType()) { | ||||||
|             case GET_PUBKEY: { |             case GET_PUBKEY: { | ||||||
| @@ -125,7 +128,7 @@ class DefaultMessageListener implements NetworkHandler.MessageListener { | |||||||
|                 if (!object.isSignatureValid(plaintext.getFrom().getPubkey())) { |                 if (!object.isSignatureValid(plaintext.getFrom().getPubkey())) { | ||||||
|                     LOG.warn("Msg with IV " + object.getInventoryVector() + " was successfully decrypted, but signature check failed. Ignoring."); |                     LOG.warn("Msg with IV " + object.getInventoryVector() + " was successfully decrypted, but signature check failed. Ignoring."); | ||||||
|                 } else { |                 } else { | ||||||
|                     receive(object.getInventoryVector(), msg.getPlaintext()); |                     receive(object.getInventoryVector(), plaintext); | ||||||
|                 } |                 } | ||||||
|                 break; |                 break; | ||||||
|             } catch (DecryptionFailedException ignore) { |             } catch (DecryptionFailedException ignore) { | ||||||
| @@ -133,6 +136,16 @@ class DefaultMessageListener implements NetworkHandler.MessageListener { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     protected void receive(GenericPayload ack) { | ||||||
|  |         if (ack.getData().length == Msg.ACK_LENGTH) { | ||||||
|  |             Plaintext msg = ctx.getMessageRepository().getMessageForAck(ack.getData()); | ||||||
|  |             if (msg != null) { | ||||||
|  |                 ctx.getLabeler().markAsAcknowledged(msg); | ||||||
|  |                 ctx.getMessageRepository().save(msg); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     protected void receive(ObjectMessage object, Broadcast broadcast) throws IOException { |     protected void receive(ObjectMessage object, Broadcast broadcast) throws IOException { | ||||||
|         byte[] tag = broadcast instanceof V5Broadcast ? ((V5Broadcast) broadcast).getTag() : null; |         byte[] tag = broadcast instanceof V5Broadcast ? ((V5Broadcast) broadcast).getTag() : null; | ||||||
|         for (BitmessageAddress subscription : ctx.getAddressRepository().getSubscriptions(broadcast.getVersion())) { |         for (BitmessageAddress subscription : ctx.getAddressRepository().getSubscriptions(broadcast.getVersion())) { | ||||||
|   | |||||||
| @@ -12,6 +12,8 @@ import org.slf4j.Logger; | |||||||
| import org.slf4j.LoggerFactory; | import org.slf4j.LoggerFactory; | ||||||
|  |  | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Timer; | ||||||
|  | import java.util.TimerTask; | ||||||
|  |  | ||||||
| import static ch.dissem.bitmessage.InternalContext.NETWORK_EXTRA_BYTES; | import static ch.dissem.bitmessage.InternalContext.NETWORK_EXTRA_BYTES; | ||||||
| import static ch.dissem.bitmessage.InternalContext.NETWORK_NONCE_TRIALS_PER_BYTE; | import static ch.dissem.bitmessage.InternalContext.NETWORK_NONCE_TRIALS_PER_BYTE; | ||||||
| @@ -29,14 +31,21 @@ public class ProofOfWorkService implements ProofOfWorkEngine.Callback, InternalC | |||||||
|     private MessageRepository messageRepo; |     private MessageRepository messageRepo; | ||||||
|  |  | ||||||
|     public void doMissingProofOfWork() { |     public void doMissingProofOfWork() { | ||||||
|         List<byte[]> items = powRepo.getItems(); |         final List<byte[]> items = powRepo.getItems(); | ||||||
|         if (items.isEmpty()) return; |         if (items.isEmpty()) return; | ||||||
|  |  | ||||||
|         LOG.info("Doing POW for " + items.size() + " tasks."); |         // Wait for 30 seconds, to let the application start up before putting heavy load on the CPU | ||||||
|         for (byte[] initialHash : items) { |         new Timer().schedule(new TimerTask() { | ||||||
|             Item item = powRepo.getItem(initialHash); |             @Override | ||||||
|             cryptography.doProofOfWork(item.object, item.nonceTrialsPerByte, item.extraBytes, this); |             public void run() { | ||||||
|         } |                 LOG.info("Doing POW for " + items.size() + " tasks."); | ||||||
|  |                 for (byte[] initialHash : items) { | ||||||
|  |                     Item item = powRepo.getItem(initialHash); | ||||||
|  |                     cryptography.doProofOfWork(item.object, item.nonceTrialsPerByte, item.extraBytes, | ||||||
|  |                             ProofOfWorkService.this); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, 30_000); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void doProofOfWork(ObjectMessage object) { |     public void doProofOfWork(ObjectMessage object) { | ||||||
| @@ -79,7 +88,6 @@ public class ProofOfWorkService implements ProofOfWorkEngine.Callback, InternalC | |||||||
|                 messageRepo.save(plaintext); |                 messageRepo.save(plaintext); | ||||||
|             } |             } | ||||||
|             ctx.getInventory().storeObject(object); |             ctx.getInventory().storeObject(object); | ||||||
|             powRepo.removeObject(initialHash); |  | ||||||
|             ctx.getNetworkHandler().offer(object.getInventoryVector()); |             ctx.getNetworkHandler().offer(object.getInventoryVector()); | ||||||
|         } else { |         } else { | ||||||
|             item.message.getAckMessage().setNonce(nonce); |             item.message.getAckMessage().setNonce(nonce); | ||||||
| @@ -96,6 +104,7 @@ public class ProofOfWorkService implements ProofOfWorkEngine.Callback, InternalC | |||||||
|             } |             } | ||||||
|             doProofOfWork(item.message.getTo(), object); |             doProofOfWork(item.message.getTo(), object); | ||||||
|         } |         } | ||||||
|  |         powRepo.removeObject(initialHash); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|   | |||||||
| @@ -16,6 +16,7 @@ | |||||||
|  |  | ||||||
| package ch.dissem.bitmessage.entity; | package ch.dissem.bitmessage.entity; | ||||||
|  |  | ||||||
|  | import ch.dissem.bitmessage.entity.payload.Msg; | ||||||
| import ch.dissem.bitmessage.entity.payload.Pubkey; | import ch.dissem.bitmessage.entity.payload.Pubkey; | ||||||
| import ch.dissem.bitmessage.entity.valueobject.InventoryVector; | import ch.dissem.bitmessage.entity.valueobject.InventoryVector; | ||||||
| import ch.dissem.bitmessage.entity.valueobject.Label; | import ch.dissem.bitmessage.entity.valueobject.Label; | ||||||
| @@ -511,7 +512,7 @@ public class Plaintext implements Streamable { | |||||||
|                 to = new BitmessageAddress(0, 0, destinationRipe); |                 to = new BitmessageAddress(0, 0, destinationRipe); | ||||||
|             } |             } | ||||||
|             if (type == Type.MSG && ackMessage == null && ackData == null) { |             if (type == Type.MSG && ackMessage == null && ackData == null) { | ||||||
|                 ackData = cryptography().randomBytes(32); |                 ackData = cryptography().randomBytes(Msg.ACK_LENGTH); | ||||||
|             } |             } | ||||||
|             return new Plaintext(this); |             return new Plaintext(this); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -53,6 +53,10 @@ public class GenericPayload extends ObjectPayload { | |||||||
|         return stream; |         return stream; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public byte[] getData() { | ||||||
|  |         return data; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void write(OutputStream stream) throws IOException { |     public void write(OutputStream stream) throws IOException { | ||||||
|         stream.write(data); |         stream.write(data); | ||||||
|   | |||||||
| @@ -33,6 +33,7 @@ import static ch.dissem.bitmessage.entity.Plaintext.Type.MSG; | |||||||
|  */ |  */ | ||||||
| public class Msg extends ObjectPayload implements Encrypted, PlaintextHolder { | public class Msg extends ObjectPayload implements Encrypted, PlaintextHolder { | ||||||
|     private static final long serialVersionUID = 4327495048296365733L; |     private static final long serialVersionUID = 4327495048296365733L; | ||||||
|  |     public static final int ACK_LENGTH = 32; | ||||||
|  |  | ||||||
|     private long stream; |     private long stream; | ||||||
|     private CryptoBox encrypted; |     private CryptoBox encrypted; | ||||||
|   | |||||||
| @@ -23,6 +23,8 @@ import ch.dissem.bitmessage.entity.Plaintext; | |||||||
| import ch.dissem.bitmessage.entity.payload.*; | import ch.dissem.bitmessage.entity.payload.*; | ||||||
| import ch.dissem.bitmessage.entity.valueobject.PrivateKey; | import ch.dissem.bitmessage.entity.valueobject.PrivateKey; | ||||||
| import ch.dissem.bitmessage.exception.NodeException; | import ch.dissem.bitmessage.exception.NodeException; | ||||||
|  | import ch.dissem.bitmessage.utils.TTL; | ||||||
|  | import ch.dissem.bitmessage.utils.UnixTime; | ||||||
| import org.slf4j.Logger; | import org.slf4j.Logger; | ||||||
| import org.slf4j.LoggerFactory; | import org.slf4j.LoggerFactory; | ||||||
|  |  | ||||||
| @@ -210,6 +212,6 @@ public class Factory { | |||||||
|         if (plaintext == null || plaintext.getAckData() == null) |         if (plaintext == null || plaintext.getAckData() == null) | ||||||
|             return null; |             return null; | ||||||
|         GenericPayload ack = new GenericPayload(3, plaintext.getFrom().getStream(), plaintext.getAckData()); |         GenericPayload ack = new GenericPayload(3, plaintext.getFrom().getStream(), plaintext.getAckData()); | ||||||
|         return new ObjectMessage.Builder().objectType(MSG).payload(ack).build(); |         return new ObjectMessage.Builder().objectType(MSG).payload(ack).expiresTime(UnixTime.now(+TTL.msg())).build(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -34,6 +34,8 @@ public interface MessageRepository { | |||||||
|  |  | ||||||
|     Plaintext getMessage(byte[] initialHash); |     Plaintext getMessage(byte[] initialHash); | ||||||
|  |  | ||||||
|  |     Plaintext getMessageForAck(byte[] ackData); | ||||||
|  |  | ||||||
|     List<Plaintext> findMessages(Label label); |     List<Plaintext> findMessages(Label label); | ||||||
|  |  | ||||||
|     List<Plaintext> findMessages(Status status); |     List<Plaintext> findMessages(Status status); | ||||||
|   | |||||||
| @@ -5,11 +5,13 @@ import ch.dissem.bitmessage.entity.BitmessageAddress; | |||||||
| import ch.dissem.bitmessage.entity.Plaintext; | import ch.dissem.bitmessage.entity.Plaintext; | ||||||
| import ch.dissem.bitmessage.networking.DefaultNetworkHandler; | import ch.dissem.bitmessage.networking.DefaultNetworkHandler; | ||||||
| import ch.dissem.bitmessage.ports.DefaultLabeler; | import ch.dissem.bitmessage.ports.DefaultLabeler; | ||||||
|  | import ch.dissem.bitmessage.ports.Labeler; | ||||||
| import ch.dissem.bitmessage.repository.*; | import ch.dissem.bitmessage.repository.*; | ||||||
| import ch.dissem.bitmessage.utils.TTL; | import ch.dissem.bitmessage.utils.TTL; | ||||||
| import org.junit.After; | import org.junit.After; | ||||||
| import org.junit.Before; | import org.junit.Before; | ||||||
| import org.junit.Test; | import org.junit.Test; | ||||||
|  | import org.mockito.Mockito; | ||||||
| import org.slf4j.Logger; | import org.slf4j.Logger; | ||||||
| import org.slf4j.LoggerFactory; | import org.slf4j.LoggerFactory; | ||||||
|  |  | ||||||
| @@ -20,6 +22,7 @@ import static ch.dissem.bitmessage.entity.payload.Pubkey.Feature.DOES_ACK; | |||||||
| import static ch.dissem.bitmessage.utils.UnixTime.MINUTE; | import static ch.dissem.bitmessage.utils.UnixTime.MINUTE; | ||||||
| import static org.hamcrest.CoreMatchers.equalTo; | import static org.hamcrest.CoreMatchers.equalTo; | ||||||
| import static org.junit.Assert.assertThat; | import static org.junit.Assert.assertThat; | ||||||
|  | import static org.mockito.Matchers.any; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @author Christian Basler |  * @author Christian Basler | ||||||
| @@ -29,6 +32,7 @@ public class SystemTest { | |||||||
|  |  | ||||||
|     private BitmessageContext alice; |     private BitmessageContext alice; | ||||||
|     private TestListener aliceListener = new TestListener(); |     private TestListener aliceListener = new TestListener(); | ||||||
|  |     private Labeler aliceLabeler = Mockito.spy(new DebugLabeler("Alice")); | ||||||
|     private BitmessageAddress aliceIdentity; |     private BitmessageAddress aliceIdentity; | ||||||
|  |  | ||||||
|     private BitmessageContext bob; |     private BitmessageContext bob; | ||||||
| @@ -53,7 +57,7 @@ public class SystemTest { | |||||||
|                 .networkHandler(new DefaultNetworkHandler()) |                 .networkHandler(new DefaultNetworkHandler()) | ||||||
|                 .cryptography(new BouncyCryptography()) |                 .cryptography(new BouncyCryptography()) | ||||||
|                 .listener(aliceListener) |                 .listener(aliceListener) | ||||||
|                 .labeler(new DebugLabeler("Alice")) |                 .labeler(aliceLabeler) | ||||||
|                 .build(); |                 .build(); | ||||||
|         alice.startup(); |         alice.startup(); | ||||||
|         aliceIdentity = alice.createIdentity(false, DOES_ACK); |         aliceIdentity = alice.createIdentity(false, DOES_ACK); | ||||||
| @@ -93,6 +97,9 @@ public class SystemTest { | |||||||
|  |  | ||||||
|         assertThat(plaintext.getType(), equalTo(Plaintext.Type.MSG)); |         assertThat(plaintext.getType(), equalTo(Plaintext.Type.MSG)); | ||||||
|         assertThat(plaintext.getText(), equalTo(originalMessage)); |         assertThat(plaintext.getText(), equalTo(originalMessage)); | ||||||
|  |  | ||||||
|  |         Mockito.verify(aliceLabeler, Mockito.timeout(TimeUnit.MINUTES.toMillis(15)).atLeastOnce()) | ||||||
|  |                 .markAsAcknowledged(any()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Test |     @Test | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ package ch.dissem.bitmessage.repository; | |||||||
|  |  | ||||||
| import ch.dissem.bitmessage.entity.Streamable; | import ch.dissem.bitmessage.entity.Streamable; | ||||||
| import ch.dissem.bitmessage.entity.payload.ObjectType; | import ch.dissem.bitmessage.entity.payload.ObjectType; | ||||||
|  | import ch.dissem.bitmessage.exception.ApplicationException; | ||||||
| import org.slf4j.Logger; | import org.slf4j.Logger; | ||||||
| import org.slf4j.LoggerFactory; | import org.slf4j.LoggerFactory; | ||||||
|  |  | ||||||
| @@ -25,6 +26,7 @@ import java.io.ByteArrayOutputStream; | |||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.sql.PreparedStatement; | import java.sql.PreparedStatement; | ||||||
| import java.sql.SQLException; | import java.sql.SQLException; | ||||||
|  | import java.util.Collection; | ||||||
|  |  | ||||||
| import static ch.dissem.bitmessage.utils.Strings.hex; | import static ch.dissem.bitmessage.utils.Strings.hex; | ||||||
|  |  | ||||||
| @@ -85,4 +87,12 @@ public abstract class JdbcHelper { | |||||||
|             ps.setBytes(parameterIndex, os.toByteArray()); |             ps.setBytes(parameterIndex, os.toByteArray()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     protected <T> T single(Collection<T> collection) { | ||||||
|  |         if (collection.size() > 1) { | ||||||
|  |             throw new ApplicationException("This shouldn't happen, found " + collection.size() + | ||||||
|  |                     " messages, one or none was expected"); | ||||||
|  |         } | ||||||
|  |         return collection.stream().findAny().orElse(null); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -137,16 +137,12 @@ public class JdbcMessageRepository extends JdbcHelper implements MessageReposito | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public Plaintext getMessage(byte[] initialHash) { |     public Plaintext getMessage(byte[] initialHash) { | ||||||
|         List<Plaintext> plaintexts = find("initial_hash=X'" + Strings.hex(initialHash) + "'"); |         return single(find("initial_hash=X'" + Strings.hex(initialHash) + "'")); | ||||||
|         switch (plaintexts.size()) { |     } | ||||||
|             case 0: |  | ||||||
|                 return null; |     @Override | ||||||
|             case 1: |     public Plaintext getMessageForAck(byte[] ackData) { | ||||||
|                 return plaintexts.get(0); |         return single(find("ack_data=X'" + Strings.hex(ackData) + "' AND status='" + Plaintext.Status.SENT + "'")); | ||||||
|             default: |  | ||||||
|                 throw new ApplicationException("This shouldn't happen, found " + plaintexts.size() + |  | ||||||
|                         " messages, one or none was expected"); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -174,7 +170,7 @@ public class JdbcMessageRepository extends JdbcHelper implements MessageReposito | |||||||
|         try ( |         try ( | ||||||
|                 Connection connection = config.getConnection(); |                 Connection connection = config.getConnection(); | ||||||
|                 Statement stmt = connection.createStatement(); |                 Statement stmt = connection.createStatement(); | ||||||
|                 ResultSet rs = stmt.executeQuery("SELECT id, iv, type, sender, recipient, data, sent, received, status " + |                 ResultSet rs = stmt.executeQuery("SELECT id, iv, type, sender, recipient, data, ack_data, sent, received, status " + | ||||||
|                         "FROM Message WHERE " + where) |                         "FROM Message WHERE " + where) | ||||||
|         ) { |         ) { | ||||||
|             while (rs.next()) { |             while (rs.next()) { | ||||||
| @@ -187,6 +183,7 @@ public class JdbcMessageRepository extends JdbcHelper implements MessageReposito | |||||||
|                 builder.IV(new InventoryVector(iv)); |                 builder.IV(new InventoryVector(iv)); | ||||||
|                 builder.from(ctx.getAddressRepository().getAddress(rs.getString("sender"))); |                 builder.from(ctx.getAddressRepository().getAddress(rs.getString("sender"))); | ||||||
|                 builder.to(ctx.getAddressRepository().getAddress(rs.getString("recipient"))); |                 builder.to(ctx.getAddressRepository().getAddress(rs.getString("recipient"))); | ||||||
|  |                 builder.ackData(rs.getBytes("ack_data")); | ||||||
|                 builder.sent(rs.getLong("sent")); |                 builder.sent(rs.getLong("sent")); | ||||||
|                 builder.received(rs.getLong("received")); |                 builder.received(rs.getLong("received")); | ||||||
|                 builder.status(Plaintext.Status.valueOf(rs.getString("status"))); |                 builder.status(Plaintext.Status.valueOf(rs.getString("status"))); | ||||||
| @@ -268,8 +265,8 @@ public class JdbcMessageRepository extends JdbcHelper implements MessageReposito | |||||||
|  |  | ||||||
|     private void insert(Connection connection, Plaintext message) throws SQLException, IOException { |     private void insert(Connection connection, Plaintext message) throws SQLException, IOException { | ||||||
|         try (PreparedStatement ps = connection.prepareStatement( |         try (PreparedStatement ps = connection.prepareStatement( | ||||||
|                 "INSERT INTO Message (iv, type, sender, recipient, data, sent, received, status, initial_hash) " + |                 "INSERT INTO Message (iv, type, sender, recipient, data, ack_data, sent, received, status, initial_hash) " + | ||||||
|                         "VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)", |                         "VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", | ||||||
|                 Statement.RETURN_GENERATED_KEYS) |                 Statement.RETURN_GENERATED_KEYS) | ||||||
|         ) { |         ) { | ||||||
|             ps.setBytes(1, message.getInventoryVector() == null ? null : message.getInventoryVector().getHash()); |             ps.setBytes(1, message.getInventoryVector() == null ? null : message.getInventoryVector().getHash()); | ||||||
| @@ -277,10 +274,11 @@ public class JdbcMessageRepository extends JdbcHelper implements MessageReposito | |||||||
|             ps.setString(3, message.getFrom().getAddress()); |             ps.setString(3, message.getFrom().getAddress()); | ||||||
|             ps.setString(4, message.getTo() == null ? null : message.getTo().getAddress()); |             ps.setString(4, message.getTo() == null ? null : message.getTo().getAddress()); | ||||||
|             writeBlob(ps, 5, message); |             writeBlob(ps, 5, message); | ||||||
|             ps.setLong(6, message.getSent()); |             ps.setBytes(6, message.getAckData()); | ||||||
|             ps.setLong(7, message.getReceived()); |             ps.setLong(7, message.getSent()); | ||||||
|             ps.setString(8, message.getStatus() == null ? null : message.getStatus().name()); |             ps.setLong(8, message.getReceived()); | ||||||
|             ps.setBytes(9, message.getInitialHash()); |             ps.setString(9, message.getStatus() == null ? null : message.getStatus().name()); | ||||||
|  |             ps.setBytes(10, message.getInitialHash()); | ||||||
|  |  | ||||||
|             ps.executeUpdate(); |             ps.executeUpdate(); | ||||||
|             // get generated id |             // get generated id | ||||||
|   | |||||||
| @@ -0,0 +1 @@ | |||||||
|  | ALTER TABLE Message ADD COLUMN ack_data BINARY(32); | ||||||
		Reference in New Issue
	
	Block a user