All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ratpack.exec.internal.DefaultThrottle Maven / Gradle / Ivy

There is a newer version: 2.0.0-rc-1
Show newest version
/*
 * Copyright 2014 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package ratpack.exec.internal;

import ratpack.exec.Downstream;
import ratpack.exec.Promise;
import ratpack.exec.Throttle;

import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.atomic.AtomicInteger;

public class DefaultThrottle implements Throttle {

  private final int size;

  private final Queue queue = new ConcurrentLinkedQueue<>();
  private final AtomicInteger active = new AtomicInteger();
  private final AtomicInteger waiting = new AtomicInteger();

  public DefaultThrottle(int size) {
    if (size < 1) {
      throw new IllegalArgumentException("throttle size must be greater than 0");
    }
    this.size = size;
  }

  @Override
  public  Promise throttle(Promise promise) {
    return promise.transform(up -> down -> {
      waiting.incrementAndGet();
      if (active.getAndIncrement() < size) {
        waiting.decrementAndGet();
        up.connect(down);
      } else {
        active.decrementAndGet();
        Promise.>async(innerDown -> {
          queue.add(() -> innerDown.success(down));
          drain();
        })
          .then(up::connect);
      }
    })
      .transform(up -> down ->
        up.connect(new Downstream() {
          @Override
          public void success(T value) {
            post();
            down.success(value);
          }

          @Override
          public void error(Throwable throwable) {
            post();
            down.error(throwable);
          }

          @Override
          public void complete() {
            post();
            down.complete();
          }
        })
      );
  }

  private void post() {
    active.decrementAndGet();
    drain();
  }

  @Override
  public int getSize() {
    return size;
  }

  @Override
  public int getActive() {
    return active.get();
  }

  @Override
  public int getWaiting() {
    return waiting.get();
  }

  private void drain() {
    if (!queue.isEmpty()) {
      if (active.getAndIncrement() < size) {
        Runnable job = queue.poll();
        if (job == null) {
          active.decrementAndGet();
        } else {
          waiting.decrementAndGet();
          job.run();
        }
      } else {
        if (active.decrementAndGet() < size) {
          drain();
        }
      }
    }
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy