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

org.jooby.Request Maven / Gradle / Ivy

There is a newer version: 1.0.0
Show newest version
/**
 *                                  Apache License
 *                            Version 2.0, January 2004
 *                         http://www.apache.org/licenses/
 *
 *    TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
 *
 *    1. Definitions.
 *
 *       "License" shall mean the terms and conditions for use, reproduction,
 *       and distribution as defined by Sections 1 through 9 of this document.
 *
 *       "Licensor" shall mean the copyright owner or entity authorized by
 *       the copyright owner that is granting the License.
 *
 *       "Legal Entity" shall mean the union of the acting entity and all
 *       other entities that control, are controlled by, or are under common
 *       control with that entity. For the purposes of this definition,
 *       "control" means (i) the power, direct or indirect, to cause the
 *       direction or management of such entity, whether by contract or
 *       otherwise, or (ii) ownership of fifty percent (50%) or more of the
 *       outstanding shares, or (iii) beneficial ownership of such entity.
 *
 *       "You" (or "Your") shall mean an individual or Legal Entity
 *       exercising permissions granted by this License.
 *
 *       "Source" form shall mean the preferred form for making modifications,
 *       including but not limited to software source code, documentation
 *       source, and configuration files.
 *
 *       "Object" form shall mean any form resulting from mechanical
 *       transformation or translation of a Source form, including but
 *       not limited to compiled object code, generated documentation,
 *       and conversions to other media types.
 *
 *       "Work" shall mean the work of authorship, whether in Source or
 *       Object form, made available under the License, as indicated by a
 *       copyright notice that is included in or attached to the work
 *       (an example is provided in the Appendix below).
 *
 *       "Derivative Works" shall mean any work, whether in Source or Object
 *       form, that is based on (or derived from) the Work and for which the
 *       editorial revisions, annotations, elaborations, or other modifications
 *       represent, as a whole, an original work of authorship. For the purposes
 *       of this License, Derivative Works shall not include works that remain
 *       separable from, or merely link (or bind by name) to the interfaces of,
 *       the Work and Derivative Works thereof.
 *
 *       "Contribution" shall mean any work of authorship, including
 *       the original version of the Work and any modifications or additions
 *       to that Work or Derivative Works thereof, that is intentionally
 *       submitted to Licensor for inclusion in the Work by the copyright owner
 *       or by an individual or Legal Entity authorized to submit on behalf of
 *       the copyright owner. For the purposes of this definition, "submitted"
 *       means any form of electronic, verbal, or written communication sent
 *       to the Licensor or its representatives, including but not limited to
 *       communication on electronic mailing lists, source code control systems,
 *       and issue tracking systems that are managed by, or on behalf of, the
 *       Licensor for the purpose of discussing and improving the Work, but
 *       excluding communication that is conspicuously marked or otherwise
 *       designated in writing by the copyright owner as "Not a Contribution."
 *
 *       "Contributor" shall mean Licensor and any individual or Legal Entity
 *       on behalf of whom a Contribution has been received by Licensor and
 *       subsequently incorporated within the Work.
 *
 *    2. Grant of Copyright License. Subject to the terms and conditions of
 *       this License, each Contributor hereby grants to You a perpetual,
 *       worldwide, non-exclusive, no-charge, royalty-free, irrevocable
 *       copyright license to reproduce, prepare Derivative Works of,
 *       publicly display, publicly perform, sublicense, and distribute the
 *       Work and such Derivative Works in Source or Object form.
 *
 *    3. Grant of Patent License. Subject to the terms and conditions of
 *       this License, each Contributor hereby grants to You a perpetual,
 *       worldwide, non-exclusive, no-charge, royalty-free, irrevocable
 *       (except as stated in this section) patent license to make, have made,
 *       use, offer to sell, sell, import, and otherwise transfer the Work,
 *       where such license applies only to those patent claims licensable
 *       by such Contributor that are necessarily infringed by their
 *       Contribution(s) alone or by combination of their Contribution(s)
 *       with the Work to which such Contribution(s) was submitted. If You
 *       institute patent litigation against any entity (including a
 *       cross-claim or counterclaim in a lawsuit) alleging that the Work
 *       or a Contribution incorporated within the Work constitutes direct
 *       or contributory patent infringement, then any patent licenses
 *       granted to You under this License for that Work shall terminate
 *       as of the date such litigation is filed.
 *
 *    4. Redistribution. You may reproduce and distribute copies of the
 *       Work or Derivative Works thereof in any medium, with or without
 *       modifications, and in Source or Object form, provided that You
 *       meet the following conditions:
 *
 *       (a) You must give any other recipients of the Work or
 *           Derivative Works a copy of this License; and
 *
 *       (b) You must cause any modified files to carry prominent notices
 *           stating that You changed the files; and
 *
 *       (c) You must retain, in the Source form of any Derivative Works
 *           that You distribute, all copyright, patent, trademark, and
 *           attribution notices from the Source form of the Work,
 *           excluding those notices that do not pertain to any part of
 *           the Derivative Works; and
 *
 *       (d) If the Work includes a "NOTICE" text file as part of its
 *           distribution, then any Derivative Works that You distribute must
 *           include a readable copy of the attribution notices contained
 *           within such NOTICE file, excluding those notices that do not
 *           pertain to any part of the Derivative Works, in at least one
 *           of the following places: within a NOTICE text file distributed
 *           as part of the Derivative Works; within the Source form or
 *           documentation, if provided along with the Derivative Works; or,
 *           within a display generated by the Derivative Works, if and
 *           wherever such third-party notices normally appear. The contents
 *           of the NOTICE file are for informational purposes only and
 *           do not modify the License. You may add Your own attribution
 *           notices within Derivative Works that You distribute, alongside
 *           or as an addendum to the NOTICE text from the Work, provided
 *           that such additional attribution notices cannot be construed
 *           as modifying the License.
 *
 *       You may add Your own copyright statement to Your modifications and
 *       may provide additional or different license terms and conditions
 *       for use, reproduction, or distribution of Your modifications, or
 *       for any such Derivative Works as a whole, provided Your use,
 *       reproduction, and distribution of the Work otherwise complies with
 *       the conditions stated in this License.
 *
 *    5. Submission of Contributions. Unless You explicitly state otherwise,
 *       any Contribution intentionally submitted for inclusion in the Work
 *       by You to the Licensor shall be under the terms and conditions of
 *       this License, without any additional terms or conditions.
 *       Notwithstanding the above, nothing herein shall supersede or modify
 *       the terms of any separate license agreement you may have executed
 *       with Licensor regarding such Contributions.
 *
 *    6. Trademarks. This License does not grant permission to use the trade
 *       names, trademarks, service marks, or product names of the Licensor,
 *       except as required for reasonable and customary use in describing the
 *       origin of the Work and reproducing the content of the NOTICE file.
 *
 *    7. Disclaimer of Warranty. Unless required by applicable law or
 *       agreed to in writing, Licensor provides the Work (and each
 *       Contributor provides its Contributions) on an "AS IS" BASIS,
 *       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
 *       implied, including, without limitation, any warranties or conditions
 *       of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
 *       PARTICULAR PURPOSE. You are solely responsible for determining the
 *       appropriateness of using or redistributing the Work and assume any
 *       risks associated with Your exercise of permissions under this License.
 *
 *    8. Limitation of Liability. In no event and under no legal theory,
 *       whether in tort (including negligence), contract, or otherwise,
 *       unless required by applicable law (such as deliberate and grossly
 *       negligent acts) or agreed to in writing, shall any Contributor be
 *       liable to You for damages, including any direct, indirect, special,
 *       incidental, or consequential damages of any character arising as a
 *       result of this License or out of the use or inability to use the
 *       Work (including but not limited to damages for loss of goodwill,
 *       work stoppage, computer failure or malfunction, or any and all
 *       other commercial damages or losses), even if such Contributor
 *       has been advised of the possibility of such damages.
 *
 *    9. Accepting Warranty or Additional Liability. While redistributing
 *       the Work or Derivative Works thereof, You may choose to offer,
 *       and charge a fee for, acceptance of support, warranty, indemnity,
 *       or other liability obligations and/or rights consistent with this
 *       License. However, in accepting such obligations, You may act only
 *       on Your own behalf and on Your sole responsibility, not on behalf
 *       of any other Contributor, and only if You agree to indemnify,
 *       defend, and hold each Contributor harmless for any liability
 *       incurred by, or claims asserted against, such Contributor by reason
 *       of your accepting any such warranty or additional liability.
 *
 *    END OF TERMS AND CONDITIONS
 *
 *    APPENDIX: How to apply the Apache License to your work.
 *
 *       To apply the Apache License to your work, attach the following
 *       boilerplate notice, with the fields enclosed by brackets "{}"
 *       replaced with your own identifying information. (Don't include
 *       the brackets!)  The text should be enclosed in the appropriate
 *       comment syntax for the file format. We also recommend that a
 *       file or class name and description of purpose be included on the
 *       same "printed page" as the copyright notice for easier
 *       identification within third-party archives.
 *
 *    Copyright 2014 Edgar Espina
 *
 *    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 org.jooby;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.net.UrlEscapers;
import com.google.inject.Key;
import com.google.inject.TypeLiteral;
import static java.util.Objects.requireNonNull;
import org.jooby.scope.RequestScoped;

import javax.annotation.Nonnull;
import java.io.IOException;
import java.nio.charset.Charset;
import java.util.List;
import java.util.Locale;
import java.util.Locale.LanguageRange;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Optional;
import java.util.function.BiFunction;

/**
 * Give you access at the current HTTP request in order to read parameters, headers and body.
 *
 * 

HTTP parameter and headers

*

* Access to HTTP parameter/header is available via {@link #param(String)} and * {@link #header(String)} methods. See some examples: *

* *
 *   // str param
 *   String value = request.param("str").value();
 *
 *   // optional str
 *   String value = request.param("str").value("defs");
 *
 *   // int param
 *   int value = request.param("some").intValue();
 *
 *   // optional int param
 *   Optional{@literal <}Integer{@literal >} value = request.param("some").toOptional(Integer.class);

 *   // list param
 *   List{@literal <}String{@literal >} values = request.param("some").toList(String.class);
 * 
* *

form post/multi-param request

*

* Due that form post are treated as HTTP params you can collect all them into a Java Object via * {@link #params(Class)} or {@link #form(Class)} methods: *

* *
{@code
 * {
 *   get("/search", req -> {
 *     Query q = req.params(Query.class);
 *   });
 *
 *   post("/person", req -> {
 *     Person person = req.form(Person.class);
 *   });
 * }
 * }
* *

form file upload

*

* Form post file upload are available via {@link #files(String)} or {@link #file(String)} methods: *

*
{@code
 * {
 *   post("/upload", req  -> {
 *     try(Upload upload = req.file("myfile")) {
 *       File file = upload.file();
 *       // work with file.
 *     }
 *   });
 * }
 * }
* * @author edgar * @since 0.1.0 */ public interface Request extends Registry { /** * Flash scope. * * @author edgar * @since 1.2.0 */ interface Flash extends Map { /** * Keep flash cookie for next request. */ void keep(); } /** * Forwarding request. * * @author edgar * @since 0.1.0 */ class Forwarding implements Request { /** Target request. */ private Request req; /** * Creates a new {@link Forwarding} request. * * @param request A target request. */ public Forwarding(final Request request) { this.req = requireNonNull(request, "A HTTP request is required."); } @Override public String path() { return req.path(); } @Override public String rawPath() { return req.rawPath(); } @Override public Optional queryString() { return req.queryString(); } @Override public String path(final boolean escape) { return req.path(escape); } @Override public boolean matches(final String pattern) { return req.matches(pattern); } @Override public String contextPath() { return req.contextPath(); } @Override public String method() { return req.method(); } @Override public MediaType type() { return req.type(); } @Override public List accept() { return req.accept(); } @Override public Optional accepts(final List types) { return req.accepts(types); } @Override public Optional accepts(final MediaType... types) { return req.accepts(types); } @Override public Optional accepts(final String... types) { return req.accepts(types); } @Override public boolean is(final List types) { return req.is(types); } @Override public boolean is(final MediaType... types) { return req.is(types); } @Override public boolean is(final String... types) { return req.is(types); } @Override public boolean isSet(final String name) { return req.isSet(name); }; @Override public Mutant params() { return req.params(); } @Override public Mutant params(final String... xss) { return req.params(xss); } @Override public T params(final Class type) { return req.params(type); } @Override public T params(final Class type, final String... xss) { return req.params(type, xss); } @Override public Mutant param(final String name) { return req.param(name); } @Override public Mutant param(final String name, final String... xss) { return req.param(name, xss); } @Override public Upload file(final String name) throws IOException { return req.file(name); } @Override public List files(final String name) throws IOException { return req.files(name); } @Override public Mutant header(final String name) { return req.header(name); } @Override public Mutant header(final String name, final String... xss) { return req.header(name, xss); } @Override public Map headers() { return req.headers(); } @Override public Mutant cookie(final String name) { return req.cookie(name); } @Override public List cookies() { return req.cookies(); } @Override public Mutant body() throws Exception { return req.body(); } @Override public T body(final Class type) throws Exception { return req.body(type); } @Override public T require(final Class type) { return req.require(type); } @Override public T require(final TypeLiteral type) { return req.require(type); } @Override public T require(final Key key) { return req.require(key); } @Override public Charset charset() { return req.charset(); } @Override public long length() { return req.length(); } @Override public Locale locale() { return req.locale(); } @Override public Locale locale(final BiFunction, List, Locale> filter) { return req.locale(filter); } @Override public List locales( final BiFunction, List, List> filter) { return req.locales(filter); } @Override public List locales() { return req.locales(); } @Override public String ip() { return req.ip(); } @Override public int port() { return req.port(); } @Override public Route route() { return req.route(); } @Override public Session session() { return req.session(); } @Override public Optional ifSession() { return req.ifSession(); } @Override public String hostname() { return req.hostname(); } @Override public String protocol() { return req.protocol(); } @Override public boolean secure() { return req.secure(); } @Override public boolean xhr() { return req.xhr(); } @Override public Map attributes() { return req.attributes(); } @Override public Optional ifGet(final String name) { return req.ifGet(name); } @Override public T get(final String name) { return req.get(name); } @Override public T get(final String name, final T def) { return req.get(name, def); } @Override public Request set(final String name, final Object value) { req.set(name, value); return this; } @Override public Request set(final Key key, final Object value) { req.set(key, value); return this; } @Override public Request set(final Class type, final Object value) { req.set(type, value); return this; } @Override public Request set(final TypeLiteral type, final Object value) { req.set(type, value); return this; } @Override public Optional unset(final String name) { return req.unset(name); } @Override public Flash flash() throws NoSuchElementException { return req.flash(); } @Override public String flash(final String name) throws NoSuchElementException { return req.flash(name); } @Override public Request flash(final String name, final Object value) { req.flash(name, value); return this; } @Override public Optional ifFlash(final String name) { return req.ifFlash(name); } @Override public Request push(final String path) { req.push(path); return this; } @Override public Request push(final String path, final Map headers) { req.push(path, headers); return this; } @Override public long timestamp() { return req.timestamp(); } @Override public String toString() { return req.toString(); } /** * Unwrap a request in order to find out the target instance. * * @param req A request. * @return A target instance (not a {@link Forwarding}). */ public static Request unwrap(final Request req) { requireNonNull(req, "A request is required."); Request root = req; while (root instanceof Forwarding) { root = ((Forwarding) root).req; } return root; } } /** * Given: * *
   *  http://domain.com/some/path.html {@literal ->} /some/path.html
   *  http://domain.com/a.html         {@literal ->} /a.html
   * 
* * @return The request URL pathname. */ @Nonnull default String path() { return path(false); } /** * Raw path, like {@link #path()} but without decoding. * * @return Raw path, like {@link #path()} but without decoding. */ @Nonnull String rawPath(); /** * The query string, without the leading ?. * * @return The query string, without the leading ?. */ @Nonnull Optional queryString(); /** * Escape the path using {@link UrlEscapers#urlFragmentEscaper()}. * * Given: * *
{@code
   *  http://domain.com/404

X

{@literal ->} /404%3Ch1%3EX%3C/h1%3E * }
* * @param escape True if we want to escape this path. * @return The request URL pathname. */ @Nonnull default String path(final boolean escape) { String path = route().path(); return escape ? UrlEscapers.urlFragmentEscaper().escape(path) : path; } /** * Application path (a.k.a context path). It is the value defined by: * application.path. Default is: / * * This method returns empty string for /. Otherwise, it is identical the value of * application.path. * * @return Application context path.. */ @Nonnull String contextPath(); /** * @return HTTP method. */ @Nonnull default String method() { return route().method(); } /** * @return The Content-Type header. Default is: {@literal*}/{@literal*}. */ @Nonnull MediaType type(); /** * @return The value of the Accept header. Default is: {@literal*}/{@literal*}. */ @Nonnull List accept(); /** * Check if the given types are acceptable, returning the best match when true, or else * Optional.empty. * *
   * // Accept: text/html
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   *
   * // Accept: text/*, application/json
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   * req.accepts("application/json" "text/plain");
   * // {@literal =>} "application/json"
   * req.accepts("application/json");
   * // {@literal =>} "application/json"
   *
   * // Accept: text/*, application/json
   * req.accepts("image/png");
   * // {@literal =>} Optional.empty
   *
   * // Accept: text/*;q=.5, application/json
   * req.accepts("text/html", "application/json");
   * // {@literal =>} "application/json"
   * 
* * @param types Types to test. * @return The best acceptable type. */ @Nonnull default Optional accepts(final String... types) { return accepts(MediaType.valueOf(types)); } /** * Test if the given request path matches the pattern. * * @param pattern A pattern to test for. * @return True, if the request path matches the pattern. */ boolean matches(String pattern); /** * True, if request accept any of the given types. * * @param types Types to test * @return True if any of the given type is accepted. */ default boolean is(final String... types) { return accepts(types).isPresent(); } /** * True, if request accept any of the given types. * * @param types Types to test * @return True if any of the given type is accepted. */ default boolean is(final MediaType... types) { return accepts(types).isPresent(); } /** * True, if request accept any of the given types. * * @param types Types to test * @return True if any of the given type is accepted. */ default boolean is(final List types) { return accepts(types).isPresent(); } /** * Check if the given types are acceptable, returning the best match when true, or else * Optional.empty. * *
   * // Accept: text/html
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   *
   * // Accept: text/*, application/json
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   * req.accepts("application/json" "text/plain");
   * // {@literal =>} "application/json"
   * req.accepts("application/json");
   * // {@literal =>} "application/json"
   *
   * // Accept: text/*, application/json
   * req.accepts("image/png");
   * // {@literal =>} Optional.empty
   *
   * // Accept: text/*;q=.5, application/json
   * req.accepts("text/html", "application/json");
   * // {@literal =>} "application/json"
   * 
* * @param types Types to test. * @return The best acceptable type. */ @Nonnull default Optional accepts(final MediaType... types) { return accepts(ImmutableList.copyOf(types)); } /** * Check if the given types are acceptable, returning the best match when true, or else * Optional.empty. * *
   * // Accept: text/html
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   *
   * // Accept: text/*, application/json
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   * req.accepts("text/html");
   * // {@literal =>} "text/html"
   * req.accepts("application/json" "text/plain");
   * // {@literal =>} "application/json"
   * req.accepts("application/json");
   * // {@literal =>} "application/json"
   *
   * // Accept: text/*, application/json
   * req.accepts("image/png");
   * // {@literal =>} Optional.empty
   *
   * // Accept: text/*;q=.5, application/json
   * req.accepts("text/html", "application/json");
   * // {@literal =>} "application/json"
   * 
* * @param types Types to test for. * @return The best acceptable type. */ @Nonnull Optional accepts(List types); /** * Get all the available parameters. A HTTP parameter can be provided in any of * these forms: * *
    *
  • Path parameter, like: /path/:name or /path/{name}
  • *
  • Query parameter, like: ?name=jooby
  • *
  • Body parameter when Content-Type is * application/x-www-form-urlencoded or multipart/form-data
  • *
* * @return All the parameters. */ @Nonnull Mutant params(); /** * Get all the available parameters. A HTTP parameter can be provided in any of * these forms: * *
    *
  • Path parameter, like: /path/:name or /path/{name}
  • *
  • Query parameter, like: ?name=jooby
  • *
  • Body parameter when Content-Type is * application/x-www-form-urlencoded or multipart/form-data
  • *
* * @param xss Xss filter to apply. * @return All the parameters. */ @Nonnull Mutant params(String... xss); /** * Short version of params().to(type). * * @param type Object type. * @param Value type. * @return Instance of object. */ @Nonnull default T params(final Class type) { return params().to(type); } /** * Short version of params().to(type). * * @param type Object type. * @param Value type. * @return Instance of object. */ @Nonnull default T form(final Class type) { return params().to(type); } /** * Short version of params(xss).to(type). * * @param type Object type. * @param xss Xss filter to apply. * @param Value type. * @return Instance of object. */ @Nonnull default T params(final Class type, final String... xss) { return params(xss).to(type); } /** * Short version of params(xss).to(type). * * @param type Object type. * @param xss Xss filter to apply. * @param Value type. * @return Instance of object. */ @Nonnull default T form(final Class type, final String... xss) { return params(xss).to(type); } /** * Get a HTTP request parameter under the given name. A HTTP parameter can be provided in any of * these forms: *
    *
  • Path parameter, like: /path/:name or /path/{name}
  • *
  • Query parameter, like: ?name=jooby
  • *
  • Body parameter when Content-Type is * application/x-www-form-urlencoded or multipart/form-data
  • *
* * The order of precedence is: path, query and body. For * example a pattern like: GET /path/:name for /path/jooby?name=rocks * produces: * *
   *  assertEquals("jooby", req.param(name).value());
   *
   *  assertEquals("jooby", req.param(name).toList().get(0));
   *  assertEquals("rocks", req.param(name).toList().get(1));
   * 
* * Uploads can be retrieved too when Content-Type is multipart/form-data * see {@link Upload} for more information. * * @param name A parameter's name. * @return A HTTP request parameter. */ @Nonnull Mutant param(String name); /** * Get a HTTP request parameter under the given name. A HTTP parameter can be provided in any of * these forms: *
    *
  • Path parameter, like: /path/:name or /path/{name}
  • *
  • Query parameter, like: ?name=jooby
  • *
  • Body parameter when Content-Type is * application/x-www-form-urlencoded or multipart/form-data
  • *
* * The order of precedence is: path, query and body. For * example a pattern like: GET /path/:name for /path/jooby?name=rocks * produces: * *
   *  assertEquals("jooby", req.param(name).value());
   *
   *  assertEquals("jooby", req.param(name).toList().get(0));
   *  assertEquals("rocks", req.param(name).toList().get(1));
   * 
* * Uploads can be retrieved too when Content-Type is multipart/form-data * see {@link Upload} for more information. * * @param name A parameter's name. * @param xss Xss filter to apply. * @return A HTTP request parameter. */ @Nonnull Mutant param(String name, String... xss); /** * Get a file {@link Upload} with the given name. The request must be a POST with * multipart/form-data content-type. * * @param name File's name. * @return An {@link Upload}. * @throws IOException */ @Nonnull default Upload file(final String name) throws IOException { List files = files(name); if (files.size() == 0) { throw new Err.Missing(name); } return files.get(0); } /** * Get a file {@link Upload} with the given name or empty. The request must be a POST with * multipart/form-data content-type. * * @param name File's name. * @return An {@link Upload}. * @throws IOException */ @Nonnull default Optional ifFile(final String name) throws IOException { List files = files(name); return files.size() == 0 ? Optional.empty() : Optional.of(files.get(0)); } /** * Get a list of file {@link Upload} with the given name. The request must be a POST with * multipart/form-data content-type. * * @param name File's name. * @return A list of {@link Upload}. * @throws IOException */ @Nonnull List files(final String name) throws IOException; /** * Get a HTTP header. * * @param name A header's name. * @return A HTTP request header. */ @Nonnull Mutant header(String name); /** * Get a HTTP header and apply the XSS escapers. * * @param name A header's name. * @param xss Xss escapers. * @return A HTTP request header. */ @Nonnull Mutant header(final String name, final String... xss); /** * @return All the headers. */ @Nonnull Map headers(); /** * Get a cookie with the given name (if present). * * @param name Cookie's name. * @return A cookie or an empty optional. */ @Nonnull Mutant cookie(String name); /** * @return All the cookies. */ @Nonnull List cookies(); /** * HTTP body. Please don't use this method for form submits. This method is used for getting * raw data or a data like json, xml, etc... * * @return The HTTP body. * @throws Exception If body can't be converted or there is no HTTP body. */ @Nonnull Mutant body() throws Exception; /** * Short version of body().to(type). * * HTTP body. Please don't use this method for form submits. This method is used for getting * raw or a parsed data like json, xml, etc... * * @param type Object type. * @param Value type. * @return Instance of object. * @throws Exception If body can't be converted or there is no HTTP body. */ @Nonnull default T body(final Class type) throws Exception { return body().to(type); } /** * The charset defined in the request body. If the request doesn't specify a character * encoding, this method return the global charset: application.charset. * * @return A current charset. */ @Nonnull Charset charset(); /** * Get a list of locale that best matches the current request as per {@link Locale#filter}. * * @return A list of matching locales or empty list. */ @Nonnull default List locales() { return locales(Locale::filter); } /** * Get a list of locale that best matches the current request. * * The first filter argument is the value of Accept-Language as * {@link Locale.LanguageRange} and filter while the second argument is a list of supported * locales defined by the application.lang property. * * The next example returns a list of matching {@code Locale} instances using the filtering * mechanism defined in RFC 4647: * *
{@code
   * req.locales(Locale::filter)
   * }
* * @param filter A locale filter. * @return A list of matching locales. */ @Nonnull List locales(BiFunction, List, List> filter); /** * Get a locale that best matches the current request. * * The first filter argument is the value of Accept-Language as * {@link Locale.LanguageRange} and filter while the second argument is a list of supported * locales defined by the application.lang property. * * The next example returns a {@code Locale} instance for the best-matching language * tag using the lookup mechanism defined in RFC 4647. * *
{@code
   * req.locale(Locale::lookup)
   * }
* * @param filter A locale filter. * @return A matching locale. */ @Nonnull Locale locale(BiFunction, List, Locale> filter); /** * Get a locale that best matches the current request or the default locale as specified * in application.lang. * * @return A matching locale. */ @Nonnull default Locale locale() { return locale((priorityList, locales) -> Optional.ofNullable(Locale.lookup(priorityList, locales)) .orElse(locales.get(0))); } /** * @return The length, in bytes, of the request body and made available by the input stream, or * -1 if the length is not known. */ long length(); /** * @return The IP address of the client or last proxy that sent the request. */ @Nonnull String ip(); /** * @return Server port, from host header or the server port where the client * connection was accepted on. */ int port(); /** * @return The currently matched {@link Route}. */ @Nonnull Route route(); /** * The fully qualified name of the resource being requested, as obtained from the Host HTTP * header. * * @return The fully qualified name of the server. */ @Nonnull String hostname(); /** * @return The current session associated with this request or if the request does not have a * session, creates one. */ @Nonnull Session session(); /** * @return The current session associated with this request if there is one. */ @Nonnull Optional ifSession(); /** * @return True if the X-Requested-With header is set to XMLHttpRequest. */ default boolean xhr() { return header("X-Requested-With") .toOptional(String.class) .map("XMLHttpRequest"::equalsIgnoreCase) .orElse(Boolean.FALSE); } /** * @return The name and version of the protocol the request uses in the form * protocol/majorVersion.minorVersion, for example, HTTP/1.1 */ @Nonnull String protocol(); /** * @return True if this request was made using a secure channel, such as HTTPS. */ boolean secure(); /** * Set local attribute. * * @param name Attribute's name. * @param value Attribute's local. NOT null. * @return This request. */ @Nonnull Request set(String name, Object value); /** * Give you access to flash scope. Usage: * *
{@code
   * {
   *   use(new FlashScope());
   *
   *   get("/", req -> {
   *     Map flash = req.flash();
   *     return flash;
   *   });
   * }
   * }
* * As you can see in the example above, the {@link FlashScope} needs to be install it by calling * {@link Jooby#use(org.jooby.Jooby.Module)} otherwise a call to this method ends in * {@link Err BAD_REQUEST}. * * @return A mutable map with attributes from {@link FlashScope}. * @throws Err Bad request error if the {@link FlashScope} was not installed it. */ @Nonnull default Flash flash() throws Err { Optional flash = ifGet(FlashScope.NAME); return flash.orElseThrow(() -> new Err(Status.BAD_REQUEST, "Flash scope isn't available. Install via: use(new FlashScope());")); } /** * Set a flash attribute. Flash scope attributes are accessible from template engines, by * prefixing attributes with flash.. For example a call to * flash("success", "OK") is accessible from template engines using * flash.success * * @param name Attribute's name. * @param value Attribute's value. * @return This request. */ @Nonnull default Request flash(final String name, final Object value) { requireNonNull(name, "Attribute's name is required."); Map flash = flash(); if (value == null) { flash.remove(name); } else { flash.put(name, value.toString()); } return this; } /** * Get an optional for the given flash attribute's name. * * @param name Attribute's name. * @return Optional flash attribute. */ @Nonnull default Optional ifFlash(final String name) { return Optional.ofNullable(flash().get(name)); } /** * Get a flash attribute value or throws {@link Err BAD_REQUEST error} if missing. * * @param name Attribute's name. * @return Flash attribute. * @throws Err Bad request error if flash attribute is missing. */ @Nonnull default String flash(final String name) throws Err { return ifFlash(name) .orElseThrow(() -> new Err(Status.BAD_REQUEST, "Required flash attribute: '" + name + "' is not present")); } /** * @param name Attribute's name. * @return True if the local attribute is set. */ default boolean isSet(final String name) { return ifGet(name).isPresent(); } /** * Get a request local attribute. * * @param name Attribute's name. * @param Target type. * @return A local attribute. */ @Nonnull Optional ifGet(String name); /** * Get a request local attribute. * * @param name Attribute's name. * @param def A default value. * @param Target type. * @return A local attribute. */ @Nonnull default T get(final String name, final T def) { Optional opt = ifGet(name); return opt.orElse(def); } /** * Get a request local attribute. * * @param name Attribute's name. * @param Target type. * @return A local attribute. * @throws Err with {@link Status#BAD_REQUEST}. */ @Nonnull default T get(final String name) { Optional opt = ifGet(name); return opt.orElseThrow( () -> new Err(Status.BAD_REQUEST, "Required local attribute: " + name + " is not present")); } /** * Remove a request local attribute. * * @param name Attribute's name. * @param Target type. * @return A local attribute. */ @Nonnull Optional unset(String name); /** * A read only version of the current locals. * * @return Attributes locals. */ @Nonnull Map attributes(); /** * Seed a {@link RequestScoped} object. * * @param type Object type. * @param value Actual object to bind. * @return Current request. */ @Nonnull default Request set(final Class type, final Object value) { return set(TypeLiteral.get(type), value); } /** * Seed a {@link RequestScoped} object. * * @param type Seed type. * @param value Actual object to bind. * @return Current request. */ @Nonnull default Request set(final TypeLiteral type, final Object value) { return set(Key.get(type), value); } /** * Seed a {@link RequestScoped} object. * * @param key Seed key. * @param value Actual object to bind. * @return Current request. */ @Nonnull Request set(Key key, Object value); /** * Send a push promise frame to the client and push the resource identified by the given path. * * @param path Path of the resource to push. * @return This request. */ @Nonnull default Request push(final String path) { return push(path, ImmutableMap.of()); } /** * Send a push promise frame to the client and push the resource identified by the given path. * * @param path Path of the resource to push. * @param headers Headers to send. * @return This request. */ @Nonnull Request push(final String path, final Map headers); /** * Request timestamp. * * @return The time that the request was received. */ long timestamp(); }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy