Response.java

1
package com.renomad.minum.web;
2
3
import com.renomad.minum.utils.FileUtils;
4
5
import java.io.ByteArrayOutputStream;
6
import java.io.IOException;
7
import java.io.RandomAccessFile;
8
import java.net.URI;
9
import java.nio.ByteBuffer;
10
import java.nio.channels.FileChannel;
11
import java.nio.charset.StandardCharsets;
12
import java.nio.file.Files;
13
import java.nio.file.Path;
14
import java.util.Arrays;
15
import java.util.HashMap;
16
import java.util.Map;
17
import java.util.Objects;
18
import java.util.zip.GZIPOutputStream;
19
20
import static com.renomad.minum.web.StatusLine.StatusCode.CODE_200_OK;
21
import static com.renomad.minum.web.StatusLine.StatusCode.CODE_206_PARTIAL_CONTENT;
22
23
/**
24
 * Represents an HTTP response. This is what will get sent back to the
25
 * client (that is, to the browser).  There are a variety of overloads
26
 * of this record for different situations.  The overarching paradigm is
27
 * to provide you high flexibility.
28
 * <p>
29
 * A response message is sent by a server to a client as a reply to its former request message.
30
 * </p>
31
 * <h3>
32
 * Response syntax
33
 * </h3>
34
 * <p>
35
 * A server sends response messages to the client, which consist of:
36
 * </p>
37
 * <ul>
38
 * <li>
39
 * a status line, consisting of the protocol version, a space, the
40
 * response status code, another space, a possibly empty reason
41
 * phrase, a carriage return and a line feed, e.g.:
42
 * <pre>
43
 * HTTP/1.1 200 OK
44
 * </pre>
45
 * </li>
46
 *
47
 * <li>
48
 * zero or more response header fields, each consisting of the case-insensitive
49
 * field name, a colon, optional leading whitespace, the field value, an
50
 * optional trailing whitespace and ending with a carriage return and a line feed, e.g.:
51
 * <pre>
52
 * Content-Type: text/html
53
 * </pre>
54
 * </li>
55
 *
56
 * <li>
57
 * an empty line, consisting of a carriage return and a line feed;
58
 * </li>
59
 *
60
 * <li>
61
 * an optional message body.
62
 * </li>
63
 *</ul>
64
65
 */
