Back to project page bitfynd-wallet-android.
The source code is released under:
GNU General Public License
If you think the Android project bitfynd-wallet-android listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.
/* * Copyright 2013-2014 the original author or authors. *//from w w w. j ava 2 s.c o m * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package de.schildbach.wallet.offline; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.HttpURLConnection; import java.net.URL; import javax.annotation.CheckForNull; import javax.annotation.Nonnull; import javax.annotation.Nullable; import org.bitcoin.protocols.payments.Protos; import org.bitcoin.protocols.payments.Protos.Payment; import org.bitcoinj.protocols.payments.PaymentProtocol; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import android.bluetooth.BluetoothAdapter; import android.bluetooth.BluetoothDevice; import android.bluetooth.BluetoothSocket; import android.os.Handler; import android.os.Looper; import de.schildbach.wallet.Constants; import de.schildbach.wallet.util.Bluetooth; import de.schildbach.wallet.R; /** * @author Andreas Schildbach */ public abstract class DirectPaymentTask { private final Handler backgroundHandler; private final Handler callbackHandler; private final ResultCallback resultCallback; private static final Logger log = LoggerFactory.getLogger(DirectPaymentTask.class); public interface ResultCallback { void onResult(boolean ack); void onFail(int messageResId, Object... messageArgs); } public DirectPaymentTask(@Nonnull final Handler backgroundHandler, @Nonnull final ResultCallback resultCallback) { this.backgroundHandler = backgroundHandler; this.callbackHandler = new Handler(Looper.myLooper()); this.resultCallback = resultCallback; } public final static class HttpPaymentTask extends DirectPaymentTask { private final String url; @CheckForNull private final String userAgent; public HttpPaymentTask(@Nonnull final Handler backgroundHandler, @Nonnull final ResultCallback resultCallback, @Nonnull final String url, @Nullable final String userAgent) { super(backgroundHandler, resultCallback); this.url = url; this.userAgent = userAgent; } @Override public void send(@Nonnull final Payment payment) { super.backgroundHandler.post(new Runnable() { @Override public void run() { log.info("trying to send tx to {}", url); HttpURLConnection connection = null; OutputStream os = null; InputStream is = null; try { connection = (HttpURLConnection) new URL(url).openConnection(); connection.setInstanceFollowRedirects(false); connection.setConnectTimeout(Constants.HTTP_TIMEOUT_MS); connection.setReadTimeout(Constants.HTTP_TIMEOUT_MS); connection.setUseCaches(false); connection.setDoInput(true); connection.setDoOutput(true); connection.setRequestMethod("POST"); connection.setRequestProperty("Content-Type", PaymentProtocol.MIMETYPE_PAYMENT); connection.setRequestProperty("Accept", PaymentProtocol.MIMETYPE_PAYMENTACK); connection.setRequestProperty("Content-Length", Integer.toString(payment.getSerializedSize())); if (userAgent != null) connection.addRequestProperty("User-Agent", userAgent); connection.connect(); os = connection.getOutputStream(); payment.writeTo(os); os.flush(); log.info("tx sent via http"); final int responseCode = connection.getResponseCode(); if (responseCode == HttpURLConnection.HTTP_OK) { is = connection.getInputStream(); final Protos.PaymentACK paymentAck = Protos.PaymentACK.parseFrom(is); final boolean ack = !"nack".equals(PaymentProtocol.parsePaymentAck(paymentAck).getMemo()); log.info("received {} via http", ack ? "ack" : "nack"); onResult(ack); } else { final String responseMessage = connection.getResponseMessage(); log.info("got http error {}: {}", responseCode, responseMessage); onFail(R.string.error_http, responseCode, responseMessage); } } catch (final IOException x) { log.info("problem sending", x); onFail(R.string.error_io, x.getMessage()); } finally { if (os != null) { try { os.close(); } catch (final IOException x) { // swallow } } if (is != null) { try { is.close(); } catch (final IOException x) { // swallow } } if (connection != null) connection.disconnect(); } } }); } } public final static class BluetoothPaymentTask extends DirectPaymentTask { private final BluetoothAdapter bluetoothAdapter; private final String bluetoothMac; public BluetoothPaymentTask(@Nonnull final Handler backgroundHandler, @Nonnull final ResultCallback resultCallback, @Nonnull final BluetoothAdapter bluetoothAdapter, @Nonnull final String bluetoothMac) { super(backgroundHandler, resultCallback); this.bluetoothAdapter = bluetoothAdapter; this.bluetoothMac = bluetoothMac; } @Override public void send(@Nonnull final Payment payment) { super.backgroundHandler.post(new Runnable() { @Override public void run() { log.info("trying to send tx via bluetooth {}", bluetoothMac); if (payment.getTransactionsCount() != 1) throw new IllegalArgumentException("wrong transactions count"); final BluetoothDevice device = bluetoothAdapter.getRemoteDevice(Bluetooth.decompressMac(bluetoothMac)); BluetoothSocket socket = null; DataOutputStream os = null; DataInputStream is = null; try { socket = device.createInsecureRfcommSocketToServiceRecord(Bluetooth.BIP70_PAYMENT_PROTOCOL_UUID); socket.connect(); log.info("connected to payment protocol {}", bluetoothMac); is = new DataInputStream(socket.getInputStream()); os = new DataOutputStream(socket.getOutputStream()); payment.writeDelimitedTo(os); os.flush(); log.info("tx sent via bluetooth"); final Protos.PaymentACK paymentAck = Protos.PaymentACK.parseDelimitedFrom(is); final boolean ack = "ack".equals(PaymentProtocol.parsePaymentAck(paymentAck).getMemo()); log.info("received {} via bluetooth", ack ? "ack" : "nack"); onResult(ack); } catch (final IOException x) { log.info("problem sending", x); onFail(R.string.error_io, x.getMessage()); } finally { if (os != null) { try { os.close(); } catch (final IOException x) { // swallow } } if (is != null) { try { is.close(); } catch (final IOException x) { // swallow } } if (socket != null) { try { socket.close(); } catch (final IOException x) { // swallow } } } } }); } } public abstract void send(@Nonnull Payment payment); protected void onResult(final boolean ack) { callbackHandler.post(new Runnable() { @Override public void run() { resultCallback.onResult(ack); } }); } protected void onFail(final int messageResId, final Object... messageArgs) { callbackHandler.post(new Runnable() { @Override public void run() { resultCallback.onFail(messageResId, messageArgs); } }); } }