/** * Copyright 2010 The University of Nottingham * * This file is part of lobbyservice. * * lobbyservice is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * lobbyservice 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 Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with lobbyservice. If not, see <http://www.gnu.org/licenses/>. * */ package uk.ac.horizon.ug.lobby.browser; import com.google.appengine.api.datastore.Key; import com.google.appengine.api.datastore.KeyFactory; import com.google.appengine.api.users.User; import com.google.appengine.api.users.UserService; import com.google.appengine.api.users.UserServiceFactory; import java.io.BufferedReader; import java.io.IOException; import java.io.OutputStreamWriter; import java.io.Writer; import java.security.NoSuchAlgorithmException; import java.security.SecureRandom; import java.util.HashMap; import java.util.LinkedList; import java.util.List; import java.util.Map; import java.util.Random; import java.util.TreeSet; import java.util.logging.Logger; import javax.persistence.EntityManager; import javax.persistence.EntityManagerFactory; import javax.persistence.EntityTransaction; import javax.persistence.Query; import javax.servlet.http.*; import org.json.JSONException; import org.json.JSONObject; import org.json.JSONWriter; import uk.ac.horizon.ug.lobby.ConfigurationUtils; import uk.ac.horizon.ug.lobby.Constants; import uk.ac.horizon.ug.lobby.HttpUtils; import uk.ac.horizon.ug.lobby.RequestException; import uk.ac.horizon.ug.lobby.model.Account; import uk.ac.horizon.ug.lobby.model.EMF; import uk.ac.horizon.ug.lobby.model.GUIDFactory; import uk.ac.horizon.ug.lobby.model.GameClient; import uk.ac.horizon.ug.lobby.model.GameClientTemplate; import uk.ac.horizon.ug.lobby.model.GameIndex; import uk.ac.horizon.ug.lobby.model.GameInstance; import uk.ac.horizon.ug.lobby.model.GameInstanceFactory; import uk.ac.horizon.ug.lobby.model.GameInstanceFactoryStatus; import uk.ac.horizon.ug.lobby.model.GameInstanceFactoryType; import uk.ac.horizon.ug.lobby.model.GameInstanceNominalStatus; import uk.ac.horizon.ug.lobby.model.GameInstanceSlot; import uk.ac.horizon.ug.lobby.model.GameInstanceSlotStatus; import uk.ac.horizon.ug.lobby.model.GameServer; import uk.ac.horizon.ug.lobby.model.GameServerStatus; import uk.ac.horizon.ug.lobby.model.GameTemplate; import uk.ac.horizon.ug.lobby.model.GameTemplateVisibility; import uk.ac.horizon.ug.lobby.model.ServerConfiguration; import uk.ac.horizon.ug.lobby.protocol.ClientRequest; import uk.ac.horizon.ug.lobby.protocol.ClientRequestScope; import uk.ac.horizon.ug.lobby.protocol.ClientRequestType; import uk.ac.horizon.ug.lobby.protocol.ClientResponse; import uk.ac.horizon.ug.lobby.protocol.ClientResponseStatus; import uk.ac.horizon.ug.lobby.protocol.GameJoinRequest; import uk.ac.horizon.ug.lobby.protocol.GameJoinRequestType; import uk.ac.horizon.ug.lobby.protocol.GameJoinResponse; import uk.ac.horizon.ug.lobby.protocol.GameJoinResponseStatus; import uk.ac.horizon.ug.lobby.protocol.GameQuery; import uk.ac.horizon.ug.lobby.protocol.GameTemplateInfo; import uk.ac.horizon.ug.lobby.protocol.JSONUtils; import uk.ac.horizon.ug.lobby.protocol.LocationConstraint; import uk.ac.horizon.ug.lobby.protocol.TimeConstraint; import uk.ac.horizon.ug.lobby.server.CronExpressionException; import uk.ac.horizon.ug.lobby.server.FactoryTasks; import uk.ac.horizon.ug.lobby.server.FactoryUtils; import uk.ac.horizon.ug.lobby.server.ServerProtocol; import uk.ac.horizon.ug.lobby.user.UserGameTemplateServlet; import uk.me.jstott.jcoord.LatLng; /** * Handle request a ClientRequest, initially LIST_GAMES. * * @author cmg * */ @SuppressWarnings("serial") public class ClientRequestServlet extends HttpServlet implements Constants { static Logger logger = Logger.getLogger(ClientRequestServlet.class.getName()); @Override public void doPost(HttpServletRequest req, HttpServletResponse resp) throws IOException { // parse request String line = null; String auth = null; ClientRequest creq = null; try { BufferedReader br = req.getReader(); line = br.readLine(); JSONObject json = new JSONObject(line); creq = JSONUtils.parseClientRequest(json); // second line is digital signature (if given) auth = br.readLine(); logger.info("ClientRequest "+creq); } catch (JSONException e) { logger.warning(e.toString()); resp.sendError(HttpServletResponse.SC_BAD_REQUEST, e.toString()); return; } try { // app logic... ClientResponse cresp = handleRequest(creq, req.getRequestURI(), line, auth); JSONUtils.sendClientResponse(resp, cresp); } catch (RequestException e) { resp.sendError(e.getErrorCode(), e.getMessage()); return; } catch (JoinException e) { ClientResponse cresp = new ClientResponse(); try { cresp.setStatus(ClientResponseStatus.valueOf(e.getStatus().toString())); } catch (Exception e2) { logger.warning("Cannot map JoinException "+e.getStatus()+" to ClientResponseStatus: "+e2); resp.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR, "Mapping error return "+e.getStatus()); return; } cresp.setTime(System.currentTimeMillis()); //cresp.setType(gjreq.getType()); cresp.setMessage(e.getMessage()); logger.warning(e.toString()); JSONUtils.sendClientResponse(resp, cresp); return; } } private ClientResponse handleRequest(ClientRequest creq, String requestUri, String line, String auth) throws IOException, JoinException, RequestException { // authenticate JoinUtils.JoinAuthInfo jai = JoinUtils.authenticate(creq.getClientId(), null, null, true, requestUri, line, auth); ClientResponse cresp = null; switch (creq.getType()) { case LIST_GAMES: cresp = handleListGames(creq, jai.gc, jai.account); break; default: throw new RequestException(HttpServletResponse.SC_BAD_REQUEST, "Request type must be LIST_GAMES ("+creq.getType()+")"); } return cresp; } private ClientResponse handleListGames(ClientRequest creq, GameClient gc, Account account) throws JoinException { ClientResponse cresp = new ClientResponse(); cresp.setTime(System.currentTimeMillis()); // hopefully :) cresp.setStatus(ClientResponseStatus.OK); if (creq.getScope()==null) { // default to CLIENT creq.setScope(ClientRequestScope.CLIENT); } cresp.setGames(getGameList(gc, account, creq.getScope(), creq.getIncludePlanned(), creq.getIncludeAvailable(), creq.getIncludeEnded())); return cresp; } public static List<GameTemplateInfo> getGameList(GameClient gc, Account account, ClientRequestScope scope, Boolean includePlanned, Boolean includeAvailable, Boolean includeEnded) throws JoinException { ServerConfiguration sc = ConfigurationUtils.getServerConfiguration(); // need to get suitable games... List<GameTemplateInfo> gtis = new LinkedList<GameTemplateInfo>(); // TODO filter / structure request to cope with large numbers of slots ?! // TODO map status to slot to allow filter in query EntityManager em = EMF.get().createEntityManager(); // readonly - no transaction?! try { Query q = null; switch (scope) { case ACCOUNT: if (account==null) throw new JoinException(GameJoinResponseStatus.ERROR_USER_AUTHENTICATION_REQUIRED, "Account-scope query requires user authentication"); q = em.createQuery("SELECT x FROM "+GameInstanceSlot.class.getSimpleName()+" x WHERE x."+ACCOUNT_KEY+" = :"+ACCOUNT_KEY); q.setParameter(ACCOUNT_KEY, account.getKey()); break; case CLIENT: if (gc==null) throw new JoinException(GameJoinResponseStatus.ERROR_CLIENT_AUTHENTICATION_REQUIRED, "Client-scope query requires client authentication"); q = em.createQuery("SELECT x FROM "+GameInstanceSlot.class.getSimpleName()+" x WHERE x."+GAME_CLIENT_KEY+" = :"+GAME_CLIENT_KEY); q.setParameter(GAME_CLIENT_KEY, gc.getKey()); break; } List<GameInstanceSlot> posgiss = (List<GameInstanceSlot>)q.getResultList(); for (GameInstanceSlot posgis: posgiss) { if (posgis.getGameInstanceKey()==null) { logger.warning("GameInstanceSlot without gameInstanceKey: "+posgis); continue; } GameInstance gi = em.find(GameInstance.class, posgis.getGameInstanceKey()); if (gi==null) { logger.warning("Could not find GameInstance for slot: "+posgis); continue; } boolean include = false; // include? switch (gi.getNominalStatus()) { case CANCELLED: case ENDED: if (includeEnded==null || includeEnded==true) include = true; break; case PLANNED: if (includePlanned==null || includePlanned==true) include = true; break; case TEMPORARILY_UNAVAILABLE: case AVAILABLE: if (includeAvailable==null || includeAvailable==true) include = true; break; } if (include) { GameTemplateInfo gti = new GameTemplateInfo(); gti.setGameInstance(gi); GameTemplate gt = em.find(GameTemplate.class, GameTemplate.idToKey(gi.getGameTemplateId())); gti.setGameTemplate(gt); // GameClientTemplates?? gti.setJoinUrl(QueryGameTemplateServlet.makeJoinUrl(sc, gi)); gtis.add(gti); } } } finally { em.close(); } return gtis; } }