66
public final class Response implements IResponse {
67
68
    private final StatusLine.StatusCode statusCode;
69
    private final Map<String, String> extraHeaders;
70
    private final byte[] body;
71
    private final ThrowingConsumer<ISocketWrapper> outputGenerator;
72
    private final long bodyLength;
73
74
75
    /**
76
     * This is the constructor that provides access to all fields.  It is not intended
77
     * to be used from outside this class.
78
     * @param extraHeaders extra headers we want to return with the response.
79
     * @param outputGenerator a {@link ThrowingConsumer} that will use a {@link ISocketWrapper} parameter
80
     *                        to send bytes on the wire back to the client.  See the static factory methods
81
     *                        such as {@link #buildResponse(StatusLine.StatusCode, Map, byte[])} for more details on this.
82
     * @param bodyLength this is used to set the content-length header for the response.  If this is
83
     *                   not provided, we set the header to "transfer-encoding: chunked", or in other words, streaming.
84
     */
85
    Response(StatusLine.StatusCode statusCode, Map<String, String> extraHeaders, byte[] body,
86
             ThrowingConsumer<ISocketWrapper> outputGenerator, long bodyLength) {
87
        this.statusCode = statusCode;
88
        this.extraHeaders = new HashMap<>(extraHeaders);
89
        this.body = body;
90
        this.outputGenerator = outputGenerator;
91
        this.bodyLength = bodyLength;
92
    }
93
94
    /**
95
     * This factory method is intended for situations where the user wishes to stream data
96
     * but lacks the content length.  This is only for unusual situations where the developer
97
     * needs the extra control.  In most cases, other methods are more suitable.
98
     * @param extraHeaders any extra headers for the response, such as the content-type
99
     * @param outputGenerator a function that will be given a {@link ISocketWrapper}, providing the
100
     *                        ability to send bytes on the socket.
101
     */
102
    public static IResponse buildStreamingResponse(StatusLine.StatusCode statusCode, Map<String, String> extraHeaders, ThrowingConsumer<ISocketWrapper> outputGenerator) {
103 1 1. buildStreamingResponse : replaced return value with null for com/renomad/minum/web/Response::buildStreamingResponse → KILLED
        return new Response(statusCode, extraHeaders, null, outputGenerator, 0);
104
    }
105
106
    /**
107
     * Similar to {@link Response#buildStreamingResponse(StatusLine.StatusCode, Map, ThrowingConsumer)} but here we know
108
     * the body length, so that will be sent to the client as content-length.
109
     * @param extraHeaders any extra headers for the response, such as the content-type
110
     * @param outputGenerator a function that will be given a {@link ISocketWrapper}, providing the
111
     *                        ability to send bytes on the socket.
112
     * @param bodyLength the length, in bytes, of the data to be sent to the client
113
     */
114
    public static IResponse buildStreamingResponse(StatusLine.StatusCode statusCode, Map<String, String> extraHeaders, ThrowingConsumer<ISocketWrapper> outputGenerator, long bodyLength) {
115 1 1. buildStreamingResponse : replaced return value with null for com/renomad/minum/web/Response::buildStreamingResponse → KILLED
        return new Response(statusCode, extraHeaders, null, outputGenerator, bodyLength);
116
    }
117
118
    /**
119
     * A constructor for situations where the developer wishes to send a small (less than a megabyte) byte array
120
     * to the client.  If there is need to send something of larger size, choose one these
121
     * alternate constructors:
122
     * FileChannel - for sending a large file: {@link Response#buildLargeFileResponse(Map, String, Headers)}
123
     * Streaming - for more custom streaming control with a known body size: {@link Response#buildStreamingResponse(StatusLine.StatusCode, Map, ThrowingConsumer, long)}
124
     * Streaming - for more custom streaming control with body size unknown: {@link Response#buildStreamingResponse(StatusLine.StatusCode, Map, ThrowingConsumer)}
125
     *
126
     * @param extraHeaders any extra headers for the response, such as the content-type.
127
     */
128
    public static IResponse buildResponse(StatusLine.StatusCode statusCode, Map<String, String> extraHeaders, byte[] body) {
129 2 1. lambda$buildResponse$0 : removed call to com/renomad/minum/web/Response::sendByteArrayResponse → TIMED_OUT
2. buildResponse : replaced return value with null for com/renomad/minum/web/Response::buildResponse → KILLED
        return new Response(statusCode, extraHeaders, body, socketWrapper -> sendByteArrayResponse(socketWrapper, body), body.length);
130
    }
131
132
    /**
133
     * Build an ordinary response, with a known body
134
     * @param extraHeaders extra HTTP headers, like <pre>content-type: text/html</pre>
135
     */
136
    public static IResponse buildResponse(StatusLine.StatusCode statusCode, Map<String, String> extraHeaders, String body) {
137
        byte[] bytes = body.getBytes(StandardCharsets.UTF_8);
138 2 1. lambda$buildResponse$1 : removed call to com/renomad/minum/web/Response::sendByteArrayResponse → KILLED
2. buildResponse : replaced return value with null for com/renomad/minum/web/Response::buildResponse → KILLED
        return new Response(statusCode, extraHeaders, bytes, socketWrapper -> sendByteArrayResponse(socketWrapper, bytes), bytes.length);
139
    }
140
141
    /**
142
     * This will send a large file to the client as a stream.
143
     * <em>Note that this does not check that the requested file is within a directory. If you expect the
144
     * path value to be untrusted - that is, set by a user - use {@link #buildLargeFileResponse(Map, String, String, Headers)}</em>
145
     * @param extraHeaders extra headers you would like in the response, as key-value pairs in a map - for
146
     *                     example "Content-Type: video/mp4" or "Content-Type: application/zip"
147
     * @param filePath the string value of the path to this file. <em>Caution! if this is set by the user.  See notes above</em>
148
     * @param requestHeaders these are the headers from the request, which is needed here to set proper
149
     *                       response headers in some cases.
150
     */
151
    public static IResponse buildLargeFileResponse(Map<String, String> extraHeaders, String filePath, Headers requestHeaders) throws IOException {
152
        Map<String, String> adjustedHeaders = new HashMap<>(extraHeaders);
153
        long fileSize = Files.size(Path.of(filePath));
154
        var range = new Range(requestHeaders, fileSize);
155
        StatusLine.StatusCode responseCode = CODE_200_OK;
156
        long length = fileSize;
157 1 1. buildLargeFileResponse : negated conditional → KILLED
        if (range.hasRangeHeader()) {
158
            long offset = range.getOffset();
159
            length = range.getLength();
160 2 1. buildLargeFileResponse : Replaced long subtraction with addition → SURVIVED
2. buildLargeFileResponse : Replaced long addition with subtraction → TIMED_OUT
            var lastIndex = (offset + length) - 1;
161
            adjustedHeaders.put("Content-Range", String.format("bytes %d-%d/%d", offset, lastIndex, fileSize));
162
            responseCode = CODE_206_PARTIAL_CONTENT;
163
        }
164
165
        ThrowingConsumer<ISocketWrapper> outputGenerator = socketWrapper -> {
166
            try (RandomAccessFile reader = new RandomAccessFile(filePath, "r")) {
167 1 1. lambda$buildLargeFileResponse$2 : removed call to java/io/RandomAccessFile::seek → KILLED
                reader.seek(range.getOffset());
168
                var fileChannel = reader.getChannel();
169 1 1. lambda$buildLargeFileResponse$2 : removed call to com/renomad/minum/web/Response::sendFileChannelResponse → KILLED
                sendFileChannelResponse(socketWrapper, fileChannel, range.getLength());
170
            }
171
        };
172
173 1 1. buildLargeFileResponse : replaced return value with null for com/renomad/minum/web/Response::buildLargeFileResponse → TIMED_OUT
        return new Response(responseCode, adjustedHeaders, null, outputGenerator, length);
174
    }
175
176
    /**
177
     * This will send a large file to the client as a stream.
178
     * This is a safe version of the method, for when the file path is set by a user, meaning it is untrusted.
179
     * By setting a parentDirectory, the system will ensure that the requested path is within that directory, thus
180
     * disallowing path strings that could escape it, like prefixing with a slash.
181
     * @param extraHeaders extra headers you would like in the response, as key-value pairs in a map - for
182
     *                     example "Content-Type: video/mp4" or "Content-Type: application/zip"
183
     * @param filePath the string value of the path to this file. <em>Caution! if this is set by the user.  See notes above</em>
184
     * @param requestHeaders these are the headers from the request, which is needed here to set proper
185
     *                       response headers in some cases.
186
     * @param parentDirectory this value is presumed to be set by the developer, and thus isn't checked for safety. This
187
     *                        allows the developer to use directories anywhere in the system.
188
     */
189
    public static IResponse buildLargeFileResponse(Map<String, String> extraHeaders, String filePath, String parentDirectory, Headers requestHeaders) throws IOException {
190
        Path path = FileUtils.safeResolve(parentDirectory, filePath);
191 1 1. buildLargeFileResponse : replaced return value with null for com/renomad/minum/web/Response::buildLargeFileResponse → SURVIVED
        return buildLargeFileResponse(extraHeaders, path.toString(), requestHeaders);
192
    }
193
194
    /**
195
     * Build a {@link Response} with just a status code and headers, without a body
196
     * @param extraHeaders extra HTTP headers
197
     */
198
    public static IResponse buildLeanResponse(StatusLine.StatusCode statusCode, Map<String, String> extraHeaders) {
199 1 1. buildLeanResponse : replaced return value with null for com/renomad/minum/web/Response::buildLeanResponse → KILLED
        return new Response(statusCode, extraHeaders, null, socketWrapper -> {}, 0);
200
    }
201
202
    /**
203
     * Build a {@link Response} with only a status code, with no body and no extra headers.
204
     */
205
    public static IResponse buildLeanResponse(StatusLine.StatusCode statusCode) {
206 1 1. buildLeanResponse : replaced return value with null for com/renomad/minum/web/Response::buildLeanResponse → KILLED
        return new Response(statusCode, Map.of(), null, socketWrapper -> {}, 0);
207
    }
208
209
    /**
210
     * A helper method to create a response that returns a
211
     * 303 status code ("see other").  Provide a url that will
212
     * be handed to the browser.  This url may be relative or absolute.
213
     */
214
    public static IResponse redirectTo(String locationUrl) {
215
        try {
216
            // check if we can create a URL from this - any failures indicate invalid syntax.
217
            URI.create(locationUrl);
218
        } catch (Exception ex) {
219
            throw new WebServerException("Failure in redirect to (" + locationUrl + "). Exception: " + ex);
220
        }
221 1 1. redirectTo : replaced return value with null for com/renomad/minum/web/Response::redirectTo → KILLED
        return buildResponse(StatusLine.StatusCode.CODE_303_SEE_OTHER, Map.of("location", locationUrl, "Content-Type", "text/html; charset=UTF-8"), "<p>See <a href=\""+locationUrl+"\">this link</a></p>");
222
    }
223
224
    /**
225
     * If you are returning HTML text with a 200 ok, this is a helper that
226
     * lets you skip some of the boilerplate. This version of the helper
227
     * lets you add extra headers on top of the basic content-type headers
228
     * that are needed to specify this is HTML.
229
     */
230
    public static IResponse htmlOk(String body, Map<String, String> extraHeaders) {
231
        var headers = new HashMap<String, String>();
232
        headers.put("Content-Type", "text/html; charset=UTF-8");
233 1 1. htmlOk : removed call to java/util/HashMap::putAll → KILLED
        headers.putAll(extraHeaders);
234 1 1. htmlOk : replaced return value with null for com/renomad/minum/web/Response::htmlOk → KILLED
        return buildResponse(StatusLine.StatusCode.CODE_200_OK, headers, body);
235
    }
236
237
    /**
238
     * If you are returning HTML text with a 200 ok, this is a helper that
239
     * lets you skip some of the boilerplate.
240
     */
241
    public static IResponse htmlOk(String body) {
242 1 1. htmlOk : replaced return value with null for com/renomad/minum/web/Response::htmlOk → KILLED
        return htmlOk(body, Map.of());
243
    }
244
245
    @Override
246
    public Map<String, String> getExtraHeaders() {
247 1 1. getExtraHeaders : replaced return value with Collections.emptyMap for com/renomad/minum/web/Response::getExtraHeaders → KILLED
        return new HashMap<>(extraHeaders);
248
    }
249
250
    @Override
251
    public StatusLine.StatusCode getStatusCode() {
252 1 1. getStatusCode : replaced return value with null for com/renomad/minum/web/Response::getStatusCode → KILLED
        return statusCode;
253
    }
254
255
    /**
256
     * Gets the length of the body for this response.  If the body
257
     * is an array of bytes set by the user, we grab this value by the
258
     * length() method.  If the outgoing data is set by a lambda, the user
259
     * will set the bodyLength value.
260
     */
261
    long getBodyLength() {
262 1 1. getBodyLength : negated conditional → KILLED
        if (body != null) {
263 1 1. getBodyLength : replaced long return with 0 for com/renomad/minum/web/Response::getBodyLength → KILLED
            return body.length;
264
        } else {
265 1 1. getBodyLength : replaced long return with 0 for com/renomad/minum/web/Response::getBodyLength → KILLED
            return bodyLength;
266
        }
267
    }
268
269
    /**
270
     * By calling this method with a {@link ISocketWrapper} parameter, the method
271
     * will send bytes on the associated socket.
272
     */
273
    void sendBody(ISocketWrapper sw) throws IOException {
274
        try {
275 1 1. sendBody : removed call to com/renomad/minum/web/ThrowingConsumer::accept → KILLED
            outputGenerator.accept(sw);
276
        } catch (Exception ex) {
277
            throw new IOException(ex.getMessage(), ex);
278
        }
279
    }
280
281
    /**
282
     * put bytes from a file into the socket, sending to the client
283
     * @param fileChannel the file we are reading from, based on a {@link RandomAccessFile}
284
     * @param length the number of bytes to send.  May be less than the full length of this {@link FileChannel}
285
     */
286
    private static void sendFileChannelResponse(ISocketWrapper sw, FileChannel fileChannel, long length) throws IOException {
287
        try {
288
            int bufferSize = 8 * 1024;
289
            ByteBuffer buff = ByteBuffer.allocate(bufferSize);
290
            long countBytesLeftToSend = length;
291
            while (true) {
292
                int countBytesRead = fileChannel.read(buff);
293 2 1. sendFileChannelResponse : changed conditional boundary → SURVIVED
2. sendFileChannelResponse : negated conditional → TIMED_OUT
                if (countBytesRead <= 0) {
294
                    break;
295
                } else {
296 2 1. sendFileChannelResponse : negated conditional → KILLED
2. sendFileChannelResponse : changed conditional boundary → KILLED
                    if (countBytesLeftToSend < countBytesRead) {
297 1 1. sendFileChannelResponse : removed call to com/renomad/minum/web/ISocketWrapper::send → TIMED_OUT
                        sw.send(buff.array(), 0, (int)countBytesLeftToSend);
298
                        break;
299
                    } else {
300 1 1. sendFileChannelResponse : removed call to com/renomad/minum/web/ISocketWrapper::send → KILLED
                        sw.send(buff.array(), 0, countBytesRead);
301
                    }
302
                    buff.clear();
303
                }
304 1 1. sendFileChannelResponse : Replaced long subtraction with addition → KILLED
                countBytesLeftToSend -= countBytesRead;
305
            }
306
        } finally {
307 1 1. sendFileChannelResponse : removed call to java/nio/channels/FileChannel::close → SURVIVED
            fileChannel.close();
308
        }
309
    }
310
311
    private static void sendByteArrayResponse(ISocketWrapper sw, byte[] body) throws IOException {
312 1 1. sendByteArrayResponse : removed call to com/renomad/minum/web/ISocketWrapper::send → KILLED
        sw.send(body);
313
    }
314
315
    @Override
316
    public byte[] getBody() {
317 1 1. getBody : replaced return value with null for com/renomad/minum/web/Response::getBody → KILLED
        return body;
318
    }
319
320
    /**
321
     * Compress the data in this body using gzip.
322
     * <br>
323
     * This operates by getting the body field from this instance of {@link Response} and
324
     * creating a new Response with the compressed data.
325
     */
326
    Response compressBody() throws IOException {
327
328
        ByteArrayOutputStream out = new ByteArrayOutputStream();
329
        var gos = new GZIPOutputStream(out);
330 1 1. compressBody : removed call to java/util/zip/GZIPOutputStream::write → SURVIVED
        gos.write(body);
331 1 1. compressBody : removed call to java/util/zip/GZIPOutputStream::finish → SURVIVED
        gos.finish();
332 1 1. compressBody : replaced return value with null for com/renomad/minum/web/Response::compressBody → KILLED
        return (Response)Response.buildResponse(
333
                statusCode,
334
                extraHeaders,
335
                out.toByteArray()
336
        );
337
    }
338
339
    @Override
340
    public boolean equals(Object o) {
341 2 1. equals : replaced boolean return with false for com/renomad/minum/web/Response::equals → KILLED
2. equals : negated conditional → KILLED
        if (this == o) return true;
342 3 1. equals : negated conditional → KILLED
2. equals : replaced boolean return with true for com/renomad/minum/web/Response::equals → KILLED
3. equals : negated conditional → KILLED
        if (o == null || getClass() != o.getClass()) return false;
343
        Response response = (Response) o;
344 5 1. equals : negated conditional → KILLED
2. equals : replaced boolean return with true for com/renomad/minum/web/Response::equals → KILLED
3. equals : negated conditional → KILLED
4. equals : negated conditional → KILLED
5. equals : negated conditional → KILLED
        return bodyLength == response.bodyLength && statusCode == response.statusCode && Objects.equals(extraHeaders, response.extraHeaders) && Arrays.equals(body, response.body);
345
    }
346
347
    @Override
348
    public int hashCode() {
349
        int result = Objects.hash(statusCode, extraHeaders, bodyLength);
350 2 1. hashCode : Replaced integer addition with subtraction → TIMED_OUT
2. hashCode : Replaced integer multiplication with division → KILLED
        result = 31 * result + Arrays.hashCode(body);
351 1 1. hashCode : replaced int return with 0 for com/renomad/minum/web/Response::hashCode → KILLED
        return result;
352
    }
353
354
    @Override
355
    public String toString() {
356 1 1. toString : replaced return value with "" for com/renomad/minum/web/Response::toString → KILLED
        return "Response{" +
357
                "statusCode=" + statusCode +
358
                ", extraHeaders=" + extraHeaders +
359
                ", bodyLength=" + bodyLength +
360
                '}';
361
    }
362
}

