Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Refactor HttpMethod from enum to class #27697

Closed
poutsma opened this issue Nov 18, 2021 · 8 comments
Closed

Refactor HttpMethod from enum to class #27697

poutsma opened this issue Nov 18, 2021 · 8 comments
Assignees
Labels
in: web Issues in web modules (web, webmvc, webflux, websocket) type: enhancement A general enhancement
Milestone

Comments

@poutsma
Copy link
Contributor

poutsma commented Nov 18, 2021

According to the HTTP specification, the HTTP method is not limited to the well known set (GET, HEAD, PUT, POST, etc.), but can also be an "extension-method". Well known extensions include WebDAV, which added methods like LOCK, COPY, and MOVE.

In Spring Framework, HTTP methods are enumerated in HttpMethod. Because this type is an Java enum, Spring framework needs several workarounds, to allow for HTTP methods not in the enum, such as having both HttpRequest::getMethod as well as HttpRequest::getMethodValue.

If we change HttpMethod from enum to class, we no longer need these workarounds. If we make sure that the new class has the same methods that java.lang.Enum exposes, and given that upgrading to 6.0 requires a recompilation anyway, I believe that now is the time to make this long overdue change.

Note that this issue does not include support for non-standard HTTP (i.e. WebDAV) methods in Spring MVC and/or WebFlux.

@poutsma poutsma added the in: web Issues in web modules (web, webmvc, webflux, websocket) label Nov 18, 2021
@poutsma poutsma added this to the 6.0 M1 milestone Nov 18, 2021
@poutsma poutsma self-assigned this Nov 18, 2021
@jhoeller jhoeller added the type: enhancement A general enhancement label Nov 24, 2021
@quaff
Copy link
Contributor

quaff commented Nov 25, 2021

such as having both HttpMethod::getMethod as well as HttpRequest::getMethodValue.

HttpMethod::getMethod should be HttpRequest::getMethod

@poutsma
Copy link
Contributor Author

poutsma commented Nov 25, 2021

@quaff Fixed, thanks!

poutsma added a commit that referenced this issue Nov 30, 2021
This commit contains changes made because HttpMethod changed from enum
to class.

See gh-27697
poutsma added a commit that referenced this issue Nov 30, 2021
This commit makes sure that HttpMethod::resolve uses HttpMethod::valueOf
and returns an HttpMethod for non-standard methods.

See gh-27697
@quaff
Copy link
Contributor

quaff commented Dec 1, 2021

Can we use extension-method for RequestMapping?

@poutsma
Copy link
Contributor Author

poutsma commented Dec 1, 2021

Can we use extension-method for RequestMapping?

I am not sure what you mean by that, can you elaborate? Because annotations can refer to enum elements but not classes, RequestMapping uses the RequestMethod enumeration.

@quaff
Copy link
Contributor

quaff commented Dec 2, 2021

Can we use extension-method for RequestMapping?

I am not sure what you mean by that, can you elaborate? Because annotations can refer to enum elements but not classes, RequestMapping uses the RequestMethod enumeration.

I mean should RequestMapping introduce String[] methodValue() to supports extension-method?

@poutsma
Copy link
Contributor Author

poutsma commented Dec 2, 2021

I mean should RequestMapping introduce String[] methodValue() to supports extension-method?

RequestMapping uses RequestMethod, and that's fine the way it is. As I wrote in the description, we have no intention of supporting non-standard HTTP methods. HttpMethod is a lower-level component that is used for our HTTP abstraction, and that did need support for non-standard methods.

poutsma added a commit that referenced this issue Feb 17, 2023
This commit ensures that WebFlux's RequestMethodsRequestCondition
supports HTTP methods that are not in the RequestMethod enum.

- RequestMethod::resolve is introduced, to convert from a HttpMethod
(name) to enum values.
- RequestMethod::asHttpMethod is introduced, to convert from enum value
to HttpMethod.
- HttpMethod::valueOf replaced Map-based lookup to a switch statement
- Enabled tests that check for WebDAV methods

See gh-27697
Closes gh-29981
@tamizh-m
Copy link

Hi @poutsma , After upgrading to Spring 6, I am unable to serialize and deserialize 'HttpMethod' using 'ObjectMapper' because the class does not have a public constructor or a getter for 'name' attribute. This issue did not arise previously when HttpMethod was an enum. Would be helpful if you can provide a workaround for this?

@poutsma
Copy link
Contributor Author

poutsma commented May 28, 2024

@tamizh-m Please file a new issue. This issue is closed.

rahul-chekuri added a commit to rahul-chekuri/orca that referenced this issue Jan 5, 2025
… is recently refactored as a final class from Enum.

