Welcome to our new JIRA instance! We migrated all OpenMRS ID accounts and issues to this new cloud instance. Please use your_openmrsid@id.openmrs.org as your e-mail to sign in.

REST "next" and "previous" links can return incorrect protocal

Description

When generating the “next” and “previous” links for a REST request, the module relies on the getRequestUrl() method:

https://github.com/openmrs/openmrs-module-webservices.rest/blob/master/omod-common/src/main/java/org/openmrs/module/webservices/rest/web/RequestContext.java#L173

However, when the OpenMRS server is sitting behind a load balancer (or potentially behind a reverse-proxy) the generated protocol may be incorrect… specifically the URL may use the “http” protocol instead of “https”.

See:

https://docs.oracle.com/javaee/6/api/javax/servlet/http/HttpServletRequest.html#getRequestURL()

https://stackoverflow.com/questions/29469929/why-does-request-getrequesturl-return-non-https-url

We should have more robust method to generate the URL so as to not encounter this issue.

(Thoughts … we are running into this problem on a Prod server, assumedly because we are serving them behind Caddy which is operating as a reverse proxy).

Activity

Show:

Ian Bacher February 21, 2025 at 8:05 PM

Yes, exactly. It makes sense to put this in REST, but it might even be nice as a utility method in Core (I think there’s a WebUtils class or something similar).

Mark Goodrich February 21, 2025 at 6:45 PM

Yes, it looks like we basically want to make a utility method out of this block of code?

String scheme; if (StringUtils.isNotBlank(request.getHeader("X-Forwarded-Proto"))) { scheme = request.getHeader("X-Forwarded-Proto"); } else { scheme = request.getScheme(); } String host; if (StringUtils.isNotBlank(request.getHeader("X-Forwarded-Host"))) { host = request.getHeader("X-Forwarded-Host"); } else { host = request.getServerName(); } int serverPort = request.getServerPort(); String port; if (StringUtils.isNotBlank(request.getHeader("X-Forwarded-Port"))) { port = ":" + request.getHeader("X-Forwarded-Port"); if (scheme.equalsIgnoreCase("http") && port.equals(":80")) { port = ""; } else if (scheme.equalsIgnoreCase("https") && port.equals(":443")) { port = ""; } } else if ("http".equalsIgnoreCase(request.getScheme())) { port = serverPort == 80 ? "" : ":" + serverPort; } else if ("https".equalsIgnoreCase(request.getScheme())) { port = serverPort == 443 ? "" : ":" + serverPort; } else { port = ":" + serverPort; } FhirVersionUtils.FhirVersion fhirVersion = FhirVersionUtils.getFhirVersion(request); return scheme + "://" + host + port + StringUtils.defaultString(request.getContextPath())

Ian Bacher February 21, 2025 at 4:37 PM

So, to address the same thing in FHIR2, we introduced this class (an AddressStrategy is a HAPI thing, but it’s just pulling things out of the HttpServletRequest headers). It relies on semi-standardized headers most proxy’s either should send or can be configured to send (for, e.g., our O3-hosted systems there are actually two separate nginx proxies in front of Tomcat).

The one thing it doesn’t handle is the now-standardized Forwarded header, mostly because there’s no built-in for that in nginx.

And there are definitely a few things in there that can be skipped (as a fallback, we introduced a GP so implementations could manually set what the correct URL prefix is and our requests are ultimately forwarded from /ws/fhir2/R4 to /ms/fhir2Servlet, but I we wanted the links to read /ws/fhir2/R4.

Mark Goodrich February 20, 2025 at 11:16 PM

Assumedly we want to generate the protocol and server name ourselves (which we already stored somewhere, I believe) and then append the serlvet path and query string.

Details

Assignee

Reporter

Priority

Created February 20, 2025 at 11:03 PM
Updated February 21, 2025 at 8:05 PM

Flag notifications