Mutations

103

1.1
Location : buildStreamingResponse
Killed by : com.renomad.minum.web.ResponseTests.testResponse_Streaming(com.renomad.minum.web.ResponseTests)
replaced return value with null for com/renomad/minum/web/Response::buildStreamingResponse → KILLED

115

1.1
Location : buildStreamingResponse
Killed by : com.renomad.minum.web.WebTests
replaced return value with null for com/renomad/minum/web/Response::buildStreamingResponse → KILLED

129

1.1
Location : lambda$buildResponse$0
Killed by : none
removed call to com/renomad/minum/web/Response::sendByteArrayResponse → TIMED_OUT

2.2
Location : buildResponse
Killed by : com.renomad.minum.web.WebFrameworkTests.test_compressIfRequested(com.renomad.minum.web.WebFrameworkTests)
replaced return value with null for com/renomad/minum/web/Response::buildResponse → KILLED

138

1.1
Location : lambda$buildResponse$1
Killed by : com.renomad.minum.web.ResponseTests.testResponse_EdgeCase_SendBodyWithException(com.renomad.minum.web.ResponseTests)
removed call to com/renomad/minum/web/Response::sendByteArrayResponse → KILLED

2.2
Location : buildResponse
Killed by : com.renomad.minum.web.ResponseTests.testToString(com.renomad.minum.web.ResponseTests)
replaced return value with null for com/renomad/minum/web/Response::buildResponse → KILLED