```
WebhookServiceSpec > Webhook is being called with correct parameters > com.netflix.spinnaker.orca.webhook.service.WebhookServiceSpec.Webhook is being called with correct parameters [payload: [payload1:Hello, payload2:World!], customHeaders: [:], #1] FAILED
    java.lang.IllegalArgumentException: No serializer found for class org.springframework.http.HttpMethod and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.netflix.spinnaker.orca.pipeline.model.StageContext["method"])
        at com.fasterxml.jackson.databind.ObjectMapper.valueToTree(ObjectMapper.java:3442)
        at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.contextToNode(StageExecutionImpl.java:680)
        at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.mapTo(StageExecutionImpl.java:623)
        at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.mapTo(StageExecutionImpl.java:608)
        at com.netflix.spinnaker.orca.webhook.service.WebhookService.getRestTemplateData(WebhookService.java:117)
        at com.netflix.spinnaker.orca.webhook.service.WebhookService.callWebhook(WebhookService.java:65)
        at com.netflix.spinnaker.orca.webhook.service.WebhookServiceSpec.Webhook is being called with correct parameters(WebhookServiceSpec.groovy:99)

        Caused by:
        com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class org.springframework.http.HttpMethod and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.netflix.spinnaker.orca.pipeline.model.StageContext["method"])

WebhookServiceSpec > Should accept PATCH request FAILED
    Expected no exception to be thrown, but got 'java.lang.IllegalArgumentException'
        at app//spock.lang.Specification.noExceptionThrown(Specification.java:123)
        at com.netflix.spinnaker.orca.webhook.service.WebhookServiceSpec.Should accept PATCH request(WebhookServiceSpec.groovy:201)

        Caused by:
        java.lang.IllegalArgumentException: No serializer found for class org.springframework.http.HttpMethod and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.netflix.spinnaker.orca.pipeline.model.StageContext["method"])
            at com.fasterxml.jackson.databind.ObjectMapper.valueToTree(ObjectMapper.java:3442)
            at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.contextToNode(StageExecutionImpl.java:680)
            at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.mapTo(StageExecutionImpl.java:623)
            at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.mapTo(StageExecutionImpl.java:608)
            at com.netflix.spinnaker.orca.webhook.service.WebhookService.getRestTemplateData(WebhookService.java:117)
            at com.netflix.spinnaker.orca.webhook.service.WebhookService.callWebhook(WebhookService.java:65)
            at com.netflix.spinnaker.orca.webhook.service.WebhookServiceSpec.Should accept PATCH request(WebhookServiceSpec.groovy:198)

            Caused by:
            com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class org.springframework.http.HttpMethod and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.netflix.spinnaker.orca.pipeline.model.StageContext["method"])
                at app//com.fasterxml.jackson.databind.exc.InvalidDefinitionException.from(InvalidDefinitionException.java:77)
                at app//com.fasterxml.jackson.databind.SerializerProvider.reportBadDefinition(SerializerProvider.java:1306)
                at app//com.fasterxml.jackson.databind.DatabindContext.reportBadDefinition(DatabindContext.java:408)
                at app//com.fasterxml.jackson.databind.ser.impl.UnknownSerializer.failForEmpty(UnknownSerializer.java:53)
                at app//com.fasterxml.jackson.databind.ser.impl.UnknownSerializer.serialize(UnknownSerializer.java:30)
                at app//com.fasterxml.jackson.databind.ser.std.MapSerializer.serializeOptionalFields(MapSerializer.java:869)
                at app//com.fasterxml.jackson.databind.ser.std.MapSerializer.serializeWithoutTypeInfo(MapSerializer.java:760)
                at app//com.fasterxml.jackson.databind.ser.std.MapSerializer.serialize(MapSerializer.java:720)
                at app//com.fasterxml.jackson.databind.ser.std.MapSerializer.serialize(MapSerializer.java:35)
                at app//com.fasterxml.jackson.databind.ser.DefaultSerializerProvider._serialize(DefaultSerializerProvider.java:480)
                at app//com.fasterxml.jackson.databind.ser.DefaultSerializerProvider.serializeValue(DefaultSerializerProvider.java:319)
                at app//com.fasterxml.jackson.databind.ObjectMapper.valueToTree(ObjectMapper.java:3437)
                ... 6 more

MtlsConfigurationPemTest > mTLSConnectivityPemTest() FAILED
    java.lang.IllegalArgumentException: No serializer found for class org.springframework.http.HttpMethod and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.netflix.spinnaker.orca.pipeline.model.StageContext["method"])
        at com.fasterxml.jackson.databind.ObjectMapper.valueToTree(ObjectMapper.java:3442)
        at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.contextToNode(StageExecutionImpl.java:680)
        at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.mapTo(StageExecutionImpl.java:623)
        at com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl.mapTo(StageExecutionImpl.java:608)
        at com.netflix.spinnaker.orca.webhook.service.WebhookService.getRestTemplateData(WebhookService.java:117)
        at com.netflix.spinnaker.orca.webhook.service.WebhookService.callWebhook(WebhookService.java:65)
        at com.netflix.spinnaker.orca.webhook.config.MtlsConfigurationPemTest.mTLSConnectivityPemTest(MtlsConfigurationPemTest.java:86)

        Caused by:
        com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class org.springframework.http.HttpMethod and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS) (through reference chain: com.netflix.spinnaker.orca.pipeline.model.StageContext["method"])
            at app//com.fasterxml.jackson.databind.exc.InvalidDefinitionException.from(InvalidDefinitionException.java:77)
            at app//com.fasterxml.jackson.databind.SerializerProvider.reportBadDefinition(SerializerProvider.java:1306)
            at
```

Reference:
spring-projects/spring-framework#27697
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
in: web Issues in web modules (web, webmvc, webflux, websocket) type: enhancement A general enhancement
Projects
None yet
Development

No branches or pull requests

4 participants