webidor/src/main/java/cz/xelfi/quoridor/webidor/resources/Games.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Tue, 08 Dec 2009 15:34:50 +0100
branchstrict-games-access
changeset 162 c1bfbe98152b
parent 131 19e81456eef2
child 164 2949998db4f6
permissions -rw-r--r--
Only active games are visible for regular users. Need a bit more work to allow them to see history of active games.
jtulach@36
     1
/*
jtulach@36
     2
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
jtulach@36
     3
 *
jtulach@36
     4
 * The contents of this file are subject to the terms of either the GNU
jtulach@36
     5
 * General Public License Version 2 only ("GPL") or the Common
jtulach@36
     6
 * Development and Distribution License("CDDL") (collectively, the
jtulach@36
     7
 * "License"). You may not use this file except in compliance with the
jtulach@36
     8
 * License. You can obtain a copy of the License at
jtulach@36
     9
 * http://www.netbeans.org/cddl-gplv2.html
jtulach@36
    10
 * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the
jtulach@36
    11
 * specific language governing permissions and limitations under the
jtulach@36
    12
 * License.  When distributing the software, include this License Header
jtulach@36
    13
 * Notice in each file and include the License file at
jtulach@36
    14
 * nbbuild/licenses/CDDL-GPL-2-CP.  Sun designates this
jtulach@36
    15
 * particular file as subject to the "Classpath" exception as provided
jtulach@36
    16
 * by Sun in the GPL Version 2 section of the License file that
jtulach@36
    17
 * accompanied this code. If applicable, add the following below the
jtulach@36
    18
 * License Header, with the fields enclosed by brackets [] replaced by
jtulach@36
    19
 * your own identifying information:
jtulach@36
    20
 * "Portions Copyrighted [year] [name of copyright owner]"
jtulach@36
    21
 *
jtulach@36
    22
 * Contributor(s):
jtulach@36
    23
 *
jtulach@36
    24
 * Portions Copyrighted 2009 Jaroslav Tulach
jtulach@36
    25
 */
jtulach@36
    26
jtulach@36
    27
package cz.xelfi.quoridor.webidor.resources;
jtulach@36
    28
jtulach@36
    29
import cz.xelfi.quoridor.IllegalPositionException;
jtulach@36
    30
import cz.xelfi.quoridor.webidor.*;
jtulach@36
    31
import cz.xelfi.quoridor.Move;
jtulach@37
    32
import java.io.BufferedReader;
jtulach@37
    33
import java.io.BufferedWriter;
jtulach@37
    34
import java.io.File;
jtulach@117
    35
import java.io.FileInputStream;
jtulach@117
    36
import java.io.FileOutputStream;
jtulach@36
    37
import java.io.IOException;
jtulach@117
    38
import java.io.InputStream;
jtulach@117
    39
import java.io.InputStreamReader;
jtulach@117
    40
import java.io.OutputStreamWriter;
jtulach@37
    41
import java.io.PrintWriter;
jtulach@117
    42
import java.io.Writer;
jtulach@36
    43
import java.util.ArrayList;
jaroslav@96
    44
import java.util.Collections;
jaroslav@48
    45
import java.util.Date;
jtulach@36
    46
import java.util.List;
jtulach@37
    47
import java.util.logging.Level;
jtulach@37
    48
import java.util.logging.Logger;
jaroslav@115
    49
import java.util.regex.Matcher;
jaroslav@115
    50
import java.util.regex.Pattern;
jaroslav@100
    51
import javax.ws.rs.DefaultValue;
jtulach@36
    52
import javax.ws.rs.GET;
jtulach@36
    53
import javax.ws.rs.POST;
jtulach@36
    54
import javax.ws.rs.PUT;
jtulach@36
    55
import javax.ws.rs.Path;
jtulach@36
    56
import javax.ws.rs.PathParam;
jtulach@36
    57
import javax.ws.rs.Produces;
jtulach@36
    58
import javax.ws.rs.QueryParam;
jtulach@82
    59
import javax.ws.rs.WebApplicationException;
jtulach@36
    60
import javax.ws.rs.core.MediaType;
jtulach@82
    61
import javax.ws.rs.core.Response.Status;
jtulach@36
    62
jtulach@36
    63
/**
jtulach@36
    64
 *
jtulach@36
    65
 * @author Jaroslav Tulach <jtulach@netbeans.org>
jtulach@36
    66
 */