157

1.1
Location : buildLargeFileResponse
Killed by : com.renomad.minum.FunctionalTests
negated conditional → KILLED

160

1.1
Location : buildLargeFileResponse
Killed by : none
Replaced long addition with subtraction → TIMED_OUT

2.2
Location : buildLargeFileResponse
Killed by : none
Replaced long subtraction with addition → SURVIVED
Covering tests

167

1.1
Location : lambda$buildLargeFileResponse$2
Killed by : com.renomad.minum.FunctionalTests
removed call to java/io/RandomAccessFile::seek → KILLED

169

1.1
Location : lambda$buildLargeFileResponse$2
Killed by : com.renomad.minum.FunctionalTests
removed call to com/renomad/minum/web/Response::sendFileChannelResponse → KILLED

173

1.1
Location : buildLargeFileResponse
Killed by : none
replaced return value with null for com/renomad/minum/web/Response::buildLargeFileResponse → TIMED_OUT

191

1.1
Location : buildLargeFileResponse
Killed by : none
replaced return value with null for com/renomad/minum/web/Response::buildLargeFileResponse → SURVIVED
Covering tests

199

1.1
Location : buildLeanResponse
Killed by : com.renomad.minum.FunctionalTests
replaced return value with null for com/renomad/minum/web/Response::buildLeanResponse → KILLED

