LimitedInputStreamEngine.java

/*
 * GovWay - A customizable API Gateway 
 * https://govway.org
 * 
 * Copyright (c) 2005-2025 Link.it srl (https://link.it). 
 * 
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 3, as published by
 * the Free Software Foundation.
 *
 * This program 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 *
 */

package org.openspcoop2.utils;

import java.io.FilterInputStream;
import java.io.IOException;
import java.io.InputStream;

/**
 * LimitedInputStream
 *
 *
 * @author Poli Andrea (apoli@link.it)
 * @author $Author$
 * @version $Rev$, $Date$
 */
public class LimitedInputStreamEngine extends FilterInputStream {

	private long limitBytes;
	private long count;
	
	private InputStream isWrapped = null;
	private String prefixError = "";
	private Map<Object> ctx;
	private boolean checkDisabled = false;

	private ILimitExceededNotifier notifier;
	
	protected LimitedInputStreamEngine(InputStream inputStream, long limitBytes, String prefixError, Map<Object> ctx, ILimitExceededNotifier notifier) throws IOException {
		super(inputStream);
		this.limitBytes = limitBytes;
		
		this.isWrapped = inputStream;
		if(prefixError!=null) {
			this.prefixError = prefixError;
		}
		this.ctx = ctx;
		if(this.limitBytes<=0) {
			throw new IOException("Invalid limit");
		}
		
		this.notifier = notifier;
	}
	
	public InputStream getIsWrapped() {
		return this.isWrapped;
	}

	protected void disableCheck() {
		this.checkDisabled = true;
	}
	protected void updateThreshold(long limitBytes) throws IOException {
		if(this.limitBytes<=0) {
			throw new IOException("Invalid limit");
		}
		this.limitBytes = limitBytes;
	}
	protected void updateContext(Map<Object> ctx) {
		this.ctx = ctx;
	}
	protected void updateNotifier(ILimitExceededNotifier notifier) {
		this.notifier = notifier;
	}

	private void checkLimit() throws IOException {
		if(this.checkDisabled) {
			return; // e' stato disabilitato dopo averlo creato
		}
		if (this.count > this.limitBytes) {
			
			/**System.out.println("Raggiunto limite dopo aver letto: "+this.count);*/
			
			payloadTooLarge(this.prefixError, this.ctx, this.notifier, this.count);
			
		}
	}
	public static void payloadTooLarge(String prefixError, Map<Object> ctx, ILimitExceededNotifier notifier, long count) throws LimitExceededIOException {
		limitExceeded(prefixError, LimitedInputStream.ERROR_PAYLOAD_TOO_LARGE_MSG, false, ctx, notifier, count);
	}
	public static void contentLenghtLimitExceeded(String prefixError, Map<Object> ctx, ILimitExceededNotifier notifier, long count) throws LimitExceededIOException {
		limitExceeded(prefixError, LimitedInputStream.ERROR_CONTENT_LENGTH_EXCEEDED_MSG, true, ctx, notifier, count);
	}
	private static void limitExceeded(String prefixError, String error, boolean contentLengthExceeded, Map<Object> ctx, ILimitExceededNotifier notifier, long count) throws LimitExceededIOException {
		String errorMsg = prefixError+error;
		if(ctx!=null) {
			ctx.put(LimitedInputStream.ERROR_MSG_KEY, errorMsg);
		}
		LimitExceededIOException exc = new LimitExceededIOException(errorMsg);
		if(ctx!=null) {
			ctx.put(LimitedInputStream.EXCEPTION_KEY, exc);
		}
		
		if(notifier!=null) {
			notifier.notify(count, contentLengthExceeded);
		}
		
		throw exc;
	}

	@Override
	public int read() throws IOException {
		int res = super.read();
		if (res != -1) {
			this.count++;
			checkLimit();
		}
		return res;
	}

	@Override
	public int read(byte[] b, int off, int len) throws IOException {
		int res = super.read(b, off, len);
		if (res > 0) {
			this.count += res;
			checkLimit();
		}
		return res;
	}

}