blob: 88b91cea4525c337bad1d85bd5ae21d8caf345e5 [file] [log] [blame]
Shuyi Chend7955ce2013-05-22 14:51:55 -07001/**
2 * $RCSfile$
3 * $Revision$
4 * $Date$
5 *
6 * Copyright 2003-2007 Jive Software.
7 *
8 * All rights reserved. Licensed under the Apache License, Version 2.0 (the "License");
9 * you may not use this file except in compliance with the License.
10 * You may obtain a copy of the License at
11 *
12 * http://www.apache.org/licenses/LICENSE-2.0
13 *
14 * Unless required by applicable law or agreed to in writing, software
15 * distributed under the License is distributed on an "AS IS" BASIS,
16 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17 * See the License for the specific language governing permissions and
18 * limitations under the License.
19 */
20
21package org.jivesoftware.smack;
22
23import org.jivesoftware.smack.filter.PacketIDFilter;
24import org.jivesoftware.smack.packet.Authentication;
25import org.jivesoftware.smack.packet.IQ;
26
27import org.apache.harmony.javax.security.auth.callback.CallbackHandler;
28import org.apache.harmony.javax.security.auth.callback.PasswordCallback;
29import org.apache.harmony.javax.security.auth.callback.Callback;
30
31/**
32 * Implementation of JEP-0078: Non-SASL Authentication. Follow the following
33 * <a href=http://www.jabber.org/jeps/jep-0078.html>link</a> to obtain more
34 * information about the JEP.
35 *
36 * @author Gaston Dombiak
37 */
38class NonSASLAuthentication implements UserAuthentication {
39
40 private Connection connection;
41
42 public NonSASLAuthentication(Connection connection) {
43 super();
44 this.connection = connection;
45 }
46
47 public String authenticate(String username, String resource, CallbackHandler cbh) throws XMPPException {
48 //Use the callback handler to determine the password, and continue on.
49 PasswordCallback pcb = new PasswordCallback("Password: ",false);
50 try {
51 cbh.handle(new Callback[]{pcb});
52 return authenticate(username, String.valueOf(pcb.getPassword()),resource);
53 } catch (Exception e) {
54 throw new XMPPException("Unable to determine password.",e);
55 }
56 }
57
58 public String authenticate(String username, String password, String resource) throws
59 XMPPException {
60 // If we send an authentication packet in "get" mode with just the username,
61 // the server will return the list of authentication protocols it supports.
62 Authentication discoveryAuth = new Authentication();
63 discoveryAuth.setType(IQ.Type.GET);
64 discoveryAuth.setUsername(username);
65
66 PacketCollector collector =
67 connection.createPacketCollector(new PacketIDFilter(discoveryAuth.getPacketID()));
68 // Send the packet
69 connection.sendPacket(discoveryAuth);
70 // Wait up to a certain number of seconds for a response from the server.
71 IQ response = (IQ) collector.nextResult(SmackConfiguration.getPacketReplyTimeout());
72 if (response == null) {
73 throw new XMPPException("No response from the server.");
74 }
75 // If the server replied with an error, throw an exception.
76 else if (response.getType() == IQ.Type.ERROR) {
77 throw new XMPPException(response.getError());
78 }
79 // Otherwise, no error so continue processing.
80 Authentication authTypes = (Authentication) response;
81 collector.cancel();
82
83 // Now, create the authentication packet we'll send to the server.
84 Authentication auth = new Authentication();
85 auth.setUsername(username);
86
87 // Figure out if we should use digest or plain text authentication.
88 if (authTypes.getDigest() != null) {
89 auth.setDigest(connection.getConnectionID(), password);
90 }
91 else if (authTypes.getPassword() != null) {
92 auth.setPassword(password);
93 }
94 else {
95 throw new XMPPException("Server does not support compatible authentication mechanism.");
96 }
97
98 auth.setResource(resource);
99
100 collector = connection.createPacketCollector(new PacketIDFilter(auth.getPacketID()));
101 // Send the packet.
102 connection.sendPacket(auth);
103 // Wait up to a certain number of seconds for a response from the server.
104 response = (IQ) collector.nextResult(SmackConfiguration.getPacketReplyTimeout());
105 if (response == null) {
106 throw new XMPPException("Authentication failed.");
107 }
108 else if (response.getType() == IQ.Type.ERROR) {
109 throw new XMPPException(response.getError());
110 }
111 // We're done with the collector, so explicitly cancel it.
112 collector.cancel();
113
114 return response.getTo();
115 }
116
117 public String authenticateAnonymously() throws XMPPException {
118 // Create the authentication packet we'll send to the server.
119 Authentication auth = new Authentication();
120
121 PacketCollector collector =
122 connection.createPacketCollector(new PacketIDFilter(auth.getPacketID()));
123 // Send the packet.
124 connection.sendPacket(auth);
125 // Wait up to a certain number of seconds for a response from the server.
126 IQ response = (IQ) collector.nextResult(SmackConfiguration.getPacketReplyTimeout());
127 if (response == null) {
128 throw new XMPPException("Anonymous login failed.");
129 }
130 else if (response.getType() == IQ.Type.ERROR) {
131 throw new XMPPException(response.getError());
132 }
133 // We're done with the collector, so explicitly cancel it.
134 collector.cancel();
135
136 if (response.getTo() != null) {
137 return response.getTo();
138 }
139 else {
140 return connection.getServiceName() + "/" + ((Authentication) response).getResource();
141 }
142 }
143}