206

1.1
Location : buildLeanResponse
Killed by : com.renomad.minum.web.WebFrameworkTests.test_readStaticFile_Edge_Colon(com.renomad.minum.web.WebFrameworkTests)
replaced return value with null for com/renomad/minum/web/Response::buildLeanResponse → KILLED

221

1.1
Location : redirectTo
Killed by : com.renomad.minum.FunctionalTests
replaced return value with null for com/renomad/minum/web/Response::redirectTo → KILLED

233

1.1
Location : htmlOk
Killed by : com.renomad.minum.web.WebTests
removed call to java/util/HashMap::putAll → KILLED

234

1.1
Location : htmlOk
Killed by : com.renomad.minum.web.ResponseTests.testToString(com.renomad.minum.web.ResponseTests)
replaced return value with null for com/renomad/minum/web/Response::htmlOk → KILLED

242

1.1
Location : htmlOk
Killed by : com.renomad.minum.web.ResponseTests.testToString(com.renomad.minum.web.ResponseTests)
replaced return value with null for com/renomad/minum/web/Response::htmlOk → KILLED

247

1.1
Location : getExtraHeaders
Killed by : com.renomad.minum.web.WebFrameworkTests.test_Edge_ApplicationOctetStream(com.renomad.minum.web.WebFrameworkTests)
replaced return value with Collections.emptyMap for com/renomad/minum/web/Response::getExtraHeaders → KILLED

