--- /dev/null
+/*
+ * Copyright (c) 1997, 2018 Oracle and/or its affiliates. All rights reserved.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License v. 2.0, which is available at
+ * http://www.eclipse.org/legal/epl-2.0.
+ *
+ * This Source Code may also be made available under the following Secondary
+ * Licenses when the conditions for such availability set forth in the
+ * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
+ * version 2 with the GNU Classpath Exception, which is available at
+ * https://www.gnu.org/software/classpath/license.html.
+ *
+ * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
+ */
+
+package com.sun.mail.mbox;
+
+import java.io.*;
+
+/**
+ * Update the Content-Length header in the message written to the stream.
+ */
+class ContentLengthUpdater extends FilterOutputStream {
+ private String contentLength;
+ private boolean inHeader = true;
+ private boolean sawContentLength = false;
+ private int lastb1 = -1, lastb2 = -1;
+ private StringBuilder line = new StringBuilder();
+
+ public ContentLengthUpdater(OutputStream os, long contentLength) {
+ super(os);
+ this.contentLength = "Content-Length: " + contentLength;
+ }
+
+ public void write(int b) throws IOException {
+ if (inHeader) {
+ String eol = "\n";
+ // First, determine if we're still in the header.
+ if (b == '\r') {
+ // if line terminator is CR
+ if (lastb1 == '\r') {
+ inHeader = false;
+ eol = "\r";
+ // else, if line terminator is CRLF
+ } else if (lastb1 == '\n' && lastb2 == '\r') {
+ inHeader = false;
+ eol = "\r\n";
+ }
+ // else, if line terminator is \n
+ } else if (b == '\n') {
+ if (lastb1 == '\n') {
+ inHeader = false;
+ eol = "\n";
+ }
+ }
+
+ // If we're no longer in the header, and we haven't seen
+ // a Content-Length header yet, it's time to put one out.
+ if (!inHeader && !sawContentLength) {
+ out.write(contentLength.getBytes("iso-8859-1"));
+ out.write(eol.getBytes("iso-8859-1"));
+ }
+
+ // If we have a full line, see if it's a Content-Length header.
+ if (b == '\r' || (b == '\n' && lastb1 != '\r')) {
+ if (line.toString().regionMatches(true, 0,
+ "content-length:", 0, 15)) {
+ // yup, got it
+ sawContentLength = true;
+ // put out the new version
+ out.write(contentLength.getBytes("iso-8859-1"));
+ } else {
+ // not a Content-Length header, just write it out
+ out.write(line.toString().getBytes("iso-8859-1"));
+ }
+ line.setLength(0); // clear buffer for next line
+ }
+ if (b == '\r' || b == '\n')
+ out.write(b); // write out line terminator immediately
+ else
+ line.append((char)b); // accumulate characters of the line
+
+ // rotate saved characters for next time through loop
+ lastb2 = lastb1;
+ lastb1 = b;
+ } else
+ out.write(b); // not in the header, just write it out
+ }
+
+ public void write(byte[] b) throws IOException {
+ if (inHeader)
+ write(b, 0, b.length);
+ else
+ out.write(b);
+ }
+
+ public void write(byte[] b, int off, int len) throws IOException {
+ if (inHeader) {
+ for (int i = 0 ; i < len ; i++) {
+ write(b[off + i]);
+ }
+ } else
+ out.write(b, off, len);
+ }
+
+ // for testing
+ public static void main(String argv[]) throws Exception {
+ int b;
+ ContentLengthUpdater os =
+ new ContentLengthUpdater(System.out, Long.parseLong(argv[0]));
+ while ((b = System.in.read()) >= 0)
+ os.write(b);
+ os.flush();
+ }
+}