jtulach@36
    67
public final class Games {
jtulach@82
    68
    private final Quoridor quoridor;
jaroslav@48
    69
    private final List<Game> games = new ArrayList<Game>();
jtulach@37
    70
    private final File dir;
jtulach@37
    71
    private static final Logger LOG = Logger.getLogger(Games.class.getName());
jtulach@37
    72
jtulach@82
    73
    public Games(File dir, Quoridor quoridor) {
jtulach@37
    74
        this.dir = dir;
jtulach@82
    75
        this.quoridor = quoridor;
jtulach@37
    76
        File[] arr = dir.listFiles();
jtulach@37
    77
        if (arr != null) {
jtulach@37
    78
            for (File f : arr) {
jtulach@37
    79
                try {
jtulach@37
    80
                    Game g = readGame(f);
jtulach@37
    81
                    games.add(g);
jtulach@37
    82
                } catch (IOException ex) {
jtulach@37
    83
                    LOG.log(Level.WARNING, "Wrong game in " + f, ex);
jtulach@37
    84
                }
jtulach@37
    85
            }
jtulach@37
    86
        }
jtulach@37
    87
    }
jtulach@36
    88
jtulach@36
    89
    @POST
jaroslav@48
    90
    @Produces({ MediaType.APPLICATION_JSON, MediaType.TEXT_XML })
jaroslav@48
    91
    public GameId createGame(
jtulach@82
    92
        @QueryParam("loginID") String id,
jtulach@38
    93
        @QueryParam("white") String white,
jtulach@38
    94
        @QueryParam("black") String black
jaroslav@48
    95
    ) throws IOException {
jtulach@82
    96
        String logUser = quoridor.isLoggedIn(id);
jtulach@82
    97
        if (logUser == null) {
jtulach@82
    98
            throw new WebApplicationException(Status.UNAUTHORIZED);
jtulach@82
    99
        }
jtulach@38
   100
        if (white == null) {
jtulach@82
   101
            throw new WebApplicationException(Status.PRECONDITION_FAILED);
jtulach@38
   102
        }
jtulach@38
   103
        if (black == null) {
jtulach@82
   104
            throw new WebApplicationException(Status.PRECONDITION_FAILED);
jtulach@38
   105
        }
jtulach@82
   106
        if (!logUser.equals(white) && !logUser.equals(black)) {
jtulach@82
   107
            throw new WebApplicationException(Status.PRECONDITION_FAILED);
jtulach@82
   108
        }
jtulach@82
   109
jtulach@38
   110
        Game g = new Game(white, black);
jaroslav@48
   111
        storeGame(g);
jtulach@36
   112
        games.add(g);
jaroslav@48
   113
        return g.getId();
jtulach@36
   114
    }
jtulach@36
   115
jtulach@38
   116
    @GET
jtulach@38
   117
    @Path("{id}")
jtulach@38
   118
    @Produces(MediaType.TEXT_PLAIN)
jaroslav@106
   119
    public String getBoard(
jaroslav@106
   120
        @PathParam("id") String id,
jaroslav@106
   121
        @QueryParam("move") @DefaultValue("-1") int move
jaroslav@106
   122
    ) {
jaroslav@106
   123
        Game g = findGame(id, move);
jtulach@38
   124
        if (g == null) {
jtulach@38
   125
            throw new IllegalArgumentException("Unknown game " + id);
jtulach@38
   126
        }
jtulach@38
   127
        return g.getBoard().toString();
jtulach@38
   128
    }
jtulach@38
   129
jaroslav@46
   130
    @GET
jaroslav@46
   131
    @Path("{id}")
jaroslav@48
   132
    @Produces({ MediaType.APPLICATION_JSON, MediaType.TEXT_XML })
jaroslav@100
   133
    public Game getBoardInfo(
jaroslav@162
   134
        @QueryParam("loginID") @DefaultValue("") String loginId,
jaroslav@100
   135
        @PathParam("id") String id,
jaroslav@100
   136
        @QueryParam("move") @DefaultValue("-1") int move
jaroslav@100
   137
    ) {
jaroslav@162
   138
        Game g = findGame(id, move);
jaroslav@162
   139
        if (g.getId().getStatus().isInProgress()) {
jaroslav@162
   140
            return g;
jaroslav@162
   141
        }
jaroslav@162
   142
        String logUser = quoridor.isLoggedIn(loginId);
jaroslav@162
   143
        if (logUser == null) {
jaroslav@162
   144
            throw new WebApplicationException(Status.UNAUTHORIZED);
jaroslav@162
   145
        }
jaroslav@162
   146
        if (logUser.equals(g.getId().getWhite())) {
jaroslav@162
   147
            return g;
jaroslav@162
   148
        }
jaroslav@162
   149
        if (logUser.equals(g.getId().getBlack())) {
jaroslav@162
   150
            return g;
jaroslav@162
   151
        }
jaroslav@162
   152
        throw new WebApplicationException(Status.UNAUTHORIZED);
jaroslav@46
   153
    }
jaroslav@46
   154
jtulach@36
   155
    @PUT
jtulach@36
   156
    @Path("{id}")
jaroslav@48
   157
    @Produces({ MediaType.APPLICATION_JSON, MediaType.TEXT_XML })
jaroslav@48
   158
    public GameId applyMove(
jtulach@82
   159
        @QueryParam("loginID") String loginId,
jtulach@38
   160
        @PathParam("id") String id,
jtulach@38
   161
        @QueryParam("player") String player,
jaroslav@115
   162
        @QueryParam("move") String move,
jaroslav@115
   163
        @QueryParam("comment") String comment
jtulach@38
   164
    ) throws IllegalPositionException {
jtulach@82
   165
        String logUser = quoridor.isLoggedIn(loginId);
jtulach@82
   166
        if (logUser == null) {
jtulach@82
   167
            throw new WebApplicationException(Status.UNAUTHORIZED);
jtulach@82
   168
        }
jtulach@82
   169
        if (!logUser.equals(player)) {
jtulach@82
   170
            throw new WebApplicationException(Status.UNAUTHORIZED);
jtulach@82
   171
        }
jaroslav@115
   172
        if (comment == null && move == null) {
jaroslav@115
   173
            throw new WebApplicationException(Status.BAD_REQUEST);
jaroslav@115
   174
        }
jtulach@82
   175
jtulach@36
   176
        Game g = findGame(id);
jtulach@36
   177
        if (g == null) {
jtulach@36
   178
            throw new IllegalArgumentException("Unknown game " + id);
jtulach@36
   179
        }
jaroslav@115
   180
        if (move != null) {
jaroslav@115
   181
            Move m = Move.valueOf(move);
jaroslav@115
   182
            g.apply(player, m, new Date());
jaroslav@115
   183
        }
jaroslav@115
   184
        if (comment != null) {
jaroslav@115
   185
            g.comment(player, comment, new Date());
jaroslav@115
   186
        }
jtulach@37
   187
        try {
jtulach@37
   188
            storeGame(g);
jtulach@37
   189
        } catch (IOException ex) {
jtulach@37
   190
            LOG.log(Level.WARNING, "Cannot store game " + id, ex);
jtulach@37
   191
        }
jaroslav@48
   192
        return g.getId();
jtulach@36
   193
    }
jtulach@36
   194
jtulach@36
   195
    @GET
jaroslav@48
   196
    @Produces({MediaType.APPLICATION_JSON, MediaType.TEXT_XML })
jaroslav@128
   197
    public List<GameId> listGames(
jaroslav@128
   198
        @DefaultValue("") @QueryParam("status") String status
jaroslav@128
   199
    ) {
jaroslav@48
   200
        List<GameId> arr = new ArrayList<GameId>(games.size());
jaroslav@48
   201
        for (Game g : games) {
jaroslav@128
   202
            if (status.length() == 0 || g.getId().getStatus().toString().equals(status)) {
jaroslav@128
   203
                arr.add(g.getId());
jaroslav@128
   204
            }
jaroslav@48
   205
        }
jaroslav@96
   206
        Collections.sort(arr, GameId.NEWEST_FIRST);
jaroslav@48
   207
        return arr;
jaroslav@48
   208
    }
jaroslav@48
   209
jtulach@36
   210
    public List<Game> getGames() {
jtulach@36
   211
        return games;
jtulach@36
   212
    }
jtulach@36
   213
jtulach@36
   214
    private Game findGame(String id) {
jtulach@36
   215
        for (Game g : games) {
jaroslav@48
   216
            if (g.getId().getId().equals(id)) {
jtulach@36
   217
                return g;
jtulach@36
   218
            }
jtulach@36
   219
        }
jtulach@36
   220
        return null;
jtulach@36
   221
    }
jaroslav@100
   222
    private Game findGame(String id, int move) {
jaroslav@100
   223
        Game g = findGame(id);
jaroslav@100
   224
        if (g == null) {
jaroslav@100
   225
            throw new IllegalArgumentException("Unknown game " + id);
jaroslav@100
   226
        }
jaroslav@100
   227
        try {
jaroslav@100
   228
            return move == -1 ? g : g.snapshot(move);
jaroslav@100
   229
        } catch (IllegalPositionException ex) {
jaroslav@100
   230
            Logger.getLogger(Games.class.getName()).log(Level.SEVERE, null, ex);
jaroslav@100
   231
            return null;
jaroslav@100
   232
        }
jaroslav@100
   233
    }
jtulach@36
   234
jaroslav@115
   235
    private static final Pattern saidWho = Pattern.compile("# *([^ ]*) *@(.*):$");
jaroslav@115
   236
jtulach@37
   237
    private Game readGame(File f) throws IOException {
jtulach@117
   238
        InputStream is = new FileInputStream(f);
jtulach@117
   239
        BufferedReader r = new BufferedReader(new InputStreamReader(is, "UTF-8"));
jtulach@37
   240
        String white = null;
jtulach@37
   241
        String black = null;
jtulach@37
   242
        Game g = null;
jaroslav@115
   243
        String who = null;
jaroslav@115
   244
        String when = null;
jtulach@37
   245
        for (;;) {
jtulach@37
   246
            String line = r.readLine();
jtulach@37
   247
            if (line == null) {
jaroslav@48
   248
                line = "finish";
jtulach@37
   249
            }
jtulach@39
   250
            line = line.trim();
jtulach@39
   251
            if (line.length() == 0) {
jtulach@39
   252
                continue;
jtulach@39
   253
            }
jtulach@37
   254
            if (line.startsWith("# white: ")) {
jtulach@37
   255
                white = line.substring(9);
jtulach@37
   256
                continue;
jtulach@37
   257
            }
jtulach@37
   258
            if (line.startsWith("# black: ")) {
jtulach@37
   259
                black = line.substring(9);
jtulach@37
   260
                continue;
jtulach@37
   261
            }
jtulach@37
   262
            if (line.startsWith("#")) {
jaroslav@115
   263
                Matcher m =saidWho.matcher(line);
jaroslav@115
   264
                if (m.matches()) {
jaroslav@115
   265
                    who = m.group(1);
jaroslav@115
   266
                    when = m.group(2);
jaroslav@115
   267
                    continue;
jaroslav@115
   268
                }
jaroslav@115
   269
                if (g == null) {
jaroslav@115
   270
                    continue;
jaroslav@115
   271
                }
jaroslav@115
   272
                if (line.startsWith("# ")) {
jaroslav@115
   273
                    line = line.substring(2);
jaroslav@115
   274
                } else {
jaroslav@115
   275
                    line = line.substring(1);
jaroslav@115
   276
                }
jaroslav@115
   277
                Date d = new Date();
jaroslav@115
   278
                try {
jaroslav@115
   279
                    if (when != null) {
jaroslav@115
   280
                        d = new Date(Date.parse(when));
jaroslav@115
   281
                    }
jaroslav@115
   282
                } catch (IllegalArgumentException ex) {
jaroslav@115
   283
                    LOG.warning("Unparseable date " + when + " in " + f);
jaroslav@115
   284
                }
jaroslav@115
   285
                g.comment(who, line, d);
jaroslav@115
   286
                who = null;
jaroslav@115
   287
                when = null;
jtulach@37
   288
                continue;
jtulach@37
   289
            }
jtulach@37
   290
            if (white == null || black == null) {
jtulach@37
   291
                throw new IOException("Missing white and black identification in " + f);
jtulach@37
   292
            }
jtulach@37
   293
            if (g == null) {
jaroslav@131
   294
                GameId id = new GameId(f.getName(), white, black, new Date(f.lastModified()), new Date(f.lastModified()), GameStatus.whiteMove, 0);
jaroslav@48
   295
                g = new Game(id);
jaroslav@48
   296
            }
jaroslav@114
   297
            int hash = line.indexOf('#');
jaroslav@114
   298
            if (hash >= 0) {
jaroslav@114
   299
                line = line.substring(0, hash);
jaroslav@114
   300
            }
jaroslav@48
   301
            if (line.equals("finish")) {
jaroslav@48
   302
                break;
jtulach@37
   303
            }
jtulach@37
   304
            String[] moves = line.split(" ");
jtulach@37
   305
            if (moves.length == 0) {
jtulach@37
   306
                continue;
jtulach@37
   307
            }
jtulach@37
   308
            if (moves.length > 2) {
jtulach@37
   309
                throw new IOException("Too much moves on line: " + line);
jtulach@37
   310
            }
jtulach@37
   311
            try {
jaroslav@114
   312
                if (!"...".equals(moves[0])) {
jaroslav@114
   313
                    g.apply(white, Move.valueOf(moves[0]), null);
jaroslav@114
   314
                }
jtulach@37
   315
                if (moves.length == 2) {
jtulach@79
   316
                    g.apply(black, Move.valueOf(moves[1]), null);
jtulach@37
   317
                }
jtulach@37
   318
            } catch (IllegalPositionException ex) {
jtulach@37
   319
                throw new IOException("Wrong move: " + ex.getMessage());
jtulach@37
   320
            }
jtulach@37
   321
        }
jtulach@37
   322
        if (g == null) {
jtulach@37
   323
            throw new IOException("No moves in " + f);
jtulach@37
   324
        }
jtulach@37
   325
        return g;
jtulach@36
   326
    }
jtulach@36
   327
jtulach@37
   328
    private void storeGame(Game g) throws IOException {
jtulach@37
   329
        dir.mkdirs();
jaroslav@48
   330
        File f = new File(dir, g.getId().getId());
jaroslav@115
   331
        storeGame(g, f);
jaroslav@115
   332
    }
jaroslav@115
   333
jaroslav@115
   334
    final void storeGame(Game g, File f) throws IOException {
jtulach@117
   335
        FileOutputStream os = new FileOutputStream(f);
jtulach@117
   336
        Writer w = new BufferedWriter(new OutputStreamWriter(os, "UTF-8"));
jtulach@117
   337
        PrintWriter pw = new PrintWriter(w);
jaroslav@48
   338
        pw.println("# white: " + g.getId().getWhite());
jaroslav@48
   339
        pw.println("# black: " + g.getId().getBlack());
jtulach@77
   340
        pw.println("# status: " + g.getId().getStatus());
jtulach@37
   341
        int cnt = 0;
jaroslav@115
   342
        boolean separate = true;
jaroslav@115
   343
        for (CommentedMove m : g.getMoves()) {
jaroslav@115
   344
            if (!separate && cnt % 2 == 1) {
jaroslav@115
   345
                pw.print("... ");
jaroslav@115
   346
            }
jaroslav@115
   347
            separate = true;
jaroslav@115
   348
            pw.print(m.getMove().toString());
jaroslav@115
   349
            List<Note> notes = m.getComments();
jaroslav@115
   350
            if (notes != null) {
jaroslav@115
   351
                separate = false;
jtulach@37
   352
                pw.println();
jaroslav@115
   353
                for (Note n : notes) {
jaroslav@115
   354
                    pw.print ("# ");
jaroslav@115
   355
                    pw.print(n.getWho());
jaroslav@115
   356
                    pw.print("@");
jaroslav@115
   357
                    pw.print(n.getWhen());
jaroslav@115
   358
                    pw.println(":");
jaroslav@115
   359
                    for (String l : n.getComment().split("\n")) {
jaroslav@115
   360
                        pw.print("# ");
jaroslav@115
   361
                        pw.println(l);
jaroslav@115
   362
                    }
jaroslav@115
   363
                }
jaroslav@115
   364
            }
jaroslav@115
   365
jaroslav@115
   366
            cnt++;
jaroslav@115
   367
jaroslav@115
   368
            if (separate) {
jaroslav@115
   369
                if (cnt % 2 == 0) {
jaroslav@115
   370
                    pw.println();
jaroslav@115
   371
                } else {
jaroslav@115
   372
                    pw.print(' ');
jaroslav@115
   373
                }
jtulach@37
   374
            }
jtulach@37
   375
        }
jtulach@37
   376
        pw.println();
jtulach@37
   377
        pw.println();
jtulach@37
   378
        pw.flush();
jtulach@37
   379
        pw.close();
jtulach@117
   380
        w.close();
jtulach@37
   381
    }
jtulach@36
   382
}