252

1.1
Location : getStatusCode
Killed by : com.renomad.minum.web.WebFrameworkTests.test_readStaticFile_Edge_Colon(com.renomad.minum.web.WebFrameworkTests)
replaced return value with null for com/renomad/minum/web/Response::getStatusCode → KILLED

262

1.1
Location : getBodyLength
Killed by : com.renomad.minum.web.WebTests
negated conditional → KILLED

263

1.1
Location : getBodyLength
Killed by : com.renomad.minum.web.WebFrameworkTests.test_compressIfRequested(com.renomad.minum.web.WebFrameworkTests)
replaced long return with 0 for com/renomad/minum/web/Response::getBodyLength → KILLED

265

1.1
Location : getBodyLength
Killed by : com.renomad.minum.web.WebTests
replaced long return with 0 for com/renomad/minum/web/Response::getBodyLength → KILLED

275

1.1
Location : sendBody
Killed by : com.renomad.minum.web.ResponseTests.testResponse_Streaming(com.renomad.minum.web.ResponseTests)
removed call to com/renomad/minum/web/ThrowingConsumer::accept → KILLED

293

1.1
Location : sendFileChannelResponse
Killed by : none
changed conditional boundary → SURVIVED
Covering tests

2.2
Location : sendFileChannelResponse
Killed by : none
negated conditional → TIMED_OUT

