Получение значений диапазона из HttpHeader - PullRequest
3 голосов
/ 12 января 2012

В целях нумерации страниц наш пользовательский интерфейс определяет диапазоны элементов в заголовке Http следующим образом:

Range: items=0-15

В последующих запросах диапазоны из Интернета могут быть либо

Range: items=16-31
Range: items=32-45

... и т.д ... и т.д.

В моем контроллере SearchRequestController.java я пытаюсь извлечь эти диапазоны, чтобы я мог отправить его на сервер Solr для возврата значений в запрошенных патронах, установив начало и смещение.

То, что я делаю, выглядит следующим образом (в SearchRequestController.java):

@RequestMapping("/billsSearch")
@ResponseBody
public SearchResult searchBills(HttpServletRequest request,@RequestBody SearchRequest searchRequest){
    String rangeRequest = request.getHeader("Range");
    //(1)Parse the rangeRequest using some mechanism
    //(2)Set the start using the first value of the range
    //(3) Add the second value to the start to get the offset
    searchRequest.setStart(startValue);
    searchRequest.setOffset(offsetValue);
    return searchHelper(request,searchRequest);
}

У меня есть пара вопросов: Является ли это наилучшим способом запрашивать данные в виде фрагментов с сервера?

Как извлечь диапазоны из заголовка запроса?

Я предполагаю, что request.getHeader("Range") вернет "items=16-31".

Является ли регулярное выражение лучшим способом получить 16 и 31 из этой строки?

Если я решу использовать регулярное выражение, не сломается ли выражение, если я изменю заголовок диапазона на billItems = 16-31?

Я регулярное выражение n00b, поэтому вполне возможно, что я думаю об этом неправильно.

Существуют ли альтернативы регулярному выражению для разбора информации о диапазоне, подобной этой, из заголовков http в SpringMVC?

Ответы [ 3 ]

2 голосов
/ 01 октября 2015

Простой анализатор заголовка диапазона, соответствующий http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.35

class Range {
    Integer start;
    Integer end;
    Integer suffixLength;
}

public class Main {

    public static void main(String[] args) {
        String input = "bytes=33-22,-333,20-,-55,43-0002";
        for (Range range : decodeRange(input)) {
            if (range.suffixLength == null) {
                System.out.printf("Range: %d->%s\n", range.start, range.end == null ? "enf of file" : range.end);
            } else {
                System.out.printf("Last %d bytes\n", range.suffixLength);
            }
        }
    }

    public static List<Range> decodeRange(String rangeHeader) {
        List<Range> ranges = new ArrayList<>();
        String byteRangeSetRegex = "(((?<byteRangeSpec>(?<firstBytePos>\\d+)-(?<lastBytePos>\\d+)?)|(?<suffixByteRangeSpec>-(?<suffixLength>\\d+)))(,|$))";
        String byteRangesSpecifierRegex = "bytes=(?<byteRangeSet>" + byteRangeSetRegex + "{1,})";
        Pattern byteRangeSetPattern = Pattern.compile(byteRangeSetRegex);
        Pattern byteRangesSpecifierPattern = Pattern.compile(byteRangesSpecifierRegex);
        Matcher byteRangesSpecifierMatcher = byteRangesSpecifierPattern.matcher(rangeHeader);
        if (byteRangesSpecifierMatcher.matches()) {
            String byteRangeSet = byteRangesSpecifierMatcher.group("byteRangeSet");
            Matcher byteRangeSetMatcher = byteRangeSetPattern.matcher(byteRangeSet);
            while (byteRangeSetMatcher.find()) {
                Range range = new Range();
                if (byteRangeSetMatcher.group("byteRangeSpec") != null) {
                    String start = byteRangeSetMatcher.group("firstBytePos");
                    String end = byteRangeSetMatcher.group("lastBytePos");
                    range.start = Integer.valueOf(start);
                    range.end = end == null ? null : Integer.valueOf(end);
                } else if (byteRangeSetMatcher.group("suffixByteRangeSpec") != null) {
                    range.suffixLength = Integer.valueOf(byteRangeSetMatcher.group("suffixLength"));
                } else {
                    throw new RuntimeException("Invalid range header");
                }
                ranges.add(range);
            }
        } else {
            throw new RuntimeException("Invalid range header");
        }
        return ranges;
    }
};

или посмотрите здесь

2 голосов
/ 13 января 2012

Является ли это наилучшим способом запроса данных с сервера по частям?

Другой подход заключается в использовании параметров запроса:

/billsSearch?from=15&to=31

Кстатиможно использовать @RequestHeader аннотацию:

public SearchResult searchBills(
        @RequestBody SearchRequest searchRequest,
        @RequestHeader("Range") String range) {
    //...
}

Как извлечь диапазоны из заголовка запроса?[...] Является ли регулярное выражение лучшим способом получить 16 и 31 из этой строки?

Я бы вырезал завершающую строку "items=", разделил бы на символ - и проанализировал два результирующих числа:

String[] ranges = range.substring("items=".length()).split("-");
int from = Integer.valueOf(ranges[0]);
int to = Integer.valueOf(ranges[1]);
1 голос
/ 13 января 2012

Оба аргумента @RequestParameter и @RequestHeader подлежат преобразованию типа, если целевой тип не является строковым.Если бы у вас был тип Range, вы могли бы создать конвертери зарегистрируйте его с помощью ConversionService.Тогда ваш контроллер будет выглядеть немного чище:

    public SearchResult searchBills(
            @RequestBody SearchRequest searchRequest,
            @RequestHeader Range range) {
        //...
    }

Это было бы полезно, если бы это часто происходило в контроллерах.Или, если анализировать только внутри одного метода, достаточно просто.

...