001/*- 002 * #%L 003 * HAPI FHIR - Core Library 004 * %% 005 * Copyright (C) 2014 - 2025 Smile CDR, Inc. 006 * %% 007 * Licensed under the Apache License, Version 2.0 (the "License"); 008 * you may not use this file except in compliance with the License. 009 * You may obtain a copy of the License at 010 * 011 * http://www.apache.org/licenses/LICENSE-2.0 012 * 013 * Unless required by applicable law or agreed to in writing, software 014 * distributed under the License is distributed on an "AS IS" BASIS, 015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 * #L% 019 */ 020package ca.uhn.fhir.rest.api; 021 022import org.slf4j.Logger; 023import org.slf4j.LoggerFactory; 024 025import java.util.List; 026import java.util.StringTokenizer; 027 028import static org.apache.commons.lang3.StringUtils.trim; 029 030/** 031 * Parses and stores the value(s) within HTTP Cache-Control headers 032 */ 033public class CacheControlDirective { 034 035 private static final String MAX_RESULTS_EQUALS = Constants.CACHE_CONTROL_MAX_RESULTS + "="; 036 private static final Logger ourLog = LoggerFactory.getLogger(CacheControlDirective.class); 037 private boolean myNoCache; 038 private boolean myNoStore; 039 private Integer myMaxResults; 040 041 /** 042 * Constructor 043 */ 044 public CacheControlDirective() { 045 super(); 046 } 047 048 /** 049 * If the {@link #isNoStore() no-store} directive is set, this HAPI FHIR extention 050 * to the <code>Cache-Control</code> header called <code>max-results=123</code> 051 * specified the maximum number of results which will be fetched from the 052 * database before returning. 053 */ 054 public Integer getMaxResults() { 055 return myMaxResults; 056 } 057 058 /** 059 * If the {@link #isNoStore() no-store} directive is set, this HAPI FHIR extention 060 * to the <code>Cache-Control</code> header called <code>max-results=123</code> 061 * specified the maximum number of results which will be fetched from the 062 * database before returning. 063 */ 064 public CacheControlDirective setMaxResults(Integer theMaxResults) { 065 myMaxResults = theMaxResults; 066 return this; 067 } 068 069 /** 070 * If <code>true<</code>, adds the <code>no-cache</code> directive to the 071 * request. This directive indicates that the cache should not be used to 072 * serve this request. 073 */ 074 public boolean isNoCache() { 075 return myNoCache; 076 } 077 078 /** 079 * If <code>true<</code>, adds the <code>no-cache</code> directive to the 080 * request. This directive indicates that the cache should not be used to 081 * serve this request. 082 */ 083 public CacheControlDirective setNoCache(boolean theNoCache) { 084 myNoCache = theNoCache; 085 return this; 086 } 087 088 public boolean isNoStore() { 089 return myNoStore; 090 } 091 092 public CacheControlDirective setNoStore(boolean theNoStore) { 093 myNoStore = theNoStore; 094 return this; 095 } 096 097 /** 098 * Parses a list of <code>Cache-Control</code> header values 099 * 100 * @param theValues The <code>Cache-Control</code> header values 101 */ 102 public CacheControlDirective parse(List<String> theValues) { 103 if (theValues != null) { 104 for (String nextValue : theValues) { 105 StringTokenizer tok = new StringTokenizer(nextValue, ","); 106 while (tok.hasMoreTokens()) { 107 String next = trim(tok.nextToken()); 108 if (Constants.CACHE_CONTROL_NO_CACHE.equals(next)) { 109 myNoCache = true; 110 } else if (Constants.CACHE_CONTROL_NO_STORE.equals(next)) { 111 myNoStore = true; 112 } else if (next.startsWith(MAX_RESULTS_EQUALS)) { 113 String valueString = trim(next.substring(MAX_RESULTS_EQUALS.length())); 114 try { 115 myMaxResults = Integer.parseInt(valueString); 116 } catch (NumberFormatException e) { 117 ourLog.warn("Invalid {} value: {}", Constants.CACHE_CONTROL_MAX_RESULTS, valueString); 118 } 119 } 120 } 121 } 122 } 123 124 return this; 125 } 126 127 /** 128 * Convenience factory method for a no-cache directivel 129 */ 130 public static CacheControlDirective noCache() { 131 return new CacheControlDirective().setNoCache(true); 132 } 133}