com.gopivotal.cla.github.RateLimitingClientHttpRequestInterceptor.java Source code

Java tutorial

Introduction

Here is the source code for com.gopivotal.cla.github.RateLimitingClientHttpRequestInterceptor.java

Source

/*
 * Copyright 2013 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 com.gopivotal.cla.github;

import java.io.IOException;
import java.util.Date;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.http.HttpRequest;
import org.springframework.http.client.ClientHttpRequestExecution;
import org.springframework.http.client.ClientHttpRequestInterceptor;
import org.springframework.http.client.ClientHttpResponse;
import org.springframework.stereotype.Component;

@Component
final class RateLimitingClientHttpRequestInterceptor implements ClientHttpRequestInterceptor {

    private static final int RATE_LIMIT_BUFFER = 100;

    private static final String REMAINING = "X-RateLimit-Remaining";

    private static final String RESET = "X-RateLimit-Reset";

    private final Logger logger = LoggerFactory.getLogger(this.getClass());

    private final Object monitor = new Object();

    private volatile long reset = 0;

    @Override
    public ClientHttpResponse intercept(HttpRequest request, byte[] body, ClientHttpRequestExecution execution)
            throws IOException {
        checkRateLimit(request);
        ClientHttpResponse response = execution.execute(request, body);
        updateRateLimit(response);

        return response;
    }

    private void checkRateLimit(HttpRequest request) {
        synchronized (this.monitor) {
            try {
                while (this.reset > System.currentTimeMillis()) {
                    this.logger.warn("Request '{}' blocked by rate limit until {}", getRequestString(request),
                            new Date(this.reset));
                    this.monitor.wait();
                }
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                this.logger.warn("Request '{}' interrupted waiting for rate limit to expire",
                        getRequestString(request));
                throw new RuntimeException(String.format(
                        "Request '%s' interrupted waiting for rate limit to expire", getRequestString(request)), e);
            }
        }
    }

    private void updateRateLimit(ClientHttpResponse response) {
        synchronized (this.monitor) {
            if (getRemaining(response) < RATE_LIMIT_BUFFER) {
                this.reset = getReset(response);
                new Thread(new RateLimitNotifier(this.reset, this.monitor),
                        "Rate limit notifier " + new Date(this.reset)).start();
            }
        }
    }

    private int getRemaining(ClientHttpResponse response) {
        String remaining = response.getHeaders().getFirst(REMAINING);
        int parsedRemaining = remaining != null ? Integer.parseInt(remaining) : Integer.MAX_VALUE;

        this.logger.debug("{} requests remaining before rate limit", parsedRemaining);

        return parsedRemaining;
    }

    private String getRequestString(HttpRequest request) {
        return String.format("%s %s", request.getMethod(), request.getURI());
    }

    private long getReset(ClientHttpResponse response) {
        String reset = response.getHeaders().getFirst(RESET);
        return reset != null ? Long.parseLong(reset) * 1000 : System.currentTimeMillis();
    }

    private static final class RateLimitNotifier implements Runnable {

        private final Logger logger = LoggerFactory.getLogger(this.getClass());

        private final long reset;

        private final Object monitor;

        private RateLimitNotifier(long reset, Object monitor) {
            this.reset = reset;
            this.monitor = monitor;
        }

        @Override
        public void run() {
            synchronized (this.monitor) {
                try {
                    while (System.currentTimeMillis() < this.reset) {
                        this.logger.warn("Approaching rate limit.  Blocking all outgoing connections until {}",
                                new Date(this.reset));
                        Thread.sleep(this.reset - System.currentTimeMillis());
                    }

                    this.monitor.notifyAll();
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                    this.logger.warn("Rate limit notifier interrupted waiting for rate limit to exipre");
                }
            }
        }
    }

}