Skip to content

Security requirement not serialized correctly #2300

@martincostello

Description

@martincostello

Describe the bug

I'm migrating Swashbuckle.AspNetCore to Microsoft.OpenApi 2.0.0-preview7 (as that's the version used by ASP.NET Core 10 preview 2), and an existing regression test we have for OAuth2 is failing due to the security definition not being deserialized correctly.

Distilling things down, this is what we expect:

{
  "openapi": "3.0.4",
  "info": { },
  "paths": { },
  "components": {
    "securitySchemes": {
      "oauth2": {
        "type": "oauth2",
        "flows": {
          "authorizationCode": {
            "authorizationUrl": "/auth-server/connect/authorize",
            "tokenUrl": "/auth-server/connect/token",
            "scopes": {
              "readAccess": "Access read operations",
              "writeAccess": "Access write operations"
            }
          }
        }
      }
    }
  },
  "security": [
    {
      "oauth2": [
        "readAccess",
        "writeAccess"
      ]
    }
  ]
}

However, what we get is this:

{
  "openapi": "3.0.4",
  "info": { },
  "paths": { },
  "components": {
    "securitySchemes": {
      "oauth2": {
        "type": "oauth2",
        "flows": {
          "authorizationCode": {
            "authorizationUrl": "/auth-server/connect/authorize",
            "tokenUrl": "/auth-server/connect/token",
            "scopes": {
              "readAccess": "Access read operations",
              "writeAccess": "Access write operations"
            }
          }
        }
      }
    }
  },
  "security": [
    { }
  ]
}

As far as I can tell, the Target is returning null (so nothing is emitted), but I don't understand why it isn't found via the reference into the host document:

foreach (var securitySchemeAndScopesValuePair in this.Where(static p => p.Key?.Target is not null))

It might be that there's an additional code migration I need to do here for v2, but if there is I haven't been able to spot it.

OpenApi File To Reproduce

[Fact]
public void CanSerializeSecuritySchemes()
{
    var document = new OpenApiDocument();

    document.Components ??= new();
    document.Components.SecuritySchemes.Add("oauth2", new OpenApiSecurityScheme
    {
        Type = SecuritySchemeType.OAuth2,
        Flows = new OpenApiOAuthFlows
        {
            AuthorizationCode = new OpenApiOAuthFlow
            {
                AuthorizationUrl = new Uri("/auth-server/connect/authorize", UriKind.Relative),
                TokenUrl = new Uri("/auth-server/connect/token", UriKind.Relative),
                Scopes = new Dictionary<string, string>
                {
                    { "readAccess", "Access read operations" },
                    { "writeAccess", "Access write operations" }
                }
            }
        }
    });

    document.SecurityRequirements.Add(new OpenApiSecurityRequirement
    {
        {
            new OpenApiSecuritySchemeReference("oauth2", document),
            ["readAccess", "writeAccess"]
        }
    });

    using var writer = new StringWriter();
    var jsonWriter = new OpenApiJsonWriter(writer);
    document.SerializeAsV3(jsonWriter);

    var json = writer.ToString();
}

Expected behavior

The security scheme and the required scopes are serialized to the OpenAPI document.

Metadata

Metadata

Assignees

No one assigned

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions