first file transfer between gajim and conversations. still a lot to do though
This commit is contained in:
parent
c0e0a70869
commit
259bb446ca
|
@ -1,5 +1,6 @@
|
||||||
package eu.siacs.conversations.xmpp.jingle;
|
package eu.siacs.conversations.xmpp.jingle;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
@ -72,6 +73,8 @@ public class JingleConnection {
|
||||||
status = STATUS_TERMINATED;
|
status = STATUS_TERMINATED;
|
||||||
} else if (packet.isAction("session-accept")) {
|
} else if (packet.isAction("session-accept")) {
|
||||||
accept(packet);
|
accept(packet);
|
||||||
|
} else if (packet.isAction("transport-info")) {
|
||||||
|
transportInfo(packet);
|
||||||
} else {
|
} else {
|
||||||
Log.d("xmppService","packet arrived in connection. action was "+packet.getAction());
|
Log.d("xmppService","packet arrived in connection. action was "+packet.getAction());
|
||||||
}
|
}
|
||||||
|
@ -135,11 +138,40 @@ public class JingleConnection {
|
||||||
account.getXmppConnection().sendIqPacket(response, null);
|
account.getXmppConnection().sendIqPacket(response, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void transportInfo(JinglePacket packet) {
|
||||||
|
Content content = packet.getJingleContent();
|
||||||
|
Log.d("xmppService","transport info : "+content.toString());
|
||||||
|
String cid = content.getUsedCandidate();
|
||||||
|
if (cid!=null) {
|
||||||
|
final File file = this.mXmppConnectionService.getFileBackend().getImageFile(this.message);
|
||||||
|
final SocksConnection connection = this.connections.get(cid);
|
||||||
|
if (connection.isProxy()) {
|
||||||
|
IqPacket activation = new IqPacket(IqPacket.TYPE_SET);
|
||||||
|
activation.setTo(connection.getJid());
|
||||||
|
activation.query("http://jabber.org/protocol/bytestreams").setAttribute("sid", this.getSessionId());
|
||||||
|
activation.query().addChild("activate").setContent(this.getResponder());
|
||||||
|
Log.d("xmppService","connection is proxy. need to activate "+activation.toString());
|
||||||
|
this.account.getXmppConnection().sendIqPacket(activation, new OnIqPacketReceived() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onIqPacketReceived(Account account, IqPacket packet) {
|
||||||
|
Log.d("xmppService","activation result: "+packet.toString());
|
||||||
|
connection.send(file);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
connection.send(file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void connectWithCandidates() {
|
private void connectWithCandidates() {
|
||||||
for(Element canditate : this.candidates) {
|
for(Element canditate : this.candidates) {
|
||||||
String host = canditate.getAttribute("host");
|
String host = canditate.getAttribute("host");
|
||||||
int port = Integer.parseInt(canditate.getAttribute("port"));
|
int port = Integer.parseInt(canditate.getAttribute("port"));
|
||||||
SocksConnection socksConnection = new SocksConnection(this, host, port);
|
String type = canditate.getAttribute("type");
|
||||||
|
String jid = canditate.getAttribute("jid");
|
||||||
|
SocksConnection socksConnection = new SocksConnection(this, host, jid, port,type);
|
||||||
socksConnection.connect();
|
socksConnection.connect();
|
||||||
this.connections.put(canditate.getAttribute("cid"), socksConnection);
|
this.connections.put(canditate.getAttribute("cid"), socksConnection);
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,7 +44,6 @@ public class JingleConnectionManager {
|
||||||
}
|
}
|
||||||
|
|
||||||
public JingleConnection createNewConnection(Message message) {
|
public JingleConnection createNewConnection(Message message) {
|
||||||
Account account = message.getConversation().getAccount();
|
|
||||||
JingleConnection connection = new JingleConnection(this);
|
JingleConnection connection = new JingleConnection(this);
|
||||||
connection.init(message);
|
connection.init(message);
|
||||||
connections.add(connection);
|
connections.add(connection);
|
||||||
|
@ -57,12 +56,6 @@ public class JingleConnectionManager {
|
||||||
return connection;
|
return connection;
|
||||||
}
|
}
|
||||||
|
|
||||||
private String generateInternalId(String account, String counterpart,
|
|
||||||
String sid) {
|
|
||||||
return account + "#" + counterpart + "#" + sid;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public XmppConnectionService getXmppConnectionService() {
|
public XmppConnectionService getXmppConnectionService() {
|
||||||
return this.xmppConnectionService;
|
return this.xmppConnectionService;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,8 @@
|
||||||
package eu.siacs.conversations.xmpp.jingle;
|
package eu.siacs.conversations.xmpp.jingle;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
|
@ -18,13 +21,18 @@ public class SocksConnection {
|
||||||
private JingleConnection jingleConnection;
|
private JingleConnection jingleConnection;
|
||||||
private Socket socket;
|
private Socket socket;
|
||||||
private String host;
|
private String host;
|
||||||
|
private String jid;
|
||||||
private int port;
|
private int port;
|
||||||
|
private boolean isProxy = false;
|
||||||
private String destination;
|
private String destination;
|
||||||
|
private OutputStream outputStream;
|
||||||
|
|
||||||
public SocksConnection(JingleConnection jingleConnection, String host, int port) {
|
public SocksConnection(JingleConnection jingleConnection, String host, String jid, int port, String type) {
|
||||||
this.jingleConnection = jingleConnection;
|
this.jingleConnection = jingleConnection;
|
||||||
this.host = host;
|
this.host = host;
|
||||||
|
this.jid = jid;
|
||||||
this.port = port;
|
this.port = port;
|
||||||
|
this.isProxy = "proxy".equalsIgnoreCase(type);
|
||||||
try {
|
try {
|
||||||
MessageDigest mDigest = MessageDigest.getInstance("SHA-1");
|
MessageDigest mDigest = MessageDigest.getInstance("SHA-1");
|
||||||
StringBuilder destBuilder = new StringBuilder();
|
StringBuilder destBuilder = new StringBuilder();
|
||||||
|
@ -32,7 +40,6 @@ public class SocksConnection {
|
||||||
destBuilder.append(jingleConnection.getInitiator());
|
destBuilder.append(jingleConnection.getInitiator());
|
||||||
destBuilder.append(jingleConnection.getResponder());
|
destBuilder.append(jingleConnection.getResponder());
|
||||||
mDigest.reset();
|
mDigest.reset();
|
||||||
Log.d("xmppService","plain destination: "+destBuilder.toString());
|
|
||||||
this.destination = CryptoHelper.bytesToHex(mDigest.digest(destBuilder.toString().getBytes()));
|
this.destination = CryptoHelper.bytesToHex(mDigest.digest(destBuilder.toString().getBytes()));
|
||||||
Log.d("xmppService","host="+host+", port="+port+", destination: "+destination);
|
Log.d("xmppService","host="+host+", port="+port+", destination: "+destination);
|
||||||
} catch (NoSuchAlgorithmException e) {
|
} catch (NoSuchAlgorithmException e) {
|
||||||
|
@ -44,15 +51,15 @@ public class SocksConnection {
|
||||||
try {
|
try {
|
||||||
this.socket = new Socket(this.host, this.port);
|
this.socket = new Socket(this.host, this.port);
|
||||||
InputStream is = socket.getInputStream();
|
InputStream is = socket.getInputStream();
|
||||||
OutputStream os = socket.getOutputStream();
|
this.outputStream = socket.getOutputStream();
|
||||||
byte[] login = {0x05, 0x01, 0x00};
|
byte[] login = {0x05, 0x01, 0x00};
|
||||||
byte[] expectedReply = {0x05,0x00};
|
byte[] expectedReply = {0x05,0x00};
|
||||||
byte[] reply = new byte[2];
|
byte[] reply = new byte[2];
|
||||||
os.write(login);
|
this.outputStream.write(login);
|
||||||
is.read(reply);
|
is.read(reply);
|
||||||
if (Arrays.equals(reply, expectedReply)) {
|
if (Arrays.equals(reply, expectedReply)) {
|
||||||
String connect = ""+'\u0005'+'\u0001'+'\u0000'+'\u0003'+'\u0028'+this.destination+'\u0000'+'\u0000';
|
String connect = ""+'\u0005'+'\u0001'+'\u0000'+'\u0003'+'\u0028'+this.destination+'\u0000'+'\u0000';
|
||||||
os.write(connect.getBytes());
|
this.outputStream.write(connect.getBytes());
|
||||||
byte[] result = new byte[2];
|
byte[] result = new byte[2];
|
||||||
is.read(result);
|
is.read(result);
|
||||||
int status = result[0];
|
int status = result[0];
|
||||||
|
@ -67,4 +74,40 @@ public class SocksConnection {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void send(File file) {
|
||||||
|
FileInputStream fileInputStream = null;
|
||||||
|
try {
|
||||||
|
fileInputStream = new FileInputStream(file);
|
||||||
|
int count;
|
||||||
|
byte[] buffer = new byte[8192];
|
||||||
|
while ((count = fileInputStream.read(buffer)) > 0) {
|
||||||
|
this.outputStream.write(buffer, 0, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (FileNotFoundException e) {
|
||||||
|
// TODO Auto-generated catch block
|
||||||
|
e.printStackTrace();
|
||||||
|
} catch (IOException e) {
|
||||||
|
// TODO Auto-generated catch block
|
||||||
|
e.printStackTrace();
|
||||||
|
} finally {
|
||||||
|
try {
|
||||||
|
if (fileInputStream!=null) {
|
||||||
|
fileInputStream.close();
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
// TODO Auto-generated catch block
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isProxy() {
|
||||||
|
return this.isProxy;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getJid() {
|
||||||
|
return this.jid;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,4 +43,17 @@ public class Content extends Element {
|
||||||
return transport.getChildren();
|
return transport.getChildren();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getUsedCandidate() {
|
||||||
|
Element transport = this.findChild("transport", "urn:xmpp:jingle:transports:s5b:1");
|
||||||
|
if (transport==null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
Element usedCandidate = transport.findChild("candidate-used");
|
||||||
|
if (usedCandidate==null) {
|
||||||
|
return null;
|
||||||
|
} else {
|
||||||
|
return usedCandidate.getAttribute("cid");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue