/**
* Copyright 2010 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*/
package org.waveprotocol.box.server.robots.passive;
import com.google.inject.Inject;
import com.google.wave.api.InvalidRequestException;
import com.google.wave.api.OperationRequest;
import com.google.wave.api.ProtocolVersion;
import com.google.wave.api.RobotSerializer;
import com.google.wave.api.impl.EventMessageBundle;
import com.google.wave.api.robot.CapabilityFetchException;
import com.google.wave.api.robot.RobotCapabilitiesParser;
import com.google.wave.api.robot.RobotConnection;
import com.google.wave.api.robot.RobotConnectionException;
import org.waveprotocol.box.server.account.RobotAccountData;
import org.waveprotocol.box.server.account.RobotAccountDataImpl;
import org.waveprotocol.box.server.robots.RobotCapabilities;
import org.waveprotocol.wave.util.logging.Log;
import java.util.Collections;
import java.util.List;
/**
* This class sends {@link EventMessageBundle} to a robot and receives their
* response. It will gracefully handle failure by acting like the robot sent no
* operations.
*
* @author ljvderijk@google.com (Lennard de Rijk)
*/
public class RobotConnector {
private static final Log LOG = Log.get(RobotConnector.class);
private final RobotConnection connection;
private final RobotSerializer serializer;
@Inject
public RobotConnector(RobotConnection connection, RobotSerializer serializer) {
this.connection = connection;
this.serializer = serializer;
}
/**
* Synchronously sends an {@link EventMessageBundle} off to a robot and hands
* back the response in the form of a list of {@link OperationRequest}s.
*
* @param bundle the bundle to send to the robot.
* @param robot the {@link RobotAccountData} of the robot.
* @param version the version that we should speak to the robot.
* @returns list of {@link OperationRequest}s that the robot wants to have
* executed.
*/
public List<OperationRequest> sendMessageBundle(
EventMessageBundle bundle, Robot robot, ProtocolVersion version) {
String serializedBundle = serializer.serialize(bundle, version);
String robotUrl = robot.getAccount().getUrl() + Robot.RPC_URL;
LOG.info("Sending: " + serializedBundle + " to " + robotUrl);
try {
String response = connection.postJson(robotUrl, serializedBundle);
LOG.info("Received: " + response + " from " + robotUrl);
return serializer.deserializeOperations(response);
} catch (RobotConnectionException e) {
LOG.info("Failed to receive a response from " + robotUrl, e);
} catch (InvalidRequestException e) {
LOG.info("Failed to deserialize passive API response", e);
}
// Return an empty list and let the caller ignore the failure
return Collections.emptyList();
}
/**
* Returns a new {@link RobotAccountData} updated with the new capabilities
* using the given {@link RobotAccountData}.
*
* @param account The {@link RobotAccountData} to update the capabilities for.
* @param activeApiUrl the url of the Active Robot API.
* @throws CapabilityFetchException if the capabilities couldn't be retrieved
* or parsed.
*/
public RobotAccountData fetchCapabilities(RobotAccountData account, String activeApiUrl)
throws CapabilityFetchException {
RobotCapabilitiesParser parser = new RobotCapabilitiesParser(
account.getUrl() + Robot.CAPABILITIES_URL, connection, activeApiUrl);
RobotCapabilities capabilities = new RobotCapabilities(
parser.getCapabilities(), parser.getCapabilitiesHash(), parser.getProtocolVersion());
return new RobotAccountDataImpl(account.getId(), account.getUrl(), account.getConsumerSecret(),
capabilities, account.isVerified());
}
}