diff --git a/src/CloudNative.CloudEvents.Amqp/AmqpExtensions.cs b/src/CloudNative.CloudEvents.Amqp/AmqpExtensions.cs
index 001b832..591e0af 100644
--- a/src/CloudNative.CloudEvents.Amqp/AmqpExtensions.cs
+++ b/src/CloudNative.CloudEvents.Amqp/AmqpExtensions.cs
@@ -1,4 +1,4 @@
-// Copyright (c) Cloud Native Foundation.
+// Copyright (c) Cloud Native Foundation.
// Licensed under the Apache 2.0 license.
// See LICENSE file in the project root for full license information.
@@ -8,6 +8,7 @@
using CloudNative.CloudEvents.Core;
using System;
using System.Collections.Generic;
+using System.Diagnostics.CodeAnalysis;
using System.IO;
using System.Net.Mime;
@@ -145,7 +146,7 @@ public static CloudEvent ToCloudEvent(
}
}
- private static bool HasCloudEventsContentType(Message message, out string? contentType)
+ private static bool HasCloudEventsContentType(Message message, [NotNullWhen(true)] out string? contentType)
{
contentType = message.Properties.ContentType?.ToString();
return MimeUtilities.IsCloudEventsContentType(contentType);
@@ -249,4 +250,4 @@ private static ApplicationProperties MapHeaders(CloudEvent cloudEvent, string pr
return applicationProperties;
}
}
-}
\ No newline at end of file
+}
diff --git a/src/CloudNative.CloudEvents.Amqp/CloudNative.CloudEvents.Amqp.csproj b/src/CloudNative.CloudEvents.Amqp/CloudNative.CloudEvents.Amqp.csproj
index 74e5833..2680861 100644
--- a/src/CloudNative.CloudEvents.Amqp/CloudNative.CloudEvents.Amqp.csproj
+++ b/src/CloudNative.CloudEvents.Amqp/CloudNative.CloudEvents.Amqp.csproj
@@ -12,6 +12,8 @@
+
+
diff --git a/src/CloudNative.CloudEvents.Avro/AvroEventFormatter.cs b/src/CloudNative.CloudEvents.Avro/AvroEventFormatter.cs
index 8360d8b..021a5ed 100644
--- a/src/CloudNative.CloudEvents.Avro/AvroEventFormatter.cs
+++ b/src/CloudNative.CloudEvents.Avro/AvroEventFormatter.cs
@@ -185,9 +185,9 @@ private static RecordSchema ParseEmbeddedSchema()
// will fail and that's okay since the type is useless without the proper schema.
using var sr = new StreamReader(typeof(AvroEventFormatter)
.Assembly
- .GetManifestResourceStream("CloudNative.CloudEvents.Avro.AvroSchema.json"));
+ .GetManifestResourceStream("CloudNative.CloudEvents.Avro.AvroSchema.json")!);
return (RecordSchema) Schema.Parse(sr.ReadToEnd());
}
}
-}
\ No newline at end of file
+}
diff --git a/src/CloudNative.CloudEvents.NewtonsoftJson/JsonEventFormatter.cs b/src/CloudNative.CloudEvents.NewtonsoftJson/JsonEventFormatter.cs
index 20c9150..3564135 100644
--- a/src/CloudNative.CloudEvents.NewtonsoftJson/JsonEventFormatter.cs
+++ b/src/CloudNative.CloudEvents.NewtonsoftJson/JsonEventFormatter.cs
@@ -1,4 +1,4 @@
-// Copyright (c) Cloud Native Foundation.
+// Copyright (c) Cloud Native Foundation.
// Licensed under the Apache 2.0 license.
// See LICENSE file in the project root for full license information.
@@ -345,7 +345,7 @@ protected virtual void DecodeStructuredModeDataBase64Property(JToken dataBase64T
{
throw new ArgumentException($"Structured mode property '{DataBase64PropertyName}' must be a string, when present.");
}
- cloudEvent.Data = Convert.FromBase64String((string?) dataBase64Token);
+ cloudEvent.Data = Convert.FromBase64String((string) dataBase64Token!);
}
///
@@ -498,7 +498,7 @@ private void WriteCloudEventForBatchOrStructuredMode(JsonWriter writer, CloudEve
///
/// The CloudEvent to infer the data content from. Must not be null.
/// The inferred data content type, or null if no inference is performed.
- protected override string? InferDataContentType(object data) => data is byte[]? null : JsonMediaType;
+ protected override string? InferDataContentType(object data) => data is byte[] ? null : JsonMediaType;
///
/// Encodes structured mode data within a CloudEvent, writing it to the specified .
@@ -524,7 +524,14 @@ protected virtual void EncodeStructuredModeData(CloudEvent cloudEvent, JsonWrite
}
else
{
- ContentType dataContentType = new ContentType(GetOrInferDataContentType(cloudEvent));
+ string? dataContentTypeText = GetOrInferDataContentType(cloudEvent);
+ // This would only happen in a derived class which overrides GetOrInferDataContentType further...
+ // This class infers application/json for anything other than byte arrays.
+ if (dataContentTypeText is null)
+ {
+ throw new ArgumentException("Data content type cannot be inferred");
+ }
+ ContentType dataContentType = new ContentType(dataContentTypeText);
if (IsJsonMediaType(dataContentType.MediaType))
{
writer.WritePropertyName(DataPropertyName);
@@ -710,4 +717,4 @@ protected override void DecodeStructuredModeDataProperty(JToken dataToken, Cloud
protected override void DecodeStructuredModeDataBase64Property(JToken dataBase64Token, CloudEvent cloudEvent) =>
throw new ArgumentException($"Data unexpectedly represented using '{DataBase64PropertyName}' within structured mode CloudEvent.");
}
-}
\ No newline at end of file
+}
diff --git a/src/CloudNative.CloudEvents/CloudEventFormatterAttribute.cs b/src/CloudNative.CloudEvents/CloudEventFormatterAttribute.cs
index 689abd8..00fd06c 100644
--- a/src/CloudNative.CloudEvents/CloudEventFormatterAttribute.cs
+++ b/src/CloudNative.CloudEvents/CloudEventFormatterAttribute.cs
@@ -56,7 +56,7 @@ public CloudEventFormatterAttribute(Type formatterType) =>
throw new ArgumentException($"The {nameof(CloudEventFormatterAttribute)} on type {targetType} has no converter type specified.", nameof(targetType));
}
- object instance;
+ object? instance;
try
{
instance = Activator.CreateInstance(formatterType);
diff --git a/src/CloudNative.CloudEvents/Core/BinaryDataUtilities.cs b/src/CloudNative.CloudEvents/Core/BinaryDataUtilities.cs
index 29dfa1d..c1915a9 100644
--- a/src/CloudNative.CloudEvents/Core/BinaryDataUtilities.cs
+++ b/src/CloudNative.CloudEvents/Core/BinaryDataUtilities.cs
@@ -65,7 +65,7 @@ public static ReadOnlyMemory ToReadOnlyMemory(Stream stream)
public static MemoryStream AsStream(ReadOnlyMemory memory)
{
var segment = GetArraySegment(memory);
- return new MemoryStream(segment.Array, segment.Offset, segment.Count, false);
+ return new MemoryStream(segment.Array!, segment.Offset, segment.Count, false);
}
///
@@ -79,7 +79,7 @@ public static string GetString(ReadOnlyMemory memory, Encoding encoding)
// TODO: If we introduce an additional netstandard2.1 target, we can use encoding.GetString(memory.Span)
var segment = GetArraySegment(memory);
- return encoding.GetString(segment.Array, segment.Offset, segment.Count);
+ return encoding.GetString(segment.Array!, segment.Offset, segment.Count);
}
///
@@ -92,7 +92,7 @@ public static async Task CopyToStreamAsync(ReadOnlyMemory source, Stream d
{
Validation.CheckNotNull(destination, nameof(destination));
var segment = GetArraySegment(source);
- await destination.WriteAsync(segment.Array, segment.Offset, segment.Count).ConfigureAwait(false);
+ await destination.WriteAsync(segment.Array!, segment.Offset, segment.Count).ConfigureAwait(false);
}
///
@@ -108,13 +108,14 @@ public static byte[] AsArray(ReadOnlyMemory memory)
var segment = GetArraySegment(memory);
// We probably don't actually need to check the offset: if the count is the same as the length,
// I can't see how the offset can be non-zero. But it doesn't *hurt* as a check.
- return segment.Offset == 0 && segment.Count == segment.Array.Length
+ return segment.Offset == 0 && segment.Count == segment.Array!.Length
? segment.Array
: memory.ToArray();
}
+ // Note: when this returns, the Array property of the returned segment is guaranteed not to be null.
private static ArraySegment GetArraySegment(ReadOnlyMemory memory) =>
- MemoryMarshal.TryGetArray(memory, out var segment)
+ MemoryMarshal.TryGetArray(memory, out var segment) && segment.Array is not null
? segment
: new ArraySegment(memory.ToArray());
}
diff --git a/src/CloudNative.CloudEvents/Core/MimeUtilities.cs b/src/CloudNative.CloudEvents/Core/MimeUtilities.cs
index b660086..d31d776 100644
--- a/src/CloudNative.CloudEvents/Core/MimeUtilities.cs
+++ b/src/CloudNative.CloudEvents/Core/MimeUtilities.cs
@@ -1,8 +1,9 @@
-// Copyright 2021 Cloud Native Foundation.
+// Copyright 2021 Cloud Native Foundation.
// Licensed under the Apache 2.0 license.
// See LICENSE file in the project root for full license information.
using System;
+using System.Diagnostics.CodeAnalysis;
using System.Net.Http.Headers;
using System.Net.Mime;
using System.Text;
@@ -57,7 +58,7 @@ public static Encoding GetEncoding(ContentType? contentType) =>
var header = new MediaTypeHeaderValue(contentType.MediaType);
foreach (string parameterName in contentType.Parameters.Keys)
{
- header.Parameters.Add(new NameValueHeaderValue(parameterName, contentType.Parameters[parameterName].ToString()));
+ header.Parameters.Add(new NameValueHeaderValue(parameterName, contentType.Parameters[parameterName]));
}
return header;
}
@@ -76,7 +77,7 @@ public static Encoding GetEncoding(ContentType? contentType) =>
///
/// The content type to check. May be null, in which case the result is false.
/// true if the given content type denotes a (non-batch) CloudEvent; false otherwise
- public static bool IsCloudEventsContentType(string? contentType) =>
+ public static bool IsCloudEventsContentType([NotNullWhen(true)] string? contentType) =>
contentType is string &&
contentType.StartsWith(MediaType, StringComparison.InvariantCultureIgnoreCase) &&
!contentType.StartsWith(BatchMediaType, StringComparison.InvariantCultureIgnoreCase);
@@ -86,7 +87,7 @@ contentType is string &&
///
/// The content type to check. May be null, in which case the result is false.
/// true if the given content type represents a CloudEvent batch; false otherwise
- public static bool IsCloudEventsBatchContentType(string? contentType) =>
+ public static bool IsCloudEventsBatchContentType([NotNullWhen(true)] string? contentType) =>
contentType is string && contentType.StartsWith(BatchMediaType, StringComparison.InvariantCultureIgnoreCase);
}
}
diff --git a/src/CloudNative.CloudEvents/Http/HttpClientExtensions.cs b/src/CloudNative.CloudEvents/Http/HttpClientExtensions.cs
index b439ffc..2ac175c 100644
--- a/src/CloudNative.CloudEvents/Http/HttpClientExtensions.cs
+++ b/src/CloudNative.CloudEvents/Http/HttpClientExtensions.cs
@@ -1,10 +1,11 @@
-// Copyright (c) Cloud Native Foundation.
+// Copyright (c) Cloud Native Foundation.
// Licensed under the Apache 2.0 license.
// See LICENSE file in the project root for full license information.
using CloudNative.CloudEvents.Core;
using System;
using System.Collections.Generic;
+using System.Diagnostics.CodeAnalysis;
using System.Linq;
using System.Net.Http;
using System.Net.Http.Headers;
@@ -130,7 +131,7 @@ public static Task ToCloudEventAsync(
return ToCloudEventInternalAsync(httpRequestMessage.Headers, httpRequestMessage.Content, formatter, extensionAttributes, nameof(httpRequestMessage));
}
- private static async Task ToCloudEventInternalAsync(HttpHeaders headers, HttpContent content,
+ private static async Task ToCloudEventInternalAsync(HttpHeaders headers, HttpContent? content,
CloudEventFormatter formatter, IEnumerable? extensionAttributes, string paramName)
{
Validation.CheckNotNull(formatter, nameof(formatter));
@@ -142,7 +143,7 @@ private static async Task ToCloudEventInternalAsync(HttpHeaders head
}
else
{
- string? versionId = MaybeGetVersionId(headers) ?? MaybeGetVersionId(content.Headers);
+ string? versionId = MaybeGetVersionId(headers) ?? MaybeGetVersionId(content?.Headers);
if (versionId is null)
{
throw new ArgumentException($"Request does not represent a CloudEvent. It has neither a {HttpUtilities.SpecVersionHttpHeader} header, nor a suitable content type.", nameof(paramName));
@@ -151,7 +152,8 @@ private static async Task ToCloudEventInternalAsync(HttpHeaders head
?? throw new ArgumentException($"Unknown CloudEvents spec version '{versionId}'", paramName);
var cloudEvent = new CloudEvent(version, extensionAttributes);
- foreach (var header in headers.Concat(content.Headers))
+ var allHeaders = content is null ? headers : headers.Concat(content.Headers);
+ foreach (var header in allHeaders)
{
string? attributeName = HttpUtilities.GetAttributeNameFromHeaderName(header.Key);
if (attributeName is null || attributeName == CloudEventsSpecVersion.SpecVersionAttribute.Name)
@@ -231,7 +233,7 @@ public static Task> ToCloudEventBatchAsync(
return ToCloudEventBatchInternalAsync(httpRequestMessage.Content, formatter, extensionAttributes, nameof(httpRequestMessage));
}
- private static async Task> ToCloudEventBatchInternalAsync(HttpContent content,
+ private static async Task> ToCloudEventBatchInternalAsync(HttpContent? content,
CloudEventFormatter formatter, IEnumerable? extensionAttributes, string paramName)
{
Validation.CheckNotNull(formatter, nameof(formatter));
@@ -331,16 +333,16 @@ public static HttpContent ToHttpContent(this IReadOnlyList cloudEven
}
private static ByteArrayContent ToByteArrayContent(ReadOnlyMemory content) =>
- MemoryMarshal.TryGetArray(content, out var segment)
+ MemoryMarshal.TryGetArray(content, out var segment) && segment.Array is not null
? new ByteArrayContent(segment.Array, segment.Offset, segment.Count)
// TODO: Just throw?
: new ByteArrayContent(content.ToArray());
// TODO: This would include "application/cloudeventsarerubbish" for example...
- private static bool HasCloudEventsContentType(HttpContent content) =>
+ private static bool HasCloudEventsContentType([NotNullWhen(true)] HttpContent? content) =>
MimeUtilities.IsCloudEventsContentType(content?.Headers?.ContentType?.MediaType);
- private static bool HasCloudEventsBatchContentType(HttpContent content) =>
+ private static bool HasCloudEventsBatchContentType([NotNullWhen(true)] HttpContent? content) =>
MimeUtilities.IsCloudEventsBatchContentType(content?.Headers?.ContentType?.MediaType);
private static string? MaybeGetVersionId(HttpHeaders? headers) =>
@@ -348,4 +350,4 @@ private static bool HasCloudEventsBatchContentType(HttpContent content) =>
? headers.GetValues(HttpUtilities.SpecVersionHttpHeader).First()
: null;
}
-}
\ No newline at end of file
+}
diff --git a/src/CloudNative.CloudEvents/Http/HttpListenerExtensions.cs b/src/CloudNative.CloudEvents/Http/HttpListenerExtensions.cs
index 1fce6ba..0d830d6 100644
--- a/src/CloudNative.CloudEvents/Http/HttpListenerExtensions.cs
+++ b/src/CloudNative.CloudEvents/Http/HttpListenerExtensions.cs
@@ -179,7 +179,7 @@ private static async Task ToCloudEventAsyncImpl(HttpListenerRequest
}
else
{
- string versionId = httpListenerRequest.Headers[HttpUtilities.SpecVersionHttpHeader];
+ string? versionId = httpListenerRequest.Headers[HttpUtilities.SpecVersionHttpHeader];
if (versionId is null)
{
throw new ArgumentException($"Request does not represent a CloudEvent. It has neither a {HttpUtilities.SpecVersionHttpHeader} header, nor a suitable content type.", nameof(httpListenerRequest));
@@ -191,12 +191,18 @@ private static async Task ToCloudEventAsyncImpl(HttpListenerRequest
var headers = httpListenerRequest.Headers;
foreach (var key in headers.AllKeys)
{
+ // It would be highly unusual for either the key or the value to be null, but
+ // the contract claims it's possible. Skip it if so.
+ if (key is null || headers[key] is not string headerValue)
+ {
+ continue;
+ }
string? attributeName = HttpUtilities.GetAttributeNameFromHeaderName(key);
if (attributeName is null || attributeName == CloudEventsSpecVersion.SpecVersionAttribute.Name)
{
continue;
}
- string attributeValue = HttpUtilities.DecodeHeaderValue(headers[key]);
+ string attributeValue = HttpUtilities.DecodeHeaderValue(headerValue);
cloudEvent.SetAttributeFromString(attributeName, attributeValue);
}