296

1.1
Location : sendFileChannelResponse
Killed by : com.renomad.minum.FunctionalTests
negated conditional → KILLED

2.2
Location : sendFileChannelResponse
Killed by : com.renomad.minum.FunctionalTests
changed conditional boundary → KILLED

297

1.1
Location : sendFileChannelResponse
Killed by : none
removed call to com/renomad/minum/web/ISocketWrapper::send → TIMED_OUT

300

1.1
Location : sendFileChannelResponse
Killed by : com.renomad.minum.FunctionalTests
removed call to com/renomad/minum/web/ISocketWrapper::send → KILLED

304

1.1
Location : sendFileChannelResponse
Killed by : com.renomad.minum.FunctionalTests
Replaced long subtraction with addition → KILLED

307

1.1
Location : sendFileChannelResponse
Killed by : none
removed call to java/nio/channels/FileChannel::close → SURVIVED
Covering tests

312

1.1
Location : sendByteArrayResponse
Killed by : com.renomad.minum.web.ResponseTests.testResponse_EdgeCase_SendBodyWithException(com.renomad.minum.web.ResponseTests)
removed call to com/renomad/minum/web/ISocketWrapper::send → KILLED

317

1.1
Location : getBody
Killed by : com.renomad.minum.web.EndpointTests.test_Endpoint_HappyPath(com.renomad.minum.web.EndpointTests)
replaced return value with null for com/renomad/minum/web/Response::getBody → KILLED

