Як використовувати перехоплювач для додавання заголовків у Retrofit 2.0?


96

Наша команда приймає рішення про модернізацію 2.0, і я провожу кілька початкових досліджень щодо цього. Я новачок у цій бібліотеці.

Мені цікаво, як використовувати interceptorдодавання індивідуальних заголовків через Retrofits 2.0 у нашому додатку для Android. Є багато підручників щодо використання interceptorдля додавання заголовків у Retrofit 1.X, але оскільки API сильно змінилися в останній версії, я не впевнений, як адаптувати ці методи в новій версії. Також компанія Retrofit ще не оновлювала свою нову документацію.

Наприклад, у наступних кодах, як мені реалізувати Interceptorклас для додавання зайвих заголовків? Крім того, що саме являє собою недокументований Chainоб'єкт ? Коли буде intercept()викликано?

    OkHttpClient client = new OkHttpClient();
    client.interceptors().add(new Interceptor() {
        @Override
        public Response intercept(Chain chain) throws IOException {
            Response response = chain.proceed(chain.request());

            // How to add extra headers?

            return response;
        }
    });

    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(BASE_API_URL)
            .client(client)
            .addConverterFactory(GsonConverterFactory.create())
            .build();

1
Переконайтеся, що ваш BASE_API_URL закінчується на, /а URL-адреси ваших API ні ( stuff/post/whatever)
EpicPandaForce

Відповіді:


120

Заціни.

public class HeaderInterceptor implements Interceptor {
    @Override
    public Response intercept(Chain chain) throws IOException {
        Request request = chain.request()
                .newBuilder()
                .addHeader("appid", "hello")
                .addHeader("deviceplatform", "android")
                .removeHeader("User-Agent")
                .addHeader("User-Agent", "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:38.0) Gecko/20100101 Firefox/38.0")
                .build();
        Response response = chain.proceed(request);
        return response;
    }
}

Котлін

class HeaderInterceptor : Interceptor {
    override fun intercept(chain: Interceptor.Chain): Response = chain.run {
        proceed(
            request()
                .newBuilder()
                .addHeader("appid", "hello")
                .addHeader("deviceplatform", "android")
                .removeHeader("User-Agent")
                .addHeader("User-Agent", "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:38.0) Gecko/20100101 Firefox/38.0")
                .build()
        )        
    }
}

Дякую!! Отже, це intercept()спрацьовує кожного разу, коли запит надсилається із програми? Чи можемо ми перехопити проміжну відповідь для перенаправлення, або ми просто отримаємо остаточну відповідь?
хакджуцу

Це вимагається для кожного запиту, і якщо я добре знаю, це тому, що ви додаєте його як перехоплювач, а не як мережевий перехоплювач. Я думаю, що ви можете отримати тут лише остаточну відповідь, але може бути конфігурація, яка дозволяє бачити переспрямування як переспрямування, яких я не знаю вгорі (є також для підключення URL-адреси http.)
EpicPandaForce

1
Просто зверніться за цим посиланням: github.com/square/okhttp/wiki/Interceptors і отримайте потрібну інформацію :) Дякую ~
хакцуцу

5
Fyi, вам потрібно використовувати будівельник замість client.interceptors(). Це виглядає такnew OkHttpClient.Builder().addInterceptor(<Your Interceptor>).build()
GLee

22

Ще одна альтернатива прийнятої відповіді

public class HeaderInterceptor implements Interceptor {

    @Override
    public Response intercept(Chain chain) throws IOException {
        Request request = chain.request();

        request = request.newBuilder()
                .addHeader("headerKey0", "HeaderVal0")
                .addHeader("headerKey0", "HeaderVal0--NotReplaced/NorUpdated") //new header added
                .build();

        //alternative
        Headers moreHeaders = request.headers().newBuilder()
                .add("headerKey1", "HeaderVal1")
                .add("headerKey2", "HeaderVal2")
                .set("headerKey2", "HeaderVal2--UpdatedHere") // existing header UPDATED if available, else added.
                .add("headerKey3", "HeaderKey3")
                .add("headerLine4 : headerLine4Val") //line with `:`, spaces doesn't matter.
                .removeAll("headerKey3") //Oops, remove this.
                .build();

        request = request.newBuilder().headers(moreHeaders).build();

        /* ##### List of headers ##### */
        // headerKey0: HeaderVal0
        // headerKey0: HeaderVal0--NotReplaced/NorUpdated
        // headerKey1: HeaderVal1
        // headerKey2: HeaderVal2--UpdatedHere
        // headerLine4: headerLine4Val

        Response response = chain.proceed(request);
        return response;
    }
}

Приємно! Тож request.newBuilder().headers(moreHeaders).build()збережете оригінальні заголовки?
хакджуцу

1
Так. Жодні заголовки не видаляються із запиту, якщо не викликано removeAll (ім'я рядка) .
VenomVendor

@VenomVendor , будь ласка , допоможіть мені з таким же питанням тут stackoverflow.com/questions/45078720 / ... спасибі
user606669

Не хочете, щоб це продовжувало створювати нові об’єкти?
TheRealChx101

3
   public class ServiceFactory {  
    public static ApiClient createService(String authToken, String userName, String password) {
            OkHttpClient defaultHttpClient = new OkHttpClient.Builder()
                    .addInterceptor(
                            chain -> {
                                Request request = chain.request().newBuilder()
                                        .headers(getJsonHeader(authToken))
                                        .build();
                                return chain.proceed(request);
                            })
                    .authenticator(getBasicAuthenticator(userName, password))
                    .build();
            return getService(defaultHttpClient);
        }
        private static Headers getJsonHeader(String authToken) {
            Headers.Builder builder = new Headers.Builder();
            builder.add("Content-Type", "application/json");
            builder.add("Accept", "application/json");
            if (authToken != null && !authToken.isEmpty()) {
                builder.add("X-MY-Auth", authToken);
            }
            return builder.build();
        }
        private static Authenticator getBasicAuthenticator(final String userName, final String password) {
            return (route, response) -> {
                String credential = Credentials.basic(userName, password);
                return response.request().newBuilder().header("Authorization", credential).build();
            };
        }
          private static ApiClient getService(OkHttpClient defaultHttpClient) {
            return new Retrofit.Builder()
                    .baseUrl(BASE_URL)
                    .addConverterFactory(GsonConverterFactory.create())
                    .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
                    .client(defaultHttpClient)
                    .build()
                    .create(ApiClient.class);
        }
}

2

Ви можете створювати заголовки, використовуючи перехоплювачі з такими вбудованими методами, як цей

   interceptors().add(new Interceptor() {
        @Override
        public Response intercept(Chain chain) throws IOException {
            Request original = chain.request();

            Request.Builder builder = original.newBuilder();

            builder.header("Authorization","Bearer "+ LeafPreference.getInstance(context).getString(LeafPreference.TOKEN));

            Request request = builder.method(original.method(), original.body())
                    .build();
            Log.e("request",request.urlString());
            Log.e("header",request.header("Authorization"));
            return chain.proceed(request);
        }
    });
}

Я хочу знати, як ви отримуєте контекст у цьому місці?
rupinderjeet

@rupinderjeet Можливо a final Context contextу списку параметрів.
TheRealChx101

@ TheRealChx101 Просто хотів зазначити, що ми не повинні бути contextтут, оскільки це ділова логіка.
rupinderjeet
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.