330

1.1
Location : compressBody
Killed by : none
removed call to java/util/zip/GZIPOutputStream::write → SURVIVED
Covering tests

331

1.1
Location : compressBody
Killed by : none
removed call to java/util/zip/GZIPOutputStream::finish → SURVIVED
Covering tests

332

1.1
Location : compressBody
Killed by : com.renomad.minum.web.WebFrameworkTests.test_compressIfRequested(com.renomad.minum.web.WebFrameworkTests)
replaced return value with null for com/renomad/minum/web/Response::compressBody → KILLED

341

1.1
Location : equals
Killed by : com.renomad.minum.EqualsTests.equalsTest(com.renomad.minum.EqualsTests)
replaced boolean return with false for com/renomad/minum/web/Response::equals → KILLED

2.2
Location : equals
Killed by : com.renomad.minum.EqualsTests.equalsTest(com.renomad.minum.EqualsTests)
negated conditional → KILLED

342

1.1
Location : equals
Killed by : com.renomad.minum.web.ResponseTests.testUseResponseAsKey(com.renomad.minum.web.ResponseTests)
negated conditional → KILLED

2.2
Location : equals
Killed by : com.renomad.minum.EqualsTests.equalsTest(com.renomad.minum.EqualsTests)
replaced boolean return with true for com/renomad/minum/web/Response::equals → KILLED

3.3
Location : equals
Killed by : com.renomad.minum.web.ResponseTests.testUseResponseAsKey(com.renomad.minum.web.ResponseTests)
negated conditional → KILLED

344

1.1
Location : equals
Killed by : com.renomad.minum.web.ResponseTests.testUseResponseAsKey(com.renomad.minum.web.ResponseTests)
negated conditional → KILLED

2.2
Location : equals
Killed by : com.renomad.minum.EqualsTests.equalsTest(com.renomad.minum.EqualsTests)
replaced boolean return with true for com/renomad/minum/web/Response::equals → KILLED

3.3
Location : equals
Killed by : com.renomad.minum.web.ResponseTests.testUseResponseAsKey(com.renomad.minum.web.ResponseTests)
negated conditional → KILLED

4.4
Location : equals
Killed by : com.renomad.minum.web.ResponseTests.testUseResponseAsKey(com.renomad.minum.web.ResponseTests)
negated conditional → KILLED

5.5
Location : equals
Killed by : com.renomad.minum.web.ResponseTests.testUseResponseAsKey(com.renomad.minum.web.ResponseTests)
negated conditional → KILLED

350

1.1
Location : hashCode
Killed by : none
Replaced integer addition with subtraction → TIMED_OUT

2.2
Location : hashCode
Killed by : com.renomad.minum.EqualsTests.equalsTest(com.renomad.minum.EqualsTests)
Replaced integer multiplication with division → KILLED

351

1.1
Location : hashCode
Killed by : com.renomad.minum.EqualsTests.equalsTest(com.renomad.minum.EqualsTests)
replaced int return with 0 for com/renomad/minum/web/Response::hashCode → KILLED

356

1.1
Location : toString
Killed by : com.renomad.minum.web.ResponseTests.testToString(com.renomad.minum.web.ResponseTests)
replaced return value with "" for com/renomad/minum/web/Response::toString → KILLED

Active mutators

Tests examined


Report generated by PIT 1.17.0