-
Notifications
You must be signed in to change notification settings - Fork 68
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
SRU2024 v10.2 #115
base: main
Are you sure you want to change the base?
SRU2024 v10.2 #115
Conversation
… SWIFT and ISO 20022
@coderabbitai review |
…ized from XMLGregorianCalendar objects
…cluding new messages such as trck.001.001.03 (#121)
Warning Rate limit exceeded@zubri has exceeded the limit for the number of commits or files that can be reviewed per hour. Please wait 19 minutes and 1 seconds before requesting another review. ⌛ How to resolve this issue?After the wait time has elapsed, a review can be triggered using the We recommend that you space out your commits to avoid hitting the rate limit. 🚦 How do rate limits work?CodeRabbit enforces hourly rate limits for each developer per organization. Our paid plans have higher rate limits than the trial, open-source and free plans. In all cases, we re-allow further reviews after a brief timeout. Please see our FAQ for further information. 📒 Files selected for processing (2)
📝 WalkthroughWalkthroughThis pull request introduces a significant update to the Prowide ISO 20022 library, transitioning from Java 8 to Java 11 and upgrading the XML and JSON handling capabilities. The changes primarily focus on modernizing date-time representations, migrating from Changes
Sequence DiagramsequenceDiagram
participant Client
participant AbstractMX
participant JsonAdapter
participant OffsetDateTime
Client->>AbstractMX: toJson()
AbstractMX->>JsonAdapter: serialize(OffsetDateTime)
JsonAdapter->>OffsetDateTime: extract date/time components
JsonAdapter-->>AbstractMX: JSON representation
AbstractMX-->>Client: JSON string
Client->>AbstractMX: fromJson(jsonString)
AbstractMX->>JsonAdapter: deserialize(JSON)
JsonAdapter->>OffsetDateTime: construct from JSON components
JsonAdapter-->>AbstractMX: OffsetDateTime instance
AbstractMX-->>Client: Reconstructed object
This diagram illustrates the high-level process of JSON serialization and deserialization using the new adapters, showcasing how date-time objects are converted between JSON and Java objects. Possibly related PRs
Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media? 🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments. CodeRabbit Commands (Invoked using PR comments)
Other keywords and placeholders
CodeRabbit Configuration File (
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Caution
Inline review comments failed to post. This is likely due to GitHub's limits when posting large numbers of comments.
🛑 Comments failed to post (124)
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeJsonAdapter.java (5)
18-18: 🧹 Nitpick (assertive)
Avoid using wildcard imports.
Replace the wildcard import with explicit class imports to improve code readability and prevent the inclusion of unnecessary classes.
68-71: 🛠️ Refactor suggestion
Avoid using system default time zone when offset is missing.
Using
ZoneOffset.systemDefault()
may lead to inconsistent behavior across different environments due to time zone differences and daylight saving time adjustments. Consider defining a default offset or explicitly handling cases where the offset is absent.Apply this diff to set a default offset (e.g., UTC):
- offsetTime = - localTime.atOffset(ZoneOffset.systemDefault().getRules().getStandardOffset(Instant.now())); + offsetTime = localTime.atOffset(ZoneOffset.UTC);📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.LocalTime localTime = LocalTime.of(timeDTO.time.hour, timeDTO.time.minute, timeDTO.time.second, nano); offsetTime = localTime.atOffset(ZoneOffset.UTC); }
74-77: 🧹 Nitpick (assertive)
Catch specific exceptions and log at an appropriate level.
Catching general
Exception
can obscure the root cause of issues. Catch more specific exceptions likeJsonSyntaxException
. Also, logging atLevel.FINEST
may cause errors to be missed; consider usingLevel.WARNING
orLevel.SEVERE
for better visibility.Apply this diff to improve error handling:
-} catch (Exception e) { - log.log(Level.FINEST, "Cannot parse JSON into OffsetTime: " + e.getMessage(), e); +} catch (JsonSyntaxException e) { + log.log(Level.SEVERE, "Cannot parse JSON into OffsetTime: " + e.getMessage(), e);Committable suggestion skipped: line range outside the PR's diff.
35-35: 🧹 Nitpick (assertive)
Make the 'gson' instance static final.
Declaring the
gson
instance asstatic final
avoids creating multiple instances and promotes reusability.Apply this diff to make the change:
-public class OffsetTimeJsonAdapter implements JsonSerializer<OffsetTime>, JsonDeserializer<OffsetTime> { - private static final Logger log = Logger.getLogger(OffsetTimeJsonAdapter.class.getName()); - private final Gson gson = new Gson(); +public class OffsetTimeJsonAdapter implements JsonSerializer<OffsetTime>, JsonDeserializer<OffsetTime> { + private static final Logger log = Logger.getLogger(OffsetTimeJsonAdapter.class.getName()); + + private static final Gson gson = new Gson();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.public class OffsetTimeJsonAdapter implements JsonSerializer<OffsetTime>, JsonDeserializer<OffsetTime> { private static final Logger log = Logger.getLogger(OffsetTimeJsonAdapter.class.getName()); private static final Gson gson = new Gson();
44-44: 🧹 Nitpick (assertive)
Remove unnecessary null check on 'offsetTime.getOffset()'.
The
getOffset()
method ofOffsetTime
always returns a non-null value. The null check is redundant and can be safely removed.Apply this diff to simplify the code:
-if (offsetTime.getOffset() != null) { timeOffsetDTO.offset = new OffsetDTO(); timeOffsetDTO.offset.totalSeconds = offsetTime.getOffset().getTotalSeconds(); -}Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/adapters/ZuluOffsetDateTimeAdapter.java (3)
98-115:
⚠️ Potential issueAvoid using system default time zone in date-time parsing
Using
ZoneId.systemDefault()
in theparseOffsetDateTime
method can lead to inconsistent behavior across different environments due to varying system time zones. It's advisable to use a consistent time zone, such as UTC, when convertingLocalDateTime
toOffsetDateTime
.Apply this diff to fix the issue:
- return localDateTime.atZone(ZoneId.systemDefault()).toOffsetDateTime(); + return localDateTime.atOffset(ZoneOffset.UTC);This ensures that the parsed
LocalDateTime
is consistently interpreted in UTC, preventing unexpected time shifts based on the system's default time zone.📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private static OffsetDateTime parseOffsetDateTime(String value) { if (value == null) { return null; } try { return OffsetDateTime.parse(value); } catch (DateTimeParseException e) { log.log(Level.FINEST, "Error parsing OffsetDateTime: " + e.getMessage()); try { // Attempt to parse as LocalDateTime and assume system default time zone LocalDateTime localDateTime = LocalDateTime.parse(value); return localDateTime.atOffset(ZoneOffset.UTC); } catch (DateTimeParseException e2) { log.log(Level.FINEST, "Error parsing LocalDateTime: " + e2.getMessage()); return null; } } }
43-43: 🧹 Nitpick (assertive)
Correct the
@since
tag formatting in JavaDocThe
@since
tag has a formatting issue. It should reflect a valid version number. Replace the comma with a period to match standard versioning conventions.Apply this diff to fix the issue:
- * @since 9.4,5 + * @since 9.4.5📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.* @since 9.4.5
83-96: 🛠️ Refactor suggestion
Specify exceptions instead of throwing generic
Exception
Declaring
throws Exception
in themarshal
method signature is too broad. It can make exception handling cumbersome for callers. It's better to handle exceptions within the method or specify the exact exceptions that can be thrown.Apply this diff to refine the exception handling:
- public String marshal(OffsetDateTime offsetDateTime) throws Exception { + public String marshal(OffsetDateTime offsetDateTime) {Since the code uses
DateTimeFormatter.format()
, which can throw anUnsupportedTemporalTypeException
orDateTimeException
, consider handling these exceptions within the method if necessary.Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/adapters/v9/V9DateTimeJsonAdapter.java (5)
62-64: 🧹 Nitpick (assertive)
Remove unnecessary null check for
OffsetDateTime.getOffset()
The method
OffsetDateTime.getOffset()
never returnsnull
, so the null check at line 62 is unnecessary. You can simplify the code by removing theif
condition.Apply this diff to simplify the code:
- if (offsetDateTime.getOffset() != null) { xMLGregorianCalendarDTO.timezone = offsetDateTime.getOffset().getTotalSeconds() / 60; - }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.xMLGregorianCalendarDTO.timezone = offsetDateTime.getOffset().getTotalSeconds() / 60;
59-60: 🧹 Nitpick (assertive)
Preserve nanosecond precision in
fractionalSecond
during serializationCurrently, the fractional seconds are rounded to three decimal places (milliseconds) using
RoundingMode.DOWN
, which may lead to loss of precision for timestamps with higher precision. Consider preserving the full nanosecond precision by adjusting the scale and rounding mode.Apply this diff to retain full nanosecond precision:
- xMLGregorianCalendarDTO.fractionalSecond = BigDecimal.valueOf(offsetDateTime.getNano()) - .divide(BigDecimal.valueOf(1_000_000_000), 3, RoundingMode.DOWN); + xMLGregorianCalendarDTO.fractionalSecond = BigDecimal.valueOf(offsetDateTime.getNano()) + .divide(BigDecimal.valueOf(1_000_000_000), 9, RoundingMode.DOWN);📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.xMLGregorianCalendarDTO.fractionalSecond = BigDecimal.valueOf(offsetDateTime.getNano()) .divide(BigDecimal.valueOf(1_000_000_000), 9, RoundingMode.DOWN);
99-100: 🧹 Nitpick (assertive)
Adjust logging level for exception handling in
deserialize
methodThe catch block logs exceptions at
Level.FINEST
, which may not be visible in production logs. To ensure that deserialization issues are noticed, consider logging atLevel.WARNING
or higher.Apply this diff to adjust the logging level:
- log.log(Level.FINEST, "Cannot parse JSON into OffsetDateTime: " + e.getMessage(), e); + log.log(Level.WARNING, "Failed to deserialize JSON into OffsetDateTime: " + e.getMessage(), e);Alternatively, you might consider throwing a custom exception to allow the caller to handle deserialization errors appropriately.
📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.log.log(Level.WARNING, "Failed to deserialize JSON into OffsetDateTime: " + e.getMessage(), e); return null;
88-96:
⚠️ Potential issueEnsure non-null values in
XMLGregorianCalendarDTO
fields during deserializationWhile default values are assigned in
XMLGregorianCalendarDTO
, if any fields arenull
due to malformed JSON input, theOffsetDateTime.of
method could throw aNullPointerException
. Consider adding null checks or default assignments before constructingOffsetDateTime
.Apply this diff to handle potential
null
values:+ int year = xMLGregorianCalendarDTO.year != null ? xMLGregorianCalendarDTO.year : 0; + int month = xMLGregorianCalendarDTO.month != null ? xMLGregorianCalendarDTO.month : 1; + int day = xMLGregorianCalendarDTO.day != null ? xMLGregorianCalendarDTO.day : 1; + int hour = xMLGregorianCalendarDTO.hour != null ? xMLGregorianCalendarDTO.hour : 0; + int minute = xMLGregorianCalendarDTO.minute != null ? xMLGregorianCalendarDTO.minute : 0; + int second = xMLGregorianCalendarDTO.second != null ? xMLGregorianCalendarDTO.second : 0; + BigDecimal fractionalSecond = xMLGregorianCalendarDTO.fractionalSecond != null ? xMLGregorianCalendarDTO.fractionalSecond : BigDecimal.ZERO; + int timezoneMinutes = xMLGregorianCalendarDTO.timezone != null ? xMLGregorianCalendarDTO.timezone : 0; + ZoneOffset zoneoffset = ZoneOffset.ofTotalSeconds(timezoneMinutes * 60); - ZoneOffset zoneoffset = ZoneOffset.ofTotalSeconds(xMLGregorianCalendarDTO.timezone * 60); - return OffsetDateTime.of( - xMLGregorianCalendarDTO.year, - xMLGregorianCalendarDTO.month, - xMLGregorianCalendarDTO.day, - xMLGregorianCalendarDTO.hour, - xMLGregorianCalendarDTO.minute, - xMLGregorianCalendarDTO.second, - (int) (xMLGregorianCalendarDTO.fractionalSecond.doubleValue() * 1000_000_000), - zoneoffset); + return OffsetDateTime.of( + year, + month, + day, + hour, + minute, + second, + fractionalSecond.multiply(BigDecimal.valueOf(1_000_000_000)).intValue(), + zoneoffset);📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.int year = xMLGregorianCalendarDTO.year != null ? xMLGregorianCalendarDTO.year : 0; int month = xMLGregorianCalendarDTO.month != null ? xMLGregorianCalendarDTO.month : 1; int day = xMLGregorianCalendarDTO.day != null ? xMLGregorianCalendarDTO.day : 1; int hour = xMLGregorianCalendarDTO.hour != null ? xMLGregorianCalendarDTO.hour : 0; int minute = xMLGregorianCalendarDTO.minute != null ? xMLGregorianCalendarDTO.minute : 0; int second = xMLGregorianCalendarDTO.second != null ? xMLGregorianCalendarDTO.second : 0; BigDecimal fractionalSecond = xMLGregorianCalendarDTO.fractionalSecond != null ? xMLGregorianCalendarDTO.fractionalSecond : BigDecimal.ZERO; int timezoneMinutes = xMLGregorianCalendarDTO.timezone != null ? xMLGregorianCalendarDTO.timezone : 0; ZoneOffset zoneoffset = ZoneOffset.ofTotalSeconds(timezoneMinutes * 60); return OffsetDateTime.of( year, month, day, hour, minute, second, fractionalSecond.multiply(BigDecimal.valueOf(1_000_000_000)).intValue(), zoneoffset);
96-96: 🧹 Nitpick (assertive)
Avoid precision loss when converting
fractionalSecond
back to nanosecondsIn the
deserialize
method, usingdoubleValue()
to convertfractionalSecond
may cause precision loss due to floating-point representation. Consider usingBigDecimal
methods to maintain exact precision.Apply this diff to improve precision:
- (int) (xMLGregorianCalendarDTO.fractionalSecond.doubleValue() * 1000_000_000), + xMLGregorianCalendarDTO.fractionalSecond + .multiply(BigDecimal.valueOf(1_000_000_000)) + .intValue(),📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.xMLGregorianCalendarDTO.fractionalSecond .multiply(BigDecimal.valueOf(1_000_000_000)) .intValue(),
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/adapters/OffsetDateTimeJsonAdapter.java (8)
39-39: 🧹 Nitpick (assertive)
Consider reusing a configured
Gson
instanceThe adapter creates a new
Gson
instance with default settings. For consistency and to apply any custom configurations (e.g., date formats, null handling), consider injecting a pre-configuredGson
instance or using a sharedGsonBuilder
.
34-34: 🧹 Nitpick (assertive)
Update the
@since
tag to reflect the correct versionThe Javadoc
@since
tag currently states10.0.1
. Given that this is for version10.2
, update the tag to accurately reflect the introduction version.Apply this diff to update the version:
* @since 10.0.1 + * @since 10.2
Committable suggestion skipped: line range outside the PR's diff.
64-81: 🛠️ Refactor suggestion
Catch specific exceptions during deserialization
Catching the generic
Exception
can obscure the root cause of failures and make debugging harder. It's better to catch more specific exceptions likeJsonSyntaxException
orJsonParseException
.Apply this diff to catch specific exceptions:
try { // Parse DTO from current OffsetDateTime model return deserializeFromOffsetDateTimeDTO(gson.fromJson(jsonElement, DateTimeOffsetDTO.class)); - } catch (final Exception e) { + } catch (final JsonSyntaxException | JsonParseException e) { log.log( Level.FINEST, "Cannot parse JSON into OffsetDateTime from current DTO format: " + e.getMessage(), e); } log.log(Level.FINEST, "Attempting parsing from legacy DTO format"); try { // Parse DTO from legacy XMLGregorianCalendar model return deserializeFromXMLGregorianCalendarDTO(gson.fromJson(jsonElement, XMLGregorianCalendarDTO.class)); - } catch (final Exception e) { + } catch (final JsonSyntaxException | JsonParseException e) { log.log(Level.FINEST, "Cannot parse JSON into OffsetDateTime from legacy DTO format: " + e.getMessage(), e); }Committable suggestion skipped: line range outside the PR's diff.
80-81:
⚠️ Potential issueAvoid returning
null
on deserialization failureReturning
null
may lead toNullPointerException
in calling code. Consider throwing aJsonParseException
to explicitly indicate deserialization failure.Apply this diff to throw an exception:
} - return null; + throw new JsonParseException("Unable to parse JSON into OffsetDateTime"); }Committable suggestion skipped: line range outside the PR's diff.
150-154:
⚠️ Potential issueHandle potential
ArithmeticException
during nanosecond conversionConverting the scaled
fractionalSecond
to anint
may throw anArithmeticException
if the value exceeds the integer range. Ensure to handle this exception or validate the value before conversion.Apply this diff to safely convert nanoseconds:
nanoseconds = xmlGregorianCalendarDTO .fractionalSecond .scaleByPowerOfTen(9) - .toBigInteger() - .intValueExact(); + .intValueExact();Or, handle the exception:
+ try { nanoseconds = xmlGregorianCalendarDTO .fractionalSecond .scaleByPowerOfTen(9) .toBigInteger() .intValueExact(); + } catch (ArithmeticException e) { + throw new JsonParseException("Fractional second value out of int range", e); + }Committable suggestion skipped: line range outside the PR's diff.
68-72: 🧹 Nitpick (assertive)
Adjust logging level for deserialization failures
Currently, deserialization exceptions are logged at
Level.FINEST
, which may not be visible in production logs. Consider increasing the logging level toLevel.WARNING
orLevel.INFO
to make these issues more apparent during runtime.Apply this diff to adjust the logging level:
- log.log( - Level.FINEST, + log.log( + Level.WARNING, "Cannot parse JSON into OffsetDateTime from current DTO format: " + e.getMessage(), e);📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.log.log( Level.WARNING, "Cannot parse JSON into OffsetDateTime from current DTO format: " + e.getMessage(), e); }
162-198: 🧹 Nitpick (assertive)
Restrict access to inner DTO classes by making them
private
The DTO classes are only used within
OffsetDateTimeJsonAdapter
. Consider declaring them asprivate static
to encapsulate implementation details and prevent unintended usage outside the adapter.Apply this diff to adjust the access modifiers:
-static class DateTimeOffsetDTO { +private static class DateTimeOffsetDTO { -static class DateTimeDTO { +private static class DateTimeDTO { -static class DateDTO { +private static class DateDTO { -static class TimeDTO { +private static class TimeDTO { -static class OffsetDTO { +private static class OffsetDTO { -static class XMLGregorianCalendarDTO { +private static class XMLGregorianCalendarDTO {📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private static class DateTimeOffsetDTO { DateTimeDTO dateTime = new DateTimeDTO(); OffsetDTO offset; } private static class DateTimeDTO { DateDTO date = new DateDTO(); TimeDTO time = new TimeDTO(); } private static class DateDTO { Integer year; Integer month; Integer day; } private static class TimeDTO { Integer hour = 0; Integer minute = 0; Integer second = 0; Integer nano = 0; } private static class OffsetDTO { Integer totalSeconds = 0; } private static class XMLGregorianCalendarDTO { Integer year; Integer month; Integer day; Integer hour = 0; Integer minute = 0; Integer second = 0; BigDecimal fractionalSecond = BigDecimal.ZERO; Integer timezone; }
53-56: 🛠️ Refactor suggestion
Remove unnecessary null check for
getOffset()
The
OffsetDateTime#getOffset()
method always returns a non-nullZoneOffset
. The null check is unnecessary and can be safely removed to simplify the code.Apply this diff to remove the unnecessary null check:
- if (offsetDateTime.getOffset() != null) { dateTimeOffsetDTO.offset = new OffsetDTO(); dateTimeOffsetDTO.offset.totalSeconds = offsetDateTime.getOffset().getTotalSeconds(); - }Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/BusinessAppHdrV02.java (1)
24-31: 💡 Codebase verification
Migration from
javax.xml.bind
tojakarta.xml.bind
Requires Attention in Generated CodeThe verification shows that
javax.xml.bind
imports are only present in generated Java files under themodel-seev-types
andmodel-auth-types
directories. While the manual changes inBusinessAppHdrV02.java
correctly usejakarta.xml.bind
, the generated code still uses the oldjavax.xml.bind
package.
model-seev-types/src/generated/java/com/prowidesoftware/swift/model/mx/dic/RejectedStatusReason56.java
model-auth-types/src/generated/java/com/prowidesoftware/swift/model/mx/dic/FloatingRateIdentification8Choice.java
You'll need to update the code generation templates or configuration to use
jakarta.xml.bind
instead ofjavax.xml.bind
to ensure consistency across the codebase.🔗 Analysis chain
Ensure Complete Migration from
javax.xml.bind
tojakarta.xml.bind
The import statements have been updated from
javax.xml.bind
tojakarta.xml.bind
. Please verify that all references tojavax.xml.bind
have been updated throughout the codebase to prevent compatibility issues.Run the following script to verify that no references to
javax.xml.bind
remain in the codebase:
Let me gather more information to understand if these are generated files and if there are any non-generated files still using javax.xml.bind.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Search for any occurrences of 'javax.xml.bind' in Java files. # Expected result: No occurrences of 'javax.xml.bind' should be found. rg 'javax\.xml\.bind' -t javaLength of output: 1269
Script:
#!/bin/bash # Search for javax.xml.bind in non-generated Java files rg 'javax\.xml\.bind' -t java --glob '!**/generated/**' # Also check if there are any pom.xml files that might have javax.xml.bind dependencies rg 'javax\.xml\.bind' -t xmlLength of output: 88
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/AbstractMX.java (1)
621-634: 🧹 Nitpick (assertive)
Refactor
getGsonBuilder
methods to reduce code duplicationThe methods
getGsonBuilderWithV9Adapters()
andgetGsonBuilderWithV10Adapters()
are nearly identical, differing only in the adapter registered forOffsetDateTime
. Consider refactoring to a single method that accepts the adapter as a parameter to reduce code duplication and improve maintainability.Apply this diff to implement the suggested refactor:
@@ -611,29 +611,25 @@ } - private static Gson getGsonBuilderWithV9Adapters() { - final Gson gson = getGsonBuilderWithCustomAdapters() - .registerTypeAdapter(OffsetDateTime.class, new V9DateTimeJsonAdapter()) - .setPrettyPrinting() - .create(); - return gson; - } - - private static Gson getGsonBuilderWithV10Adapters() { - final Gson gson = getGsonBuilderWithCustomAdapters() - .registerTypeAdapter(OffsetDateTime.class, new OffsetDateTimeJsonAdapter()) - .setPrettyPrinting() - .create(); - return gson; + private static Gson getGsonBuilderWithAdapters(TypeAdapter<OffsetDateTime> adapter) { + return getGsonBuilderWithCustomAdapters() + .registerTypeAdapter(OffsetDateTime.class, adapter) + .setPrettyPrinting() + .create(); } + private static Gson getGsonBuilderWithV9Adapters() { + return getGsonBuilderWithAdapters(new V9DateTimeJsonAdapter()); + } + + private static Gson getGsonBuilderWithV10Adapters() { + return getGsonBuilderWithAdapters(new OffsetDateTimeJsonAdapter()); + } + /** * @return same as {@link #getNamespace()} * @since 9.1.2 */ public String targetNamespace() { return getNamespace(); } }This refactoring centralizes the creation of
Gson
instances and makes it easier to manage adapter configurations.📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private static Gson getGsonBuilderWithAdapters(TypeAdapter<OffsetDateTime> adapter) { return getGsonBuilderWithCustomAdapters() .registerTypeAdapter(OffsetDateTime.class, adapter) .setPrettyPrinting() .create(); } private static Gson getGsonBuilderWithV9Adapters() { return getGsonBuilderWithAdapters(new V9DateTimeJsonAdapter()); } private static Gson getGsonBuilderWithV10Adapters() { return getGsonBuilderWithAdapters(new OffsetDateTimeJsonAdapter()); }
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/AbstractMxJsonTestAdapters.java (9)
60-60: 🧹 Nitpick (assertive)
Remove unnecessary commented-out code
There are commented-out
System.out.println
statements on lines 60 and 74. If these are no longer needed, consider removing them to clean up the code.Apply this diff to remove the commented lines:
- // System.out.println(mx.message()); - // System.out.println(mx2.message());Also applies to: 74-74
30-30:
⚠️ Potential issueTypographical error in Javadoc
@since
tagThe Javadoc comment on line 30 has an extra
@
symbol at the end. It should be corrected to maintain proper documentation standards.Apply this diff to fix the typo:
- * @since 7.10.2@ + * @since 7.10.2📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.* @since 7.10.2
343-343: 🧹 Nitpick (assertive)
Add assertion to verify parsed MX message
In
parseMxWithAppHdr_LocalDate()
, consider asserting that the parsedmx
object is not null to strengthen the test.Apply this change:
assertDoesNotThrow(() -> AbstractMX.fromJson(json)); + AbstractMX mx = AbstractMX.fromJson(json); + assertNotNull(mx);Committable suggestion skipped: line range outside the PR's diff.
547-549:
⚠️ Potential issueEnsure safe casting of
mx.getAppHdr()
to preventClassCastException
In the method
parseSerializedMxWithAppHdrBAH_V2_OffsetDateTime()
, castingmx.getAppHdr()
toBusinessAppHdrV02
without checking its type risks aClassCastException
.Apply this diff to verify the instance before casting:
- BusinessAppHdrV02 BAH_V2 = (BusinessAppHdrV02) mx.getAppHdr(); + assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV02); + BusinessAppHdrV02 BAH_V2 = (BusinessAppHdrV02) mx.getAppHdr();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV02); BusinessAppHdrV02 BAH_V2 = (BusinessAppHdrV02) mx.getAppHdr(); assertNotNull(BAH_V2); }
1156-1158:
⚠️ Potential issueEnsure safe casting of
mx.getAppHdr()
to preventClassCastException
In
testJSONOffsetDateTimeWithNoNano_TimeWithNoNanoNoOffset()
, ensure that the app header is of the expected type before casting.Apply this diff to verify the instance before casting:
- BusinessAppHdrV02 businessAppHdrV02 = (BusinessAppHdrV02) mx.getAppHdr(); + assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV02); + BusinessAppHdrV02 businessAppHdrV02 = (BusinessAppHdrV02) mx.getAppHdr();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV02); BusinessAppHdrV02 businessAppHdrV02 = (BusinessAppHdrV02) mx.getAppHdr(); assertNotNull(businessAppHdrV02); }
617-619:
⚠️ Potential issueEnsure safe casting of
mx.getAppHdr()
to preventClassCastException
In
parseSerializedMxWithAppHdrNoNameSpace_OffsetDateTime()
, castingmx.getAppHdr()
toLegacyAppHdr
without type checking may cause aClassCastException
.Apply this diff to verify the instance before casting:
- LegacyAppHdr legacyAppHdr = (LegacyAppHdr) mx.getAppHdr(); + assertTrue(mx.getAppHdr() instanceof LegacyAppHdr); + LegacyAppHdr legacyAppHdr = (LegacyAppHdr) mx.getAppHdr();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.assertTrue(mx.getAppHdr() instanceof LegacyAppHdr); LegacyAppHdr legacyAppHdr = (LegacyAppHdr) mx.getAppHdr(); assertNotNull(legacyAppHdr); }
477-479:
⚠️ Potential issueEnsure safe casting of
mx.getAppHdr()
to preventClassCastException
In the method
parseSerializedMxWithAppHdrBAH_V1_OffsetDateTime()
, you castmx.getAppHdr()
toBusinessAppHdrV01
without checking its actual type. This may lead to aClassCastException
if the header is not of the expected type.Apply this diff to verify the instance before casting:
- BusinessAppHdrV01 BAH_V1 = (BusinessAppHdrV01) mx.getAppHdr(); + assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV01); + BusinessAppHdrV01 BAH_V1 = (BusinessAppHdrV01) mx.getAppHdr();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV01); BusinessAppHdrV01 BAH_V1 = (BusinessAppHdrV01) mx.getAppHdr(); assertNotNull(BAH_V1); }
771-772:
⚠️ Potential issueEnsure safe casting of
mx.getAppHdr()
and handle invalid namespacesIn the method
parseSerializedMxWithAppHdrInvalidNamespace_OffsetDateTime()
, casting without checking the header type may lead to runtime exceptions, especially given the invalid namespace in the test data.Consider adding instance checks and handling for unexpected namespaces:
+ assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV02); BusinessAppHdrV02 businessAppHdrV02 = (BusinessAppHdrV02) mx.getAppHdr(); assertNotNull(businessAppHdrV02);
Also, handle the case where the namespace does not match any known headers.
Committable suggestion skipped: line range outside the PR's diff.
1272-1274:
⚠️ Potential issueEnsure safe casting of
mx.getAppHdr()
to preventClassCastException
In
testJSONYearMonth()
, verify the type ofmx.getAppHdr()
before casting to prevent potential runtime exceptions.Apply this diff:
- BusinessAppHdrV02 businessAppHdrV02 = (BusinessAppHdrV02) mx.getAppHdr(); + assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV02); + BusinessAppHdrV02 businessAppHdrV02 = (BusinessAppHdrV02) mx.getAppHdr();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.assertTrue(mx.getAppHdr() instanceof BusinessAppHdrV02); BusinessAppHdrV02 businessAppHdrV02 = (BusinessAppHdrV02) mx.getAppHdr(); assertNotNull(businessAppHdrV02); }
.idea/compiler.xml (1)
4-7: 🧹 Nitpick (assertive)
Document the version compatibility strategy.
Consider adding a comment in the project's README or developer documentation explaining:
- Why core modules maintain Java 8 compatibility
- Which modules can use Java 11 features
- Guidelines for developers to maintain version compatibility
Would you like me to help draft this documentation section?
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/sys/dic/package-info.java (1)
27-28: 🧹 Nitpick (assertive)
Consider documenting Jakarta EE version requirements
Since this change is part of the migration from Java 8 to 11 and javax to Jakarta namespaces, it would be helpful to:
- Document the minimum required Jakarta EE version in the project's README or documentation
- Consider adding version constraints in the build configuration to prevent accidental use of incompatible dependencies
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/adapters/YearJsonAdapter.java (3)
43-45: 🧹 Nitpick (assertive)
Improve DTO encapsulation and Gson integration
The DTO class could be improved with better encapsulation and explicit Gson annotations.
Suggested improvements:
- static class YearJsonDTO { - int year; + private static class YearJsonDTO { + @Expose + private int year; + + public int getYear() { + return year; + } + + public void setYear(int year) { + this.year = year; + } }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private static class YearJsonDTO { @Expose private int year; public int getYear() { return year; } public void setYear(int year) { this.year = year; } }
22-26: 🧹 Nitpick (assertive)
Enhance class documentation with more details
The current documentation could be more comprehensive. Consider adding:
- The expected JSON format (structure and example)
- Usage examples
- Relationship with other date/time adapters in the system
Example improvement:
/** * This adapter enables accepting Year time Json format. + * + * Example JSON format: + * { + * "year": 2024 + * } + * + * Usage: + * Gson gson = new GsonBuilder() + * .registerTypeAdapter(Year.class, new YearJsonAdapter()) + * .create(); + * + * @see com.prowidesoftware.swift.model.mx.adapters.LocalDateJsonAdapter + * @see com.prowidesoftware.swift.model.mx.adapters.OffsetDateTimeJsonAdapter * * @since 10.1.0 */Committable suggestion skipped: line range outside the PR's diff.
27-41:
⚠️ Potential issueFix potential issues in JSON serialization/deserialization
Several critical issues need to be addressed:
- The
gson
field creates a new Gson instance which could lead to infinite recursion when serializing/deserializing nested objects.- Missing null checks in both methods.
- No validation for year value range in deserialization.
Apply these improvements:
public class YearJsonAdapter implements JsonSerializer<Year>, JsonDeserializer<Year> { - private final Gson gson = new Gson(); + private static final Gson gson = new GsonBuilder() + .excludeFieldsWithoutExposeAnnotation() + .create(); @Override public JsonElement serialize(Year year, Type type, JsonSerializationContext jsonSerializationContext) { + if (year == null) { + return JsonNull.INSTANCE; + } YearJsonDTO yearJsonDTO = new YearJsonDTO(); yearJsonDTO.year = year.getValue(); - return gson.toJsonTree(yearJsonDTO, YearJsonDTO.class); + return jsonSerializationContext.serialize(yearJsonDTO); } @Override public Year deserialize(JsonElement jsonElement, Type type, JsonDeserializationContext jsonDeserializationContext) { - YearJsonDTO yearJsonDTO = gson.fromJson(jsonElement, YearJsonDTO.class); + if (jsonElement.isJsonNull()) { + return null; + } + YearJsonDTO yearJsonDTO = jsonDeserializationContext.deserialize(jsonElement, YearJsonDTO.class); + if (yearJsonDTO.year < Year.MIN_VALUE || yearJsonDTO.year > Year.MAX_VALUE) { + throw new JsonParseException("Year value " + yearJsonDTO.year + " is outside valid range"); + } return Year.of(yearJsonDTO.year); }Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/YearJsonAdapterTest.java (4)
37-42:
⚠️ Potential issueUpdate implementation to use non-deprecated API and add validation
The current implementation uses a deprecated JsonParser constructor and lacks proper validation.
private void testSerializationAndDeserializationImpl(String json) { - JsonElement jsonElement = new JsonParser().parse(json); + JsonElement jsonElement = JsonParser.parseString(json); + + // Validate JSON structure + if (!jsonElement.isJsonObject() || !jsonElement.getAsJsonObject().has("year")) { + throw new IllegalArgumentException("Invalid JSON structure. Expected {\"year\": <value>}"); + } + Year year = adapter.deserialize(jsonElement, null, null); + if (year == null) { + throw new AssertionError("Deserialized Year should not be null"); + } + JsonElement valueDateResult = adapter.serialize(year, null, null); + if (valueDateResult == null) { + throw new AssertionError("Serialized JsonElement should not be null"); + } + assertEquals(json, valueDateResult.toString()); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private void testSerializationAndDeserializationImpl(String json) { JsonElement jsonElement = JsonParser.parseString(json); // Validate JSON structure if (!jsonElement.isJsonObject() || !jsonElement.getAsJsonObject().has("year")) { throw new IllegalArgumentException("Invalid JSON structure. Expected {\"year\": <value>}"); } Year year = adapter.deserialize(jsonElement, null, null); if (year == null) { throw new AssertionError("Deserialized Year should not be null"); } JsonElement valueDateResult = adapter.serialize(year, null, null); if (valueDateResult == null) { throw new AssertionError("Serialized JsonElement should not be null"); } assertEquals(json, valueDateResult.toString()); }
29-35: 🛠️ Refactor suggestion
Enhance test coverage with more test cases
The current test implementation has duplicate test cases and limited coverage. Consider:
- Removing the duplicate cases for year 2019
- Adding edge cases (e.g., Year.MIN_VALUE, Year.MAX_VALUE)
- Adding invalid year format tests
- Testing different year representations
@Test void testSerializationAndDeserialization() { - testSerializationAndDeserializationImpl("{\"year\":2019}"); - testSerializationAndDeserializationImpl("{\"year\":2019}"); - testSerializationAndDeserializationImpl("{\"year\":2019}"); - testSerializationAndDeserializationImpl("{\"year\":1983}"); + // Valid cases + testSerializationAndDeserializationImpl("{\"year\":2019}"); + testSerializationAndDeserializationImpl("{\"year\":1983}"); + + // Edge cases + testSerializationAndDeserializationImpl("{\"year\":-999999999}"); // Year.MIN_VALUE + testSerializationAndDeserializationImpl("{\"year\":999999999}"); // Year.MAX_VALUE + testSerializationAndDeserializationImpl("{\"year\":0}"); // Year zero } + +@Test +void testInvalidYearDeserialization() { + assertThrows(JsonParseException.class, () -> + testSerializationAndDeserializationImpl("{\"year\":\"invalid\"}")); + assertThrows(DateTimeException.class, () -> + testSerializationAndDeserializationImpl("{\"year\":1000000000}")); +}Committable suggestion skipped: line range outside the PR's diff.
25-28: 🧹 Nitpick (assertive)
Improve test class organization and documentation
Consider enhancing the test class with better organization and documentation:
- Add class-level documentation explaining the test coverage
- Group tests by categories (valid cases, edge cases, error cases)
- Add tests for null handling
+/** + * Test class for {@link YearJsonAdapter} that validates JSON serialization and deserialization + * of {@link Year} objects. + * + * Test categories: + * - Valid year serialization/deserialization + * - Edge cases (MIN/MAX values) + * - Error cases (invalid format, out of range) + * - Null handling + */ public class YearJsonAdapterTest { private final YearJsonAdapter adapter = new YearJsonAdapter(); + + @Test + void testNullHandling() { + assertNull(adapter.deserialize(null, null, null)); + assertNull(adapter.serialize(null, null, null)); + }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements./** * Test class for {@link YearJsonAdapter} that validates JSON serialization and deserialization * of {@link Year} objects. * * Test categories: * - Valid year serialization/deserialization * - Edge cases (MIN/MAX values) * - Error cases (invalid format, out of range) * - Null handling */ public class YearJsonAdapterTest { private final YearJsonAdapter adapter = new YearJsonAdapter(); @Test void testNullHandling() { assertNull(adapter.deserialize(null, null, null)); assertNull(adapter.serialize(null, null, null)); }
18-23: 🛠️ Refactor suggestion
Add missing imports for enhanced test coverage
Add the following imports to support the suggested improvements:
import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNull; +import static org.junit.jupiter.api.Assertions.assertThrows; import com.google.gson.JsonElement; import com.google.gson.JsonParser; +import com.google.gson.JsonParseException; import java.time.Year; +import java.time.DateTimeException; import org.junit.jupiter.api.Test;Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/test/java/com/prowidesoftware/issues/Jira1567.java (2)
41-46:
⚠️ Potential issueImprove CustomDateTimeAdapter robustness
The current implementation has several potential issues:
- It assumes the input is always in UTC ("+00:00")
- Direct string replacement is fragile
- Missing unmarshal implementation for symmetry
Consider this more robust implementation:
public class CustomDateTimeAdapter extends OffsetDateTimeAdapter { @Override public String marshal(OffsetDateTime offsetDateTime) throws Exception { - return StringUtils.replace(super.marshal(offsetDateTime), "+00:00", "Z"); + if (offsetDateTime == null) { + return null; + } + // First normalize to UTC + OffsetDateTime utc = offsetDateTime.withOffsetSameInstant(ZoneOffset.UTC); + // Then format with 'Z' + return utc.format(DateTimeFormatter.ISO_INSTANT); + } + + @Override + public OffsetDateTime unmarshal(String value) throws Exception { + if (value == null) { + return null; + } + return OffsetDateTime.parse(value, DateTimeFormatter.ISO_INSTANT); + } }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.public class CustomDateTimeAdapter extends OffsetDateTimeAdapter { @Override public String marshal(OffsetDateTime offsetDateTime) throws Exception { if (offsetDateTime == null) { return null; } // First normalize to UTC OffsetDateTime utc = offsetDateTime.withOffsetSameInstant(ZoneOffset.UTC); // Then format with 'Z' return utc.format(DateTimeFormatter.ISO_INSTANT); } @Override public OffsetDateTime unmarshal(String value) throws Exception { if (value == null) { return null; } return OffsetDateTime.parse(value, DateTimeFormatter.ISO_INSTANT); } }
17-39: 🧹 Nitpick (assertive)
Enhance test coverage and naming
While the test logic is correct, consider these improvements for better maintainability and reliability:
- Rename the test method to describe its purpose (e.g.,
testDateTimeSerializationFormats
)- Add more specific assertions beyond simple string contains
- Validate the complete XML structure
- Include negative test cases (e.g., non-UTC timezones)
Here's a suggested improvement:
- @Test - public void test() { + @Test + public void testDateTimeSerializationFormats() { MxXsys01500101 mx = new MxXsys01500101(); // create an OffsetDateTime with the current time in UTC OffsetDateTime now = OffsetDateTime.now(ZoneOffset.UTC); LegacyAppHdr legacy = new LegacyAppHdr(); legacy.setCrDate(now); mx.setAppHdr(legacy); // default date time adapter contains the offset String xml = mx.message(); - assertTrue(xml.contains("+00:00</head:CrDate>")); + assertTrue(xml.matches(".*<head:CrDate>\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\+00:00</head:CrDate>.*")); // custom serialization using a custom date time adapter MxWriteConfiguration config = new MxWriteConfiguration(); config.adapters.dateTimeAdapter = new IsoDateTimeAdapter(new CustomDateTimeAdapter()); xml = mx.message(config); - assertTrue(xml.contains("Z</head:CrDate>")); + assertTrue(xml.matches(".*<head:CrDate>\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}Z</head:CrDate>.*")); + + // Add negative test case + OffsetDateTime nonUtc = OffsetDateTime.now(ZoneOffset.ofHours(2)); + legacy.setCrDate(nonUtc); + xml = mx.message(config); + assertTrue(xml.matches(".*<head:CrDate>\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}Z</head:CrDate>.*"), + "Non-UTC time should be converted to UTC with Z suffix"); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.@Test public void testDateTimeSerializationFormats() { MxXsys01500101 mx = new MxXsys01500101(); // create an OffsetDateTime with the current time in UTC OffsetDateTime now = OffsetDateTime.now(ZoneOffset.UTC); LegacyAppHdr legacy = new LegacyAppHdr(); legacy.setCrDate(now); mx.setAppHdr(legacy); // default date time adapter contains the offset String xml = mx.message(); assertTrue(xml.matches(".*<head:CrDate>\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\+00:00</head:CrDate>.*")); // custom serialization using a custom date time adapter MxWriteConfiguration config = new MxWriteConfiguration(); config.adapters.dateTimeAdapter = new IsoDateTimeAdapter(new CustomDateTimeAdapter()); xml = mx.message(config); assertTrue(xml.matches(".*<head:CrDate>\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}Z</head:CrDate>.*")); // Add negative test case OffsetDateTime nonUtc = OffsetDateTime.now(ZoneOffset.ofHours(2)); legacy.setCrDate(nonUtc); xml = mx.message(config); assertTrue(xml.matches(".*<head:CrDate>\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}Z</head:CrDate>.*"), "Non-UTC time should be converted to UTC with Z suffix"); }
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/LocalDateJsonAdapterTest.java (3)
29-35: 🧹 Nitpick (assertive)
Enhance test coverage with additional test cases
While the current test cases cover valid scenarios, consider adding:
- Edge cases (min/max dates)
- Invalid JSON input (error handling)
- Null handling
- Malformed date components (e.g., invalid month/day values)
Example additional test cases:
// Edge cases testSerializationAndDeserializationImpl("{\"year\":1,\"month\":1,\"day\":1}"); testSerializationAndDeserializationImpl("{\"year\":9999,\"month\":12,\"day\":31}"); // Invalid cases (add appropriate exception handling) @Test void testInvalidInput() { assertThrows(JsonParseException.class, () -> testSerializationAndDeserializationImpl("{\"year\":2019,\"month\":13,\"day\":32}")); assertThrows(JsonParseException.class, () -> testSerializationAndDeserializationImpl("{\"year\":2019}")); assertThrows(JsonParseException.class, () -> testSerializationAndDeserializationImpl("invalid json")); }
21-21: 🛠️ Refactor suggestion
Replace deprecated JsonParser usage
The
JsonParser
class is deprecated since Gson 2.8.9. Consider usingJsonParser.parseString()
static method instead.-import com.google.gson.JsonParser; +import com.google.gson.JsonParser; // For static parseString methodAnd in the implementation:
- JsonElement jsonElement = new JsonParser().parse(json); + JsonElement jsonElement = JsonParser.parseString(json);Committable suggestion skipped: line range outside the PR's diff.
37-42: 🧹 Nitpick (assertive)
Improve implementation method robustness
The current implementation could be enhanced with:
- Input validation
- Explicit error handling
- More descriptive assertion message
- private void testSerializationAndDeserializationImpl(String json) { + private void testSerializationAndDeserializationImpl(String json) { + if (json == null || json.trim().isEmpty()) { + throw new IllegalArgumentException("JSON input cannot be null or empty"); + } + JsonElement jsonElement = new JsonParser().parse(json); LocalDate date = adapter.deserialize(jsonElement, null, null); JsonElement valueDateResult = adapter.serialize(date, null, null); - assertEquals(json, valueDateResult.toString()); + assertEquals(json, valueDateResult.toString(), + String.format("Serialization/deserialization failed for input: %s", json)); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private void testSerializationAndDeserializationImpl(String json) { if (json == null || json.trim().isEmpty()) { throw new IllegalArgumentException("JSON input cannot be null or empty"); } JsonElement jsonElement = new JsonParser().parse(json); LocalDate date = adapter.deserialize(jsonElement, null, null); JsonElement valueDateResult = adapter.serialize(date, null, null); assertEquals(json, valueDateResult.toString(), String.format("Serialization/deserialization failed for input: %s", json)); }
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/adapters/YearMonthJsonAdapter.java (4)
31-36: 🛠️ Refactor suggestion
Add null check in serialize method.
The method should handle null YearMonth objects gracefully.
@Override public JsonElement serialize(YearMonth yearMonth, Type type, JsonSerializationContext jsonSerializationContext) { + if (yearMonth == null) { + return JsonNull.INSTANCE; + } YearMonthJsonDTO yearJsonDTO = new YearMonthJsonDTO(); yearJsonDTO.year = yearMonth.getYear(); yearJsonDTO.month = yearMonth.getMonth().getValue(); return gson.toJsonTree(yearJsonDTO, YearMonthJsonDTO.class); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.public JsonElement serialize(YearMonth yearMonth, Type type, JsonSerializationContext jsonSerializationContext) { if (yearMonth == null) { return JsonNull.INSTANCE; } YearMonthJsonDTO yearJsonDTO = new YearMonthJsonDTO(); yearJsonDTO.year = yearMonth.getYear(); yearJsonDTO.month = yearMonth.getMonth().getValue(); return gson.toJsonTree(yearJsonDTO, YearMonthJsonDTO.class); }
27-29: 🧹 Nitpick (assertive)
Consider making Gson instance static and final.
The Gson instance could be shared across all instances of the adapter since it's stateless.
- private final Gson gson = new Gson(); + private static final Gson gson = new Gson();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.public class YearMonthJsonAdapter implements JsonSerializer<YearMonth>, JsonDeserializer<YearMonth> { private static final Gson gson = new Gson();
45-48: 🧹 Nitpick (assertive)
Consider improving DTO encapsulation.
The DTO could benefit from better encapsulation:
- Make the class private
- Make fields private with getters/setters
- static class YearMonthJsonDTO { - int year; - int month; + private static class YearMonthJsonDTO { + private int year; + private int month; + + public int getYear() { + return year; + } + + public void setYear(int year) { + this.year = year; + } + + public int getMonth() { + return month; + } + + public void setMonth(int month) { + this.month = month; + } }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private static class YearMonthJsonDTO { private int year; private int month; public int getYear() { return year; } public void setYear(int year) { this.year = year; } public int getMonth() { return month; } public void setMonth(int month) { this.month = month; } }
38-43: 🛠️ Refactor suggestion
Add validation in deserialize method.
The method should validate the input and handle edge cases:
- Null checks
- Month value range validation (1-12)
@Override public YearMonth deserialize( JsonElement jsonElement, Type type, JsonDeserializationContext jsonDeserializationContext) { + if (jsonElement == null || jsonElement.isJsonNull()) { + return null; + } YearMonthJsonDTO yearMonthJsonDTO = gson.fromJson(jsonElement, YearMonthJsonDTO.class); + if (yearMonthJsonDTO.month < 1 || yearMonthJsonDTO.month > 12) { + throw new JsonParseException("Month value must be between 1 and 12, got: " + yearMonthJsonDTO.month); + } return YearMonth.of(yearMonthJsonDTO.year, yearMonthJsonDTO.month); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.@Override public YearMonth deserialize( JsonElement jsonElement, Type type, JsonDeserializationContext jsonDeserializationContext) { if (jsonElement == null || jsonElement.isJsonNull()) { return null; } YearMonthJsonDTO yearMonthJsonDTO = gson.fromJson(jsonElement, YearMonthJsonDTO.class); if (yearMonthJsonDTO.month < 1 || yearMonthJsonDTO.month > 12) { throw new JsonParseException("Month value must be between 1 and 12, got: " + yearMonthJsonDTO.month); } return YearMonth.of(yearMonthJsonDTO.year, yearMonthJsonDTO.month); }
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/adapters/LocalDateJsonAdapter.java (5)
29-29: 🧹 Nitpick (assertive)
Consider making Gson instance static and thread-safe.
The current implementation creates a new Gson instance per adapter instance. Consider making it static and configuring it with appropriate settings.
- private final Gson gson = new Gson(); + private static final Gson gson = new GsonBuilder() + .serializeNulls() + .create();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private static final Gson gson = new GsonBuilder() .serializeNulls() .create();
31-38: 🛠️ Refactor suggestion
Add null check in serialize method.
The method should handle null input gracefully.
@Override public JsonElement serialize(LocalDate localDate, Type typeOfSrc, JsonSerializationContext context) { + if (localDate == null) { + return JsonNull.INSTANCE; + } DateDTO date = new DateDTO(); date.year = localDate.getYear(); date.month = localDate.getMonthValue(); date.day = localDate.getDayOfMonth(); return gson.toJsonTree(date, DateDTO.class); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.@Override public JsonElement serialize(LocalDate localDate, Type typeOfSrc, JsonSerializationContext context) { if (localDate == null) { return JsonNull.INSTANCE; } DateDTO date = new DateDTO(); date.year = localDate.getYear(); date.month = localDate.getMonthValue(); date.day = localDate.getDayOfMonth(); return gson.toJsonTree(date, DateDTO.class); }
22-26: 🧹 Nitpick (assertive)
Enhance class documentation.
The current documentation is minimal. Consider adding:
- Example JSON format
- Thread-safety considerations
- Null handling behavior
/** * This adapter enables accepting LocalDate time Json format. + * + * Example JSON format: + * { + * "year": 2024, + * "month": 11, + * "day": 15 + * } + * + * Thread-safety: This adapter maintains a shared Gson instance. + * Null handling: null input results in null output. * * @since 10.1.0 */📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements./** * This adapter enables accepting LocalDate time Json format. * * Example JSON format: * { * "year": 2024, * "month": 11, * "day": 15 * } * * Thread-safety: This adapter maintains a shared Gson instance. * Null handling: null input results in null output. * * @since 10.1.0 */
40-46: 🛠️ Refactor suggestion
Add validation in deserialize method.
The method should:
- Handle null/invalid input
- Validate date values before construction
@Override public LocalDate deserialize( final JsonElement jsonElement, final Type typeOfT, final JsonDeserializationContext context) throws JsonParseException { + if (jsonElement == null || jsonElement.isJsonNull()) { + return null; + } DateDTO date = gson.fromJson(jsonElement, DateDTO.class); + if (date == null || date.year == null || date.month == null || date.day == null) { + throw new JsonParseException("Invalid date format: missing required fields"); + } + try { return LocalDate.of(date.year, date.month, date.day); + } catch (DateTimeException e) { + throw new JsonParseException("Invalid date values", e); + } }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.@Override public LocalDate deserialize( final JsonElement jsonElement, final Type typeOfT, final JsonDeserializationContext context) throws JsonParseException { if (jsonElement == null || jsonElement.isJsonNull()) { return null; } DateDTO date = gson.fromJson(jsonElement, DateDTO.class); if (date == null || date.year == null || date.month == null || date.day == null) { throw new JsonParseException("Invalid date format: missing required fields"); } try { return LocalDate.of(date.year, date.month, date.day); } catch (DateTimeException e) { throw new JsonParseException("Invalid date values", e); } }
48-52: 🧹 Nitpick (assertive)
Consider enhancing the DTO implementation.
The current implementation could benefit from:
- Making fields final for immutability
- Adding a constructor with validation
- Adding toString() for debugging
static class DateDTO { - Integer year; - Integer month; - Integer day; + final Integer year; + final Integer month; + final Integer day; + + DateDTO() { + this.year = null; + this.month = null; + this.day = null; + } + + @Override + public String toString() { + return "DateDTO{" + + "year=" + year + + ", month=" + month + + ", day=" + day + + '}'; + } }Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/JaxbContextCache.java (1)
18-19: 💡 Codebase verification
Migration to jakarta.xml.bind requires attention in generated code
The verification reveals a mixed state of JAXB dependencies:
- All non-generated code correctly uses
jakarta.xml.bind
imports (as seen in core and common-types modules)- Generated code in
model-seev-types
andmodel-auth-types
still usesjavax.xml.bind
- Statistics show 2 files with
javax.xml.bind
vs 28,766 occurrences ofjakarta.xml.bind
This indicates that while the core codebase has been properly migrated, the code generation templates or tools need to be updated to use the Jakarta namespace instead of javax.
🔗 Analysis chain
Verify JAXB dependency consistency across the project
The migration from
javax.xml.bind
tojakarta.xml.bind
is correct for Jakarta EE 9+ compatibility. However, we should ensure all JAXB-related dependencies are consistently using the jakarta namespace to prevent runtime conflicts.
The initial scan shows that there are still some files using
javax.xml.bind
imports in the generated code directories. Let's verify if these are isolated cases or if there's a broader pattern, and also check the JAXB dependencies more thoroughly.🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Check for any remaining javax.xml.bind imports and dependencies # that might cause conflicts with jakarta.xml.bind echo "Checking for any remaining javax.xml.bind imports..." rg "javax\.xml\.bind" -t java echo "Checking pom.xml files for JAXB dependencies..." find . -name "pom.xml" -type f -exec grep -l "xml.bind" {} \; | xargs catLength of output: 1564
Script:
#!/bin/bash # Check for all JAXB-related imports (both javax and jakarta) in non-generated code echo "=== Checking JAXB imports in non-generated code ===" rg "import (javax|jakarta)\.xml\.bind" --type java | grep -v "/generated/" # Check specifically for JAXB dependencies in pom files echo -e "\n=== Checking JAXB dependencies in pom.xml files ===" find . -name "pom.xml" -type f -exec sh -c 'echo "=== $1 ==="; grep -A 2 -B 2 "xml.bind\|jaxb" "$1"' sh {} \; # Count occurrences of javax vs jakarta in all Java files echo -e "\n=== Statistics of JAXB usage ===" echo "javax.xml.bind occurrences:" rg "javax\.xml\.bind" -t java --count-matches | wc -l echo "jakarta.xml.bind occurrences:" rg "jakarta\.xml\.bind" -t java --count-matches | wc -lLength of output: 11557
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/v9/V9DateTimeJsonAdapterTest.java (5)
49-49:
⚠️ Potential issueReplace deprecated JsonParser.parse() with JsonParser.parseString()
The
JsonParser.parse()
method is deprecated. UseJsonParser.parseString()
instead, or better yet, useGson.fromJson()
directly.Apply this change:
- JsonElement sourceJsonElement = new JsonParser().parse(sourceJsonV9); + JsonElement sourceJsonElement = JsonParser.parseString(sourceJsonV9); - JsonElement expectedJsonElement = new JsonParser().parse(expectedJsonV9); + JsonElement expectedJsonElement = JsonParser.parseString(expectedJsonV9);Also applies to: 55-55
10-10: 🧹 Nitpick (assertive)
Consider adding tests for error cases and edge cases
The test coverage could be improved by adding tests for:
- Invalid JSON input
- Malformed datetime strings
- Edge cases like maximum/minimum dates
- Null input handling
- Different timezone offsets (positive, negative, large offsets)
Would you like me to provide example test cases for these scenarios?
14-29: 🧹 Nitpick (assertive)
Consider splitting test cases and adding @DisplayName
While the test cases are well structured, consider splitting them into separate methods for better clarity and test reporting.
Here's a suggested refactor:
- @Test - void testDeserialization() { - String sourceJsonV9; - OffsetDateTime expectedOffsetDateTime; - - sourceJsonV9 = - "{\"year\":2022,\"month\":11,\"day\":30,\"timezone\":-180,\"hour\":10,\"minute\":43,\"second\":57,\"fractionalSecond\":0.412}"; - expectedOffsetDateTime = OffsetDateTime.parse("2022-11-30T10:43:57.412-03:00"); - testDeserialization(sourceJsonV9, expectedOffsetDateTime); - - // without offset, nano 0 - sourceJsonV9 = - "{\"year\":2019,\"month\":4,\"day\":28,\"timezone\":0,\"hour\":12,\"minute\":13,\"second\":14,\"fractionalSecond\":0}"; - expectedOffsetDateTime = OffsetDateTime.parse("2019-04-28T12:13:14.0-00:00"); - testDeserialization(sourceJsonV9, expectedOffsetDateTime); - } + @Test + @DisplayName("Should deserialize datetime with timezone offset") + void testDeserializationWithOffset() { + String sourceJsonV9 = "{\"year\":2022,\"month\":11,\"day\":30,\"timezone\":-180,\"hour\":10,\"minute\":43,\"second\":57,\"fractionalSecond\":0.412}"; + OffsetDateTime expectedOffsetDateTime = OffsetDateTime.parse("2022-11-30T10:43:57.412-03:00"); + testDeserialization(sourceJsonV9, expectedOffsetDateTime); + } + + @Test + @DisplayName("Should deserialize datetime without timezone offset") + void testDeserializationWithoutOffset() { + String sourceJsonV9 = "{\"year\":2019,\"month\":4,\"day\":28,\"timezone\":0,\"hour\":12,\"minute\":13,\"second\":14,\"fractionalSecond\":0}"; + OffsetDateTime expectedOffsetDateTime = OffsetDateTime.parse("2019-04-28T12:13:14.0-00:00"); + testDeserialization(sourceJsonV9, expectedOffsetDateTime); + }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.@Test @DisplayName("Should deserialize datetime with timezone offset") void testDeserializationWithOffset() { String sourceJsonV9 = "{\"year\":2022,\"month\":11,\"day\":30,\"timezone\":-180,\"hour\":10,\"minute\":43,\"second\":57,\"fractionalSecond\":0.412}"; OffsetDateTime expectedOffsetDateTime = OffsetDateTime.parse("2022-11-30T10:43:57.412-03:00"); testDeserialization(sourceJsonV9, expectedOffsetDateTime); } @Test @DisplayName("Should deserialize datetime without timezone offset") void testDeserializationWithoutOffset() { String sourceJsonV9 = "{\"year\":2019,\"month\":4,\"day\":28,\"timezone\":0,\"hour\":12,\"minute\":13,\"second\":14,\"fractionalSecond\":0}"; OffsetDateTime expectedOffsetDateTime = OffsetDateTime.parse("2019-04-28T12:13:14.0-00:00"); testDeserialization(sourceJsonV9, expectedOffsetDateTime); }
31-46: 🧹 Nitpick (assertive)
Split test cases and fix inconsistent parameter naming
Similar to the deserialization tests, consider splitting these into separate methods. Also, there's an inconsistent parameter name.
Here's a suggested refactor:
- @Test - void testSerialization() { - OffsetDateTime sourceOffsetDateTime; - String expectedJsonV9; - - sourceOffsetDateTime = OffsetDateTime.parse("2022-11-30T10:43:57.412-03:00"); - expectedJsonV9 = - "{\"year\":2022,\"month\":11,\"day\":30,\"timezone\":-180,\"hour\":10,\"minute\":43,\"second\":57,\"fractionalSecond\":0.412}"; - testSerialization(sourceOffsetDateTime, expectedJsonV9); - - // without offset, nano 0 - sourceOffsetDateTime = OffsetDateTime.parse("2019-04-28T12:13:14.0-00:00"); - expectedJsonV9 = - "{\"year\":2019,\"month\":4,\"day\":28,\"timezone\":0,\"hour\":12,\"minute\":13,\"second\":14,\"fractionalSecond\":0}"; - testSerialization(sourceOffsetDateTime, expectedJsonV9); - } + @Test + @DisplayName("Should serialize datetime with timezone offset") + void testSerializationWithOffset() { + OffsetDateTime sourceDateTime = OffsetDateTime.parse("2022-11-30T10:43:57.412-03:00"); + String expectedJsonV9 = "{\"year\":2022,\"month\":11,\"day\":30,\"timezone\":-180,\"hour\":10,\"minute\":43,\"second\":57,\"fractionalSecond\":0.412}"; + testSerialization(sourceDateTime, expectedJsonV9); + } + + @Test + @DisplayName("Should serialize datetime without timezone offset") + void testSerializationWithoutOffset() { + OffsetDateTime sourceDateTime = OffsetDateTime.parse("2019-04-28T12:13:14.0-00:00"); + String expectedJsonV9 = "{\"year\":2019,\"month\":4,\"day\":28,\"timezone\":0,\"hour\":12,\"minute\":13,\"second\":14,\"fractionalSecond\":0}"; + testSerialization(sourceDateTime, expectedJsonV9); + }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.@Test @DisplayName("Should serialize datetime with timezone offset") void testSerializationWithOffset() { OffsetDateTime sourceDateTime = OffsetDateTime.parse("2022-11-30T10:43:57.412-03:00"); String expectedJsonV9 = "{\"year\":2022,\"month\":11,\"day\":30,\"timezone\":-180,\"hour\":10,\"minute\":43,\"second\":57,\"fractionalSecond\":0.412}"; testSerialization(sourceDateTime, expectedJsonV9); } @Test @DisplayName("Should serialize datetime without timezone offset") void testSerializationWithoutOffset() { OffsetDateTime sourceDateTime = OffsetDateTime.parse("2019-04-28T12:13:14.0-00:00"); String expectedJsonV9 = "{\"year\":2019,\"month\":4,\"day\":28,\"timezone\":0,\"hour\":12,\"minute\":13,\"second\":14,\"fractionalSecond\":0}"; testSerialization(sourceDateTime, expectedJsonV9); }
48-58: 🧹 Nitpick (assertive)
Add parameter validation and fix inconsistent naming in helper methods
The helper methods could be improved with null parameter validation and consistent naming.
Here's a suggested refactor:
private void testDeserialization(String sourceJsonV9, OffsetDateTime expected) { + if (sourceJsonV9 == null || expected == null) { + throw new IllegalArgumentException("Parameters cannot be null"); + } JsonElement sourceJsonElement = JsonParser.parseString(sourceJsonV9); OffsetDateTime deserializeResult = adapter.deserialize(sourceJsonElement, null, null); assertEquals(expected, deserializeResult); } - private void testSerialization(OffsetDateTime sourceOffsetDateTimeV10, String expectedJsonV9) { + private void testSerialization(OffsetDateTime sourceDateTime, String expectedJsonV9) { + if (sourceDateTime == null || expectedJsonV9 == null) { + throw new IllegalArgumentException("Parameters cannot be null"); + } JsonElement expectedJsonElement = JsonParser.parseString(expectedJsonV9); - JsonElement serializeResult = adapter.serialize(sourceOffsetDateTimeV10, null, null); + JsonElement serializeResult = adapter.serialize(sourceDateTime, null, null); assertEquals(expectedJsonElement, serializeResult); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private void testDeserialization(String sourceJsonV9, OffsetDateTime expected) { if (sourceJsonV9 == null || expected == null) { throw new IllegalArgumentException("Parameters cannot be null"); } JsonElement sourceJsonElement = JsonParser.parseString(sourceJsonV9); OffsetDateTime deserializeResult = adapter.deserialize(sourceJsonElement, null, null); assertEquals(expected, deserializeResult); } private void testSerialization(OffsetDateTime sourceDateTime, String expectedJsonV9) { if (sourceDateTime == null || expectedJsonV9 == null) { throw new IllegalArgumentException("Parameters cannot be null"); } JsonElement expectedJsonElement = JsonParser.parseString(expectedJsonV9); JsonElement serializeResult = adapter.serialize(sourceDateTime, null, null); assertEquals(expectedJsonElement, serializeResult); }
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/YearMonthJsonAdapterTest.java (4)
46-46: 🧹 Nitpick (assertive)
Improve test method name for clarity
The current name
testYear_CustomPattern
could be more descriptive. Consider renaming to better reflect what's being tested, e.g.,testYearSerializationInNestedStructure
.
47-65: 🧹 Nitpick (assertive)
Extract test JSON to a resource file
The large JSON string literal makes the test harder to read and maintain. Consider moving it to a separate test resource file.
Create a new file
src/test/resources/samples/acmt.034.001.04.json
and load it in the test:String jsonOrigin = Files.readString(Path.of("src/test/resources/samples/acmt.034.001.04.json"));
39-39:
⚠️ Potential issueUpdate deprecated JsonParser.parse() method
The
JsonParser.parse()
method is deprecated. UseJsonParser.parseString()
instead for better compatibility with newer versions.Apply this change:
- JsonElement jsonElement = new JsonParser().parse(json); + JsonElement jsonElement = JsonParser.parseString(json);📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.JsonElement jsonElement = JsonParser.parseString(json);
67-69: 🛠️ Refactor suggestion
Strengthen the assertion for better test reliability
The current assertion only checks for a substring, which could pass even if the JSON structure is incorrect. Consider:
- Parsing the output JSON and asserting on the specific field
- Using JSONAssert or similar library for structured JSON comparison
Example improvement:
- assertTrue(jsonParsed.contains("\"year\": 2022")); + JsonElement parsed = JsonParser.parseString(jsonParsed); + int year = parsed.getAsJsonObject() + .getAsJsonObject("acctSwtchReqPmt") + .getAsJsonObject("cdtInstr") + .getAsJsonObject("tax") + .getAsJsonArray("rcrd") + .get(0) + .getAsJsonObject() + .getAsJsonObject("prd") + .getAsJsonObject("yr") + .get("year") + .getAsInt(); + assertEquals(2022, year);Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/IssueJira1566Test.java (5)
46-51: 🧹 Nitpick (assertive)
Consider splitting into separate test methods
The Zulu time test case contains multiple assertions testing different aspects. Consider splitting into separate test methods for better clarity and failure isolation.
Example refactor:
@Test public void testZuluTimeConversionToOffset() { MxPacs00800108 mx = new MxPacs00800108(); OffsetDateTime offsetDateTime = OffsetDateTime.parse("2021-09-19T12:13:14.000Z"); mx.getFIToFICstmrCdtTrf().getGrpHdr().setCreDtTm(offsetDateTime); String message = mx.message(); assertTrue(message.contains("2021-09-19T12:13:14+00:00")); } @Test public void testZuluTimeFormatNotPresent() { MxPacs00800108 mx = new MxPacs00800108(); OffsetDateTime offsetDateTime = OffsetDateTime.parse("2021-09-19T12:13:14.000Z"); mx.getFIToFICstmrCdtTrf().getGrpHdr().setCreDtTm(offsetDateTime); String message = mx.message(); assertFalse(message.contains("2021-09-19T12:13:14.000Z")); assertFalse(message.contains("2021-09-19T12:13:14.Z")); }
53-68: 🧹 Nitpick (assertive)
Enhance test organization and documentation
The zero nanoseconds test cases could be improved in several ways:
- Use consistent date-time pattern across test cases
- Add more descriptive comments explaining the purpose of each test case
- Consider separating into individual test methods
Example refactor:
@Test public void testZeroNanosecondsWithPattern() { MxPacs00800108 mx = new MxPacs00800108(); // Test explicit zero nanoseconds using pattern OffsetDateTime offsetDateTime = OffsetDateTime.parse( "2021-09-19T12:13:14.00000+00:00", DateTimeFormatter.ISO_OFFSET_DATE_TIME); mx.getFIToFICstmrCdtTrf().getGrpHdr().setCreDtTm(offsetDateTime); assertTrue(mx.message().contains("2021-09-19T12:13:14+00:00")); } @Test public void testZeroNanosecondsWithZulu() { MxPacs00800108 mx = new MxPacs00800108(); // Test zero nanoseconds with Zulu time OffsetDateTime offsetDateTime = OffsetDateTime.parse("2021-09-19T12:13:14.000000000Z"); mx.getFIToFICstmrCdtTrf().getGrpHdr().setCreDtTm(offsetDateTime); assertFalse(mx.message().contains("2021-09-19T12:13:14.Z")); } @Test public void testZeroNanosecondsWithOffset() { MxPacs00800108 mx = new MxPacs00800108(); // Test zero nanoseconds with specific offset OffsetDateTime offsetDateTime = OffsetDateTime.parse("2021-09-19T12:13:14.000000000-03:00"); mx.getFIToFICstmrCdtTrf().getGrpHdr().setCreDtTm(offsetDateTime); assertTrue(mx.message().contains("2021-09-19T12:13:14-03:00")); }
1-69: 🧹 Nitpick (assertive)
Add documentation about Jira issue 1566
Please add a class-level Javadoc comment explaining the purpose of this test class and referencing the specific issue it addresses (Jira 1566). This will help maintain context for future maintenance.
Example:
/** * Test cases for Jira issue 1566: [Brief description of the issue] * * Validates the handling of date-time values with nanosecond precision * in MxPacs00800108 messages, ensuring proper serialization and * deserialization of various time formats. */
14-37: 🧹 Nitpick (assertive)
Fix typo in test data
The message ID "TETS" appears to be a typo.
- "msgId": "TETS", + "msgId": "TEST",📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.String json = "{\n" + " \"fiToFICstmrCdtTrf\": {\n" + " \"grpHdr\": {\n" + " \"msgId\": \"TEST\",\n" + " \"creDtTm\": {\n" + " \"dateTime\": {\n" + " \"date\": {\n" + " \"year\": 2023,\n" + " \"month\": 8,\n" + " \"day\": 10\n" + " },\n" + " \"time\": {\n" + " \"hour\": 23,\n" + " \"minute\": 35,\n" + " \"second\": 53,\n" + " \"nano\": 2312\n" + " }\n" + " },\n" + " \"offset\": {\n" + " \"totalSeconds\": -10800\n" + " }\n" + " }\n" + " }\n" + " }\n" + "}";
39-44: 🧹 Nitpick (assertive)
Remove unnecessary method call and improve test clarity
- The
message()
call on line 41 is unused and can be removed- Consider adding a descriptive comment explaining what aspect of the date-time format is being verified
- mx.message(); assertEquals( + // Verify correct deserialization of date-time with nanosecond precision "2023-08-10T23:35:53.000002312-03:00", mx.getFIToFICstmrCdtTrf().getGrpHdr().getCreDtTm().toString());📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.MxPacs00800108 mx = MxPacs00800108.fromJson(json); assertEquals( // Verify correct deserialization of date-time with nanosecond precision "2023-08-10T23:35:53.000002312-03:00", mx.getFIToFICstmrCdtTrf().getGrpHdr().getCreDtTm().toString());
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/ZuluOffsetDateTimeAdapterUnmarshallingTest.java (3)
28-90: 🧹 Nitpick (assertive)
Enhance test coverage with additional scenarios.
While the current tests cover basic scenarios well, consider adding the following test cases for more robust validation:
- Edge cases:
@Test public void testUnmarshallNull() { assertThrows(IllegalArgumentException.class, () -> adapter.unmarshal(null)); } @Test public void testUnmarshallInvalidFormat() { assertThrows(DateTimeParseException.class, () -> adapter.unmarshal("invalid-date")); }
- Maximum precision handling:
@Test public void testUnmarshallMaxPrecision() { OffsetDateTime datetime = adapter.unmarshal("2022-03-04T12:50:08.123456789Z"); assertEquals(123456789, datetime.getNano()); }
- Positive UTC offset:
@Test public void testUnmarshallPositiveOffset() { OffsetDateTime datetime = adapter.unmarshal("2022-03-04T12:50:08+05:30"); assertEquals(19800, datetime.getOffset().getTotalSeconds()); // +5:30 hours }
28-90: 🧹 Nitpick (assertive)
Consider extracting test data to constants.
The test methods use the same date-time components repeatedly. Consider extracting these to constants for better maintainability:
public class ZuluOffsetDateTimeAdapterUnmarshallingTest { + private static final int EXPECTED_YEAR = 2022; + private static final Month EXPECTED_MONTH = Month.MARCH; + private static final int EXPECTED_DAY = 4; + private static final int EXPECTED_HOUR = 12; + private static final int EXPECTED_MINUTE = 50; + private static final int EXPECTED_SECOND = 8; + private static final String BASE_DATE_TIME = "2022-03-04T12:50:08"; private final ZuluOffsetDateTimeAdapter adapter = new ZuluOffsetDateTimeAdapter();Committable suggestion skipped: line range outside the PR's diff.
24-27: 🧹 Nitpick (assertive)
Add class-level documentation.
Consider adding Javadoc to describe the purpose of this test class and its role in validating the date-time format handling, especially given the transition from Java 8 to 11 and JAXB changes.
+/** + * Unit tests for {@link ZuluOffsetDateTimeAdapter} focusing on unmarshalling various + * date-time string formats into {@link OffsetDateTime} objects. + * <p> + * These tests ensure correct handling of: + * <ul> + * <li>Fractional seconds</li> + * <li>UTC and custom timezone offsets</li> + * <li>Date-time strings without offsets</li> + * </ul> + */ public class ZuluOffsetDateTimeAdapterUnmarshallingTest {📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements./** * Unit tests for {@link ZuluOffsetDateTimeAdapter} focusing on unmarshalling various * date-time string formats into {@link OffsetDateTime} objects. * <p> * These tests ensure correct handling of: * <ul> * <li>Fractional seconds</li> * <li>UTC and custom timezone offsets</li> * <li>Date-time strings without offsets</li> * </ul> */ public class ZuluOffsetDateTimeAdapterUnmarshallingTest { private final ZuluOffsetDateTimeAdapter adapter = new ZuluOffsetDateTimeAdapter();
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeJsonAdapterTest.java (2)
78-83:
⚠️ Potential issueUpdate deprecated JsonParser usage and improve JSON comparison
The implementation has several issues that should be addressed:
- Uses deprecated JsonParser constructor
- Missing null checks for adapter results
- Uses direct string comparison for JSON which might be fragile
Here's the suggested fix:
private void testSerializationAndDeserializationImpl(String jsonInput, String jsonResult) { - JsonElement jsonElement = new JsonParser().parse(jsonInput); + JsonElement jsonElement = JsonParser.parseString(jsonInput); OffsetTime offsetTime = adapter.deserialize(jsonElement, null, null); + // Add null check + if (offsetTime == null) { + throw new AssertionError("Failed to deserialize: " + jsonInput); + } JsonElement valueDateResult = adapter.serialize(offsetTime, null, null); - assertEquals(jsonResult, valueDateResult.toString()); + // Compare JSON structures instead of strings + assertEquals( + JsonParser.parseString(jsonResult), + valueDateResult, + "JSON serialization mismatch" + ); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private void testSerializationAndDeserializationImpl(String jsonInput, String jsonResult) { JsonElement jsonElement = JsonParser.parseString(jsonInput); OffsetTime offsetTime = adapter.deserialize(jsonElement, null, null); // Add null check if (offsetTime == null) { throw new AssertionError("Failed to deserialize: " + jsonInput); } JsonElement valueDateResult = adapter.serialize(offsetTime, null, null); // Compare JSON structures instead of strings assertEquals( JsonParser.parseString(jsonResult), valueDateResult, "JSON serialization mismatch" ); }
31-76: 🧹 Nitpick (assertive)
Consider using @ParameterizedTest for better test structure
While the test coverage is comprehensive, the test method could benefit from using JUnit's parameterized tests to make it more maintainable and readable.
Here's how you could refactor it:
- @Test - void testSerializationAndDeserialization() { - int systemOffsetSeconds = ZoneId.systemDefault() - .getRules() - .getStandardOffset(Instant.now()) - .getTotalSeconds(); - - // without offset, nano 0 - String jsonActual = "{\"time\":{\"hour\":12,\"minute\":13,\"second\":14,\"nano\":0}}"; - String jsonExpected = - "{\"time\":{\"hour\":12,\"minute\":13,\"second\":14,\"nano\":0},\"offset\":{\"totalSeconds\":" - + systemOffsetSeconds + "}}"; - testSerializationAndDeserializationImpl(jsonActual, jsonExpected); + private static Stream<Arguments> timeTestCases() { + int systemOffsetSeconds = ZoneId.systemDefault() + .getRules() + .getStandardOffset(Instant.now()) + .getTotalSeconds(); + + return Stream.of( + // without offset, nano 0 + Arguments.of( + "{\"time\":{\"hour\":12,\"minute\":13,\"second\":14,\"nano\":0}}", + "{\"time\":{\"hour\":12,\"minute\":13,\"second\":14,\"nano\":0},\"offset\":{\"totalSeconds\":" + systemOffsetSeconds + "}}" + ), + // Add other test cases here... + ); + } + + @ParameterizedTest + @MethodSource("timeTestCases") + void testSerializationAndDeserialization(String input, String expected) { + testSerializationAndDeserializationImpl(input, expected); + }Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/adapters/TypeAdaptersConfiguration.java (1)
112-120: 🧹 Nitpick (assertive)
Consider documenting adapter order significance
The implementation correctly adds new adapters to the list, but consider documenting whether the order of adapters in the returned list is significant for the marshalling/unmarshalling process.
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetDateTimeJsonAdapterTest.java (2)
85-91:
⚠️ Potential issueUpdate deprecated JsonParser usage.
The
JsonParser.parse()
method is deprecated. Replace it withJsonParser.parseString()
for better forward compatibility.Here's the fix:
private void testSerializationAndDeserializationImpl(String jsonInput, String jsonResult) { - JsonElement jsonElement = new JsonParser().parse(jsonInput); + JsonElement jsonElement = JsonParser.parseString(jsonInput); OffsetDateTime offsetDateTime = adapter.deserialize(jsonElement, null, null); JsonElement valueDateResult = adapter.serialize(offsetDateTime, null, null); assertEquals(jsonResult, valueDateResult.toString()); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private void testSerializationAndDeserializationImpl(String jsonInput, String jsonResult) { JsonElement jsonElement = JsonParser.parseString(jsonInput); OffsetDateTime offsetDateTime = adapter.deserialize(jsonElement, null, null); JsonElement valueDateResult = adapter.serialize(offsetDateTime, null, null); assertEquals(jsonResult, valueDateResult.toString()); } }
30-83: 🧹 Nitpick (assertive)
Consider refactoring test cases for better maintainability and reliability.
While the test coverage is comprehensive, consider these improvements:
- Use
@ParameterizedTest
to organize test cases more efficiently- Extract test data as constants for better maintainability
- Add edge cases for date-time boundaries
- Remove system timezone dependency to ensure consistent test behavior across different environments
Here's an example refactor for the first test case:
+ private static final String EXPECTED_FORMAT = + "{\"dateTime\":{\"date\":{\"year\":%d,\"month\":%d,\"day\":%d}," + + "\"time\":{\"hour\":%d,\"minute\":%d,\"second\":%d,\"nano\":%d}," + + "\"offset\":{\"totalSeconds\":%d}}}"; + @ParameterizedTest + @CsvSource({ + "2019,4,28,12,13,14,0", + "2019,4,28,12,13,14,123456789" + }) + void testSerializationWithDifferentPrecisions(int year, int month, int day, + int hour, int minute, int second, int nano) { + String jsonActual = String.format(EXPECTED_FORMAT, + year, month, day, hour, minute, second, nano, 0); + String jsonExpected = String.format(EXPECTED_FORMAT, + year, month, day, hour, minute, second, nano, + ZoneOffset.systemDefault().getTotalSeconds()); + testSerializationAndDeserializationImpl(jsonActual, jsonExpected); + }Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/ZuluOffsetDateTimeAdapterMarshallingTest.java (4)
50-50: 🧹 Nitpick (assertive)
Follow Java naming conventions for test methods
The test method names use underscores instead of camelCase, which is not consistent with Java naming conventions.
- public void test_marshall_message_with_BAH1_and_CreDt_Zulu() + public void testMarshallMessageWithBAH1AndCreDtZulu() - public void test_marshall_message_with_BAH1_and_CreDt_OffsetDateTime() + public void testMarshallMessageWithBAH1AndCreDtOffsetDateTime()Also applies to: 85-85
81-82: 🧹 Nitpick (assertive)
Enhance test comments with specific scenarios
The current comments could be more descriptive about the specific scenarios being tested.
- // After parse and marshalling, CreDtTm should be preserved as Zulu date time + // Verify that when input is already in Zulu format (Z), it remains unchanged after marshalling - // After parse and marshalling, the original CreDtTm should be converted to 00:00 offset Zulu date time + // Verify that when input has -03:00 offset, it's correctly converted to equivalent Zulu time (+3 hours)Also applies to: 116-117
26-26: 🧹 Nitpick (assertive)
Consider adding tests for edge cases
While the current test coverage is good, consider adding tests for these scenarios:
- Leap year dates
- Daylight Saving Time transitions
- Extreme time zones (e.g., +14:00, -12:00)
- Invalid input handling
Would you like me to help generate these additional test cases?
28-28: 🧹 Nitpick (assertive)
Consider making the adapter field final
Since the adapter instance is not modified after initialization, consider making it final for better immutability.
- private ZuluOffsetDateTimeAdapter adapter = new ZuluOffsetDateTimeAdapter(); + private final ZuluOffsetDateTimeAdapter adapter = new ZuluOffsetDateTimeAdapter();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private final ZuluOffsetDateTimeAdapter adapter = new ZuluOffsetDateTimeAdapter();
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetDateTimeAdapterTest.java (4)
25-49: 🧹 Nitpick (assertive)
Consider using parameterized tests for better maintainability.
The multiple test cases could be more maintainable using JUnit's
@ParameterizedTest
. This would make it easier to add new test cases and improve readability.Example refactor:
@ParameterizedTest @CsvSource({ "2021-09-19T12:13:14, 2021-09-19T12:13:14+01:00", "2021-09-19T12:13:14.1, 2021-09-19T12:13:14.1+01:00", // ... other test cases }) void testOffsetDateTime(String input, String expected) throws Exception { testDateTimeImpl(input, expected); }
70-71: 🧹 Nitpick (assertive)
Consider adding a proper test for the nano precision limit.
Instead of commenting out the test case, consider adding a test that verifies the parser throws an appropriate exception when given more than 9 digits for nanoseconds.
Example:
@Test void testNanoPrecisionLimit() { assertThrows(JAXBException.class, () -> testDateTimeImpl("2018-01-15T17:30:33.0000000000+02:00", "")); }
51-56: 🧹 Nitpick (assertive)
Consider optimizing the helper method.
A few suggestions to improve the helper method:
- The
OffsetDateTimeAdapter
could be a class field to avoid creating new instances.- The generic
Exception
could be replaced with more specific exceptions.Example refactor:
+ private final OffsetDateTimeAdapter offsetDateTimeAdapter = new OffsetDateTimeAdapter(); + - private void testDateTimeImpl(String value, String valueResult) throws Exception { - OffsetDateTimeAdapter offsetDateTimeAdapter = new OffsetDateTimeAdapter(); + private void testDateTimeImpl(String value, String valueResult) throws JAXBException { OffsetDateTime offsetDateTime = offsetDateTimeAdapter.unmarshal(value); String valueDateResult = offsetDateTimeAdapter.marshal(offsetDateTime); assertEquals(valueResult, valueDateResult); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private final OffsetDateTimeAdapter offsetDateTimeAdapter = new OffsetDateTimeAdapter(); private void testDateTimeImpl(String value, String valueResult) throws JAXBException { OffsetDateTime offsetDateTime = offsetDateTimeAdapter.unmarshal(value); String valueDateResult = offsetDateTimeAdapter.marshal(offsetDateTime); assertEquals(valueResult, valueDateResult); }
15-21: 🛠️ Refactor suggestion
Consider using a fixed timezone instead of system default.
Using
ZoneOffset.systemDefault()
makes the test dependent on the system's timezone settings. This could lead to non-deterministic behavior when running tests in different environments.Consider using a fixed timezone:
- ZoneId zoneId = ZoneOffset.systemDefault(); + ZoneId zoneId = ZoneOffset.of("+01:00"); // Use a fixed offset for deterministic testing📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.ZoneId zoneId = ZoneOffset.of("+01:00"); // Use a fixed offset for deterministic testing OffsetDateTime dateTime = LocalDateTime.parse("2021-09-19T12:13:14") .atZone(zoneId) .withEarlierOffsetAtOverlap() .withFixedOffsetZone() .withEarlierOffsetAtOverlap() .toOffsetDateTime();
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeAdapterTest.java (5)
28-56: 🧹 Nitpick (assertive)
Consider adding edge case tests for unmarshalling
While the current tests cover the basic scenarios well, consider adding tests for:
- Invalid time formats
- Out-of-range values (e.g., hours > 23)
- Extreme timezone offsets
@Test public void testUnmarshallInvalidFormat() { assertThrows(Exception.class, () -> adapter.unmarshal("25:00:00+01:00")); assertThrows(Exception.class, () -> adapter.unmarshal("12:61:00+01:00")); assertThrows(Exception.class, () -> adapter.unmarshal("12:00:61+01:00")); assertThrows(Exception.class, () -> adapter.unmarshal("12:00:00+99:00")); }
26-26: 🧹 Nitpick (assertive)
Consider making the adapter field final
Since the adapter instance is never reassigned, it should be marked as final for better immutability.
- private OffsetTimeAdapter adapter = new OffsetTimeAdapter(); + private final OffsetTimeAdapter adapter = new OffsetTimeAdapter();📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private final OffsetTimeAdapter adapter = new OffsetTimeAdapter();
108-113: 🧹 Nitpick (assertive)
Fix inconsistent variable naming
The variable
OffsetTime
should follow Java naming conventions (camelCase for variables).- OffsetTime OffsetTime = offsetTimeAdapter.unmarshal(value); + OffsetTime offsetTime = offsetTimeAdapter.unmarshal(value);📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private void testTimeImpl(String value, String valueResult) throws Exception { OffsetTimeAdapter offsetTimeAdapter = new OffsetTimeAdapter(); OffsetTime offsetTime = offsetTimeAdapter.unmarshal(value); String valueDateResult = offsetTimeAdapter.marshal(offsetTime); assertEquals(valueResult, valueDateResult); }
127-129: 🧹 Nitpick (assertive)
Document nano precision limitation more explicitly
Consider adding a more detailed comment explaining the 9-character limitation for nanoseconds and its origin (Java's OffsetTime implementation).
- // testTimeImpl("17:30:33.0000000000+02:00", "17:30:33+02:00"); //Cannot parse more - // than 9 characters in nano value + // Note: Java's OffsetTime implementation limits nanosecond precision to 9 digits (0-999999999). + // Attempting to parse more than 9 digits will result in a DateTimeParseException. + // testTimeImpl("17:30:33.0000000000+02:00", "17:30:33+02:00");📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.// Note: Java's OffsetTime implementation limits nanosecond precision to 9 digits (0-999999999). // Attempting to parse more than 9 digits will result in a DateTimeParseException. // testTimeImpl("17:30:33.0000000000+02:00", "17:30:33+02:00");
58-76: 🧹 Nitpick (assertive)
Improve test readability with descriptive comments
The test methods would benefit from more detailed comments explaining the expected behavior, especially for the system default timezone case.
@Test public void testMarshallNoOffset() throws Exception { + // When no offset is provided, the system default offset should be used + // This test verifies that the marshalled output includes the system offset DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS][XXX]"); ZoneOffset offset = ZoneId.systemDefault().getRules().getOffset(Instant.now());📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.@Test public void testMarshallNoFractionOfSecondsAndWithFractionOfSeconds() throws Exception { ZoneOffset zoneOffSet = ZoneOffset.of("-03:00"); DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS][XXX]"); OffsetTime offsetTime = LocalTime.parse("12:50:08", dateTimeFormatter).atOffset(zoneOffSet); assertEquals("12:50:08-03:00", adapter.marshal(offsetTime)); offsetTime = LocalTime.parse("12:50:08.123", dateTimeFormatter).atOffset(zoneOffSet); assertEquals("12:50:08.123-03:00", adapter.marshal(offsetTime)); } @Test public void testMarshallNoOffset() throws Exception { // When no offset is provided, the system default offset should be used // This test verifies that the marshalled output includes the system offset DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS][XXX]"); ZoneOffset offset = ZoneId.systemDefault().getRules().getOffset(Instant.now()); OffsetTime offsetTime = LocalTime.parse("12:50:08", dateTimeFormatter).atOffset(offset); assertEquals("12:50:08" + offset, adapter.marshal(offsetTime)); }
CHANGELOG.md (2)
3-5: 🧹 Nitpick (assertive)
Consider enhancing migration documentation for JSON serialization changes
While the changelog mentions the new
toJsonV9
andfromJsonV9
methods for backward compatibility, it would be helpful to provide more detailed migration guidance for users upgrading from Java 8.Consider adding:
- Example usage of the new methods
- Common migration scenarios
- Any gotchas or edge cases to watch out for
Would you like me to help draft detailed migration documentation?
🧰 Tools
🪛 Markdownlint
3-3: Expected: h2; Actual: h4
Heading levels should only increment by one level at a time(MD001, heading-increment)
4-4: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
3-3: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
4-4: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
54-58: 🧹 Nitpick (assertive)
Consider restructuring migration-related changes for better visibility
The migration changes (Java 11, Jakarta EE 10, DateTime handling) are significant breaking changes. Consider restructuring this section to:
- Group all migration-related changes under a dedicated "Migration Guide" section
- Highlight potential breaking changes more prominently
- Provide step-by-step migration instructions
Would you like me to propose a restructured format for the migration-related changes?
🧰 Tools
🪛 Markdownlint
54-54: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
55-55: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
56-56: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
57-57: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
58-58: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
54-54: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/LocalDateAdapterTest.java (4)
15-20: 🧹 Nitpick (assertive)
Consider adding edge cases to basic date conversion test.
While the basic conversion test is good, consider adding test cases for:
- Null input handling
- Invalid date format handling
- Boundary dates (e.g., leap years, year transitions)
43-155: 🛠️ Refactor suggestion
Refactor test to improve maintainability and security.
The test method is too long and tests multiple aspects. Consider splitting it into smaller, focused test methods:
- One for basic XML parsing
- One for each custom format pattern
The test data contains potentially sensitive information (BIC codes). Consider using dummy/mock data instead.
The XML test data should be moved to a separate test resource file.
Example refactor:
@Test public void testLocalDate_BasicXmlParsing() { String xml = loadTestResource("basic_payment.xml"); final MxPacs00900110 mx = MxPacs00900110.parse(xml); assertTrue(mx.message().contains("IntrBkSttlmDt>2023-01-24</")); } @Test public void testLocalDate_CustomSlashFormat() { MxPacs00900110 mx = createTestMessage(); MxWriteConfiguration conf = new MxWriteConfiguration(); conf.adapters.dateAdapter = new IsoDateAdapter( new LocalDateAdapter((DateTimeFormatter.ofPattern("yyyy/MM/dd"))) ); assertTrue(mx.message(conf).contains("IntrBkSttlmDt>2023/01/24</")); }
157-174: 🧹 Nitpick (assertive)
LGTM! Consider adding more offset scenarios.
The test effectively verifies offset handling, but could be enhanced by testing:
- Different offset values (positive and negative)
- UTC offset
- Daylight saving time transitions
37-41:
⚠️ Potential issueFix invalid test data in
testLocalDateDropTime
.The test contains an invalid seconds value "65" in the timestamp "2021-09-19T23:34:65.233Z". Seconds should be in the range 0-59.
- LocalDate localDate = localDateAdapter.unmarshal("2021-09-19T23:34:65.233Z"); + LocalDate localDate = localDateAdapter.unmarshal("2021-09-19T23:34:45.233Z");📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.void testLocalDateDropTime() throws Exception { LocalDateAdapter localDateAdapter = new LocalDateAdapter(); LocalDate localDate = localDateAdapter.unmarshal("2021-09-19T23:34:45.233Z"); assertEquals("2021-09-19", localDateAdapter.marshal(localDate)); }
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/YearMonthAdapterTest.java (4)
29-93: 🧹 Nitpick (assertive)
Consider improving test data and readability
- Consider using clearly fake BIC codes (e.g., 'TESTBICXXXXX') instead of what appears to be real or sensitive BIC codes.
- Consider moving the test XML to a separate resource file for better maintainability.
- The test could benefit from additional edge cases:
- Invalid date formats
- Boundary dates (e.g., 9999-12)
- Null/empty date values
95-160: 🛠️ Refactor suggestion
Reduce duplication and enhance test coverage
- Consider extracting common XML test data to a shared method or resource file to reduce duplication.
- The adapter configuration could be moved to a
@BeforeEach
setup method:private MxWriteConfiguration conf; @BeforeEach void setup() { conf = new MxWriteConfiguration(); conf.documentPrefix = null; conf.headerPrefix = null; }
- Consider adding tests for:
- Invalid date format patterns
- Pattern with unsupported format specifiers
162-192: 🧹 Nitpick (assertive)
Enhance year format testing
Consider adding test cases for:
- Year boundary conditions:
- Minimum valid year (e.g., 1900)
- Maximum valid year (e.g., 9999)
- Invalid scenarios:
- Years before allowed range
- Non-numeric years
Example test case:
@Test void testYear_BoundaryConditions() { // Test minimum year assertDoesNotThrow(() -> { conf.adapters.yearAdapter = new IsoYearAdapter( new YearAdapter(DateTimeFormatter.ofPattern("yyyy"))); // Test parsing and formatting }); // Test maximum year // Test invalid years }
27-192: 🧹 Nitpick (assertive)
Consider architectural improvements for better test structure
The test class would benefit from:
- A more structured approach using JUnit 5 features:
@Nested
classes to group related tests@ParameterizedTest
for testing multiple date formats- Test lifecycle methods (
@BeforeEach
) for common setup- Consider creating a test utilities class for common XML operations
- Add explicit test cases for error handling scenarios
This would improve maintainability and make the test coverage more comprehensive.
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/BusinessAppHdrV01.java (2)
226-226: 🧹 Nitpick (assertive)
Consider using try-with-resources for adapter management
While the adapter switching logic works, it could be improved using a try-with-resources pattern to ensure the original adapter is always restored, even in case of exceptions.
- IsoDateTimeAdapter currentAdapter = params.adapters.dateTimeAdapter; - params.adapters.dateTimeAdapter = new IsoDateTimeAdapter(new ZuluOffsetDateTimeAdapter()); + try (AdapterContext ac = new AdapterContext(params.adapters)) { + params.adapters.dateTimeAdapter = new IsoDateTimeAdapter(new ZuluOffsetDateTimeAdapter());You'll need to add this helper class:
private static class AdapterContext implements AutoCloseable { private final IsoDateTimeAdapter original; private final MxAdapters adapters; AdapterContext(MxAdapters adapters) { this.adapters = adapters; this.original = adapters.dateTimeAdapter; } @Override public void close() { adapters.dateTimeAdapter = original; } }
174-187: 🧹 Nitpick (assertive)
Consider adding null validation in creationDate()
While the changes to use
OffsetDateTime
are good, consider adding null validation in thecreationDate()
method to prevent potential NPEs.@Override public OffsetDateTime creationDate() { - return this.getCreDt(); + return Objects.requireNonNull(this.getCreDt(), "Creation date must not be null"); }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.public OffsetDateTime creationDate() { return Objects.requireNonNull(this.getCreDt(), "Creation date must not be null"); } /** * Sets the creation date. * * @param overwrite if true, the creation date will always be set overwriting any previous value; * @see #setCreDt(OffsetDateTime) */ @Override public void setCreationDate(boolean overwrite) { if (this.getCreDt() == null || overwrite) { this.setCreDt(OffsetDateTime.now(ZoneOffset.UTC));
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/AppHdrFactory.java (3)
115-115:
⚠️ Potential issueMaintain consistency in timezone handling across header versions.
V02 and V03 headers use
OffsetDateTime.now()
without an explicit timezone, while V01 uses UTC. This inconsistency could lead to timestamps in different timezones across header versions.Apply this change to maintain consistency:
- h.setCreDt(OffsetDateTime.now()); + h.setCreDt(OffsetDateTime.now(ZoneOffset.UTC));Also applies to: 162-162
69-69: 🧹 Nitpick (assertive)
Consider implementing a centralized timezone strategy.
The current implementation shows inconsistent timezone handling across different header versions. Consider:
- Creating a centralized utility method for timestamp creation
- Documenting the timezone requirements for each header version
- Adding timezone validation in the header classes
This will ensure consistent and maintainable timezone handling across the codebase.
Example implementation:
private static OffsetDateTime createTimestamp() { return OffsetDateTime.now(ZoneOffset.UTC); }Also applies to: 115-115, 162-162, 209-209, 250-250
209-209:
⚠️ Potential issueMaintain consistency in timezone handling for V04 headers.
Similar to V02 and V03, V04 uses
OffsetDateTime.now()
without an explicit timezone, which should be consistent with V01's UTC usage.Apply this change:
- h.setCreDt(OffsetDateTime.now()); + h.setCreDt(OffsetDateTime.now(ZoneOffset.UTC));📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.h.setCreDt(OffsetDateTime.now(ZoneOffset.UTC));
build.gradle (2)
487-496: 🧹 Nitpick (assertive)
LGTM: Multi-version testing configuration
Good practice to test against multiple Java versions. Consider adding a comment explaining the testing strategy and requirements for each Java version.
+// Test against multiple Java versions to ensure forward compatibility +// Java 17 (LTS): Primary target for future migration +// Java 19: Latest version testing for early detection of issues task('testOn17', type: Test) {📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.// Test against multiple Java versions to ensure forward compatibility // Java 17 (LTS): Primary target for future migration // Java 19: Latest version testing for early detection of issues task('testOn17', type: Test) { javaLauncher = javaToolchains.launcherFor { languageVersion = JavaLanguageVersion.of(17) } } task('testOn19', type: Test) { javaLauncher = javaToolchains.launcherFor { languageVersion = JavaLanguageVersion.of(19) }
210-214: 🧹 Nitpick (assertive)
LGTM: Improved jar filtering with logging
Good addition of jar filtering and logging. Consider using a constant for the '.jar' extension for maintainability.
- if (it.name.endsWith('.jar')) { + static final String JAR_EXTENSION = '.jar' + if (it.name.endsWith(JAR_EXTENSION)) {Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java (5)
32-267: 🧹 Nitpick (assertive)
Consider adding test cases for error scenarios and timezone edge cases
The test suite would benefit from additional test cases:
- Error scenarios:
- Invalid date/time formats
- Null values
- Invalid timezone offsets
- Timezone edge cases:
- DST transitions
- UTC±14 boundaries
- Leap seconds
61-61:
⚠️ Potential issueFix incorrect ChronoField comparison
The test is using
DAY_OF_WEEK
instead ofDAY_OF_YEAR
for date comparison, which could lead to false positives.- assertEquals(intrBkSttlmDt1.get(ChronoField.DAY_OF_WEEK), intrBkSttlmDt2.get(ChronoField.DAY_OF_WEEK)); + assertEquals(intrBkSttlmDt1.get(ChronoField.DAY_OF_YEAR), intrBkSttlmDt2.get(ChronoField.DAY_OF_YEAR));📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.assertEquals(intrBkSttlmDt1.get(ChronoField.DAY_OF_YEAR), intrBkSttlmDt2.get(ChronoField.DAY_OF_YEAR));
171-180: 🛠️ Refactor suggestion
Remove redundant time retrieval and comparison
The code retrieves the same time value twice (
clsTm1
andclsTm2
) from the same object, making the comparison redundant.- OffsetTime clsTm1 = mx2.getFIToFICstmrCdtTrf() - .getCdtTrfTxInf() - .get(0) - .getSttlmTmReq() - .getCLSTm(); - OffsetTime clsTm2 = mx2.getFIToFICstmrCdtTrf() - .getCdtTrfTxInf() - .get(0) - .getSttlmTmReq() - .getCLSTm(); + OffsetTime clsTm = mx2.getFIToFICstmrCdtTrf() + .getCdtTrfTxInf() + .get(0) + .getSttlmTmReq() + .getCLSTm(); - assertNotNull(clsTm2); - assertEquals(clsTm1.get(ChronoField.HOUR_OF_DAY), clsTm2.get(ChronoField.HOUR_OF_DAY)); - assertEquals(clsTm1.get(ChronoField.MINUTE_OF_HOUR), clsTm2.get(ChronoField.MINUTE_OF_HOUR)); - assertEquals(clsTm1.get(ChronoField.SECOND_OF_MINUTE), clsTm2.get(ChronoField.SECOND_OF_MINUTE)); - assertEquals(clsTm1.get(ChronoField.NANO_OF_SECOND), clsTm2.get(ChronoField.NANO_OF_SECOND)); - assertEquals(clsTm1.getOffset(), clsTm2.getOffset()); + assertNotNull(clsTm); + OffsetTime expectedTime = OffsetTime.parse("12:13:14.123" + systemOffset); + assertEquals(expectedTime.get(ChronoField.HOUR_OF_DAY), clsTm.get(ChronoField.HOUR_OF_DAY)); + assertEquals(expectedTime.get(ChronoField.MINUTE_OF_HOUR), clsTm.get(ChronoField.MINUTE_OF_HOUR)); + assertEquals(expectedTime.get(ChronoField.SECOND_OF_MINUTE), clsTm.get(ChronoField.SECOND_OF_MINUTE)); + assertEquals(expectedTime.get(ChronoField.NANO_OF_SECOND), clsTm.get(ChronoField.NANO_OF_SECOND)); + assertEquals(expectedTime.getOffset(), clsTm.getOffset());Committable suggestion skipped: line range outside the PR's diff.
286-297: 🧹 Nitpick (assertive)
Add input validation to helper methods
The
setDatesIntoMessage
method should validate its input parameters to ensure they're not null and maintain consistent state.private MxPacs00800102 setDatesIntoMessage( final LocalDate localDate, OffsetTime offsetTime, OffsetDateTime offsetDateTime) { + Objects.requireNonNull(localDate, "localDate must not be null"); + Objects.requireNonNull(offsetTime, "offsetTime must not be null"); + Objects.requireNonNull(offsetDateTime, "offsetDateTime must not be null"); + final MxPacs00800102 mx = new MxPacs00800102(); mx.setFIToFICstmrCdtTrf(new FIToFICustomerCreditTransferV02());📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private MxPacs00800102 setDatesIntoMessage( final LocalDate localDate, OffsetTime offsetTime, OffsetDateTime offsetDateTime) { Objects.requireNonNull(localDate, "localDate must not be null"); Objects.requireNonNull(offsetTime, "offsetTime must not be null"); Objects.requireNonNull(offsetDateTime, "offsetDateTime must not be null"); final MxPacs00800102 mx = new MxPacs00800102(); mx.setFIToFICstmrCdtTrf(new FIToFICustomerCreditTransferV02()); mx.getFIToFICstmrCdtTrf().setGrpHdr(new GroupHeader33()); mx.getFIToFICstmrCdtTrf().getGrpHdr().setCreDtTm(offsetDateTime); // date time mx.getFIToFICstmrCdtTrf().getGrpHdr().setIntrBkSttlmDt(localDate); // date mx.getFIToFICstmrCdtTrf().addCdtTrfTxInf(new CreditTransferTransactionInformation11()); mx.getFIToFICstmrCdtTrf().getCdtTrfTxInf().get(0).setSttlmTmReq(new SettlementTimeRequest2()); mx.getFIToFICstmrCdtTrf().getCdtTrfTxInf().get(0).getSttlmTmReq().setCLSTm(offsetTime); // time return mx; }
299-321: 🛠️ Refactor suggestion
Improve custom adapters for consistency and reliability
- Make both adapter classes consistently package-private or private inner classes
- Use fixed timestamps instead of
now()
to prevent flaky tests- class TestCustomDateAdapter extends XmlAdapter<String, LocalDate> { + private static class TestCustomDateAdapter extends XmlAdapter<String, LocalDate> { @Override public LocalDate unmarshal(String v) { - return LocalDate.now(); + return LocalDate.of(2021, 10, 19); } } - public class TestCustomDateTimeAdapter extends XmlAdapter<String, OffsetDateTime> { + private static class TestCustomDateTimeAdapter extends XmlAdapter<String, OffsetDateTime> { @Override public OffsetDateTime unmarshal(String v) { - return OffsetDateTime.now(); + return OffsetDateTime.parse("2021-10-19T12:13:14+00:00"); } }📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.private static class TestCustomDateAdapter extends XmlAdapter<String, LocalDate> { @Override public LocalDate unmarshal(String v) { return LocalDate.of(2021, 10, 19); } @Override public String marshal(LocalDate cal) { return "foobar"; } } private static class TestCustomDateTimeAdapter extends XmlAdapter<String, OffsetDateTime> { @Override public OffsetDateTime unmarshal(String v) { return OffsetDateTime.parse("2021-10-19T12:13:14+00:00"); } @Override public String marshal(OffsetDateTime offsetDateTime) { return "foobar"; } }
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/AppHdrParserTest.java (2)
82-83: 🧹 Nitpick (assertive)
Consider using Month enum for better readability
While the transition to
ChronoField
is good, the month assertion could be more readable using theMonth
enum.- assertEquals(9, h.getCrDate().get(ChronoField.MONTH_OF_YEAR)); + assertEquals(Month.SEPTEMBER.getValue(), h.getCrDate().get(ChronoField.MONTH_OF_YEAR));📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.assertEquals(2006, h.getCrDate().get(ChronoField.YEAR)); assertEquals(Month.SEPTEMBER.getValue(), h.getCrDate().get(ChronoField.MONTH_OF_YEAR));
188-189: 🧹 Nitpick (assertive)
Apply consistent date testing improvements
Similar to the previous test method, consider using the
Month
enum for better readability. Additionally, since the test XML contains a complete timestamp, consider testing more fields (day, hour, etc.) for better coverage.assertEquals(2015, bah.getCreDt().get(ChronoField.YEAR)); - assertEquals(8, bah.getCreDt().get(ChronoField.MONTH_OF_YEAR)); + assertEquals(Month.AUGUST.getValue(), bah.getCreDt().get(ChronoField.MONTH_OF_YEAR)); + assertEquals(27, bah.getCreDt().get(ChronoField.DAY_OF_MONTH)); + assertEquals(8, bah.getCreDt().get(ChronoField.HOUR_OF_DAY)); + assertEquals(59, bah.getCreDt().get(ChronoField.MINUTE_OF_HOUR));📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.assertEquals(2015, bah.getCreDt().get(ChronoField.YEAR)); assertEquals(Month.AUGUST.getValue(), bah.getCreDt().get(ChronoField.MONTH_OF_YEAR)); assertEquals(27, bah.getCreDt().get(ChronoField.DAY_OF_MONTH)); assertEquals(8, bah.getCreDt().get(ChronoField.HOUR_OF_DAY)); assertEquals(59, bah.getCreDt().get(ChronoField.MINUTE_OF_HOUR));
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/DateTimeJsonAdapterBackwardCompatibilityTest.java (3)
399-434: 🧹 Nitpick (assertive)
Improve readability by extracting helper methods.
The test contains multiple chained method calls that make it harder to read and maintain. Consider extracting helper methods for accessing nested properties.
+ private LocalDate getTradeDt(MxSese02500109 mx) { + return mx.getSctiesSttlmTxConf() + .getTradDtls() + .getTradDt() + .getDt() + .getDt(); + } + @Test void allParsedDateTimeObjectsShouldMatch() { final MxSese02500109 fromV9 = (MxSese02500109) AbstractMX.fromJson(JSON_MX_V9); final MxSese02500109 fromV10 = (MxSese02500109) AbstractMX.fromJson(JSON_MX_V10); - LocalDate tradDtV9 = fromV9.getSctiesSttlmTxConf().getTradDtls().getTradDt().getDt().getDt(); - LocalDate tradDtV10 = fromV10.getSctiesSttlmTxConf().getTradDtls().getTradDt().getDt().getDt(); + LocalDate tradDtV9 = getTradeDt(fromV9); + LocalDate tradDtV10 = getTradeDt(fromV10);Committable suggestion skipped: line range outside the PR's diff.
380-397: 🧹 Nitpick (assertive)
Add Javadoc to clarify test purpose and expectations.
While the test name is descriptive, adding Javadoc would help clarify:
- The purpose of testing legacy format parsing
- The expected behavior when handling sentinel values
- The significance of the specific date-time values being tested
+ /** + * Verifies that the legacy XMLGregorianCalendar JSON format (V9) can be correctly parsed + * into the new date-time model, properly handling undefined time components and timezone information. + */ @Test void shouldBeAbleToParseJsonFromLegacyXMLGregorianCalendar() {📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements./** * Verifies that the legacy XMLGregorianCalendar JSON format (V9) can be correctly parsed * into the new date-time model, properly handling undefined time components and timezone information. */ @Test void shouldBeAbleToParseJsonFromLegacyXMLGregorianCalendar() { final MxSese02500109 mx = (MxSese02500109) AbstractMX.fromJson(JSON_MX_V9); assertThat(mx.getSctiesSttlmTxConf().getTradDtls().getTradDt().getDt().getDt()) .isNotNull(); OffsetDateTime dateTime = mx.getSctiesSttlmTxConf().getTradDtls().getFctvSttlmDt().getDt().getDtTm(); assertThat(dateTime).isNotNull(); assertThat(dateTime.getYear()).isEqualTo(2024); assertThat(dateTime.getMonthValue()).isEqualTo(7); assertThat(dateTime.getDayOfMonth()).isEqualTo(17); assertThat(dateTime.getHour()).isEqualTo(7); assertThat(dateTime.getMinute()).isEqualTo(26); assertThat(dateTime.getSecond()).isEqualTo(4); assertThat(dateTime.getNano()).isEqualTo(207459000); }
17-192: 🧹 Nitpick (assertive)
Consider using named constants for sentinel values.
The JSON_MX_V9 constant uses
-2147483648
as a sentinel value for undefined time components. This magic number could be confusing and should be replaced with a named constant for better readability.+ private static final int UNDEFINED_TIME_COMPONENT = -2147483648; private static final String JSON_MX_V9 = "{\n" + // ... (existing JSON with -2147483648 replaced with UNDEFINED_TIME_COMPONENT)
Committable suggestion skipped: line range outside the PR's diff.
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/AbstractMxJsonV9Test.java (5)
31-142: 🧹 Nitpick (assertive)
Consider using anonymized test data.
The XML test data contains what appears to be real-looking BIC codes (e.g., "FOORUS33") and addresses. Consider using clearly fictional data (e.g., "TESTUS33") to prevent any confusion with real financial institutions.
144-296: 🧹 Nitpick (assertive)
Improve test data maintainability.
The current approach of constructing JSON test data through string concatenation is error-prone and difficult to maintain. Consider:
- Using JSON files in the test resources directory
- Using a JSON library to construct test data programmatically
Example implementation:
private String loadJsonFromResource(String filename) { return new String(getClass().getResourceAsStream(filename).readAllBytes()); } // Usage @Test public void testMxToJson() { String jsonV10 = loadJsonFromResource("/testdata/mx_v10.json"); // ... rest of the test }
298-434: 🧹 Nitpick (assertive)
Reduce test data duplication.
The V9 and V10 JSON test data contain significant duplication. Consider creating a base JSON structure and only override the datetime-specific parts for each version.
Example approach:
private JsonObject createBaseJson() { // Create common JSON structure } private String createV9Json() { JsonObject json = createBaseJson(); // Override only V9-specific datetime fields return json.toString(); } private String createV10Json() { JsonObject json = createBaseJson(); // Override only V10-specific datetime fields return json.toString(); }
436-461: 🛠️ Refactor suggestion
Enhance test coverage with edge cases.
The current tests only cover the happy path. Consider adding tests for:
- Error handling (invalid JSON format, missing required fields)
- Edge cases for datetime values (timezone boundaries, maximum/minimum values)
- Null/empty value handling
Example additional tests:
@Test public void testMxFromInvalidJson() { assertThrows(JsonParseException.class, () -> MxPacs00900108.fromJson("{invalid}")); } @Test public void testMxWithExtremeDateTimeValues() { // Test with maximum timezone offset // Test with minimum timezone offset // Test with maximum nano value } @Test public void testMxWithNullValues() { // Test with null datetime fields // Test with empty strings }
24-28: 🧹 Nitpick (assertive)
Enhance class documentation with format differences.
Consider adding documentation that explains the key differences between V9 and V10 JSON formats, particularly regarding the datetime representation changes.
/** * Test for JSON conversion in the MX model (AbstractMX and subclasses). + * + * V9 format uses legacy datetime representation with: + * - timezone (in minutes) + * - fractionalSecond (as decimal) + * + * V10 format uses modern datetime representation with: + * - offset.totalSeconds (in seconds) + * - time.nano (in nanoseconds) * * @since 7.10.2 */📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements./** * Test for JSON conversion in the MX model (AbstractMX and subclasses). * * V9 format uses legacy datetime representation with: * - timezone (in minutes) * - fractionalSecond (as decimal) * * V10 format uses modern datetime representation with: * - offset.totalSeconds (in seconds) * - time.nano (in nanoseconds) * * @since 7.10.2 */
iso20022-core/src/main/java/com/prowidesoftware/swift/model/MxSwiftMessage.java (1)
21-21: 🧹 Nitpick (assertive)
Consider runtime environment compatibility
The migration to Jakarta persistence APIs requires ensuring that:
- The runtime environment supports Jakarta EE
- All dependent libraries are compatible with Jakarta EE
- The persistence provider (e.g., Hibernate) version supports Jakarta persistence
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 2
🔭 Outside diff range comments (1)
CHANGELOG.md (1)
Line range hint
3-66
: Improve markdown formatting for better readability.The changelog has several markdown formatting issues:
- Inconsistent heading levels (h4 vs h2)
- Missing blank lines around headings and lists
- Inconsistent list indentation
Consider using a markdown linter to maintain consistent formatting. Here's an example of the correct format:
-#### 10.2.5 - January 2025 - * (PW-2113) `MxParseUtils` added a method to extract... +## 10.2.5 - January 2025 + +* (PW-2113) `MxParseUtils` added a method to extract... +🧰 Tools
🪛 Markdownlint (0.37.0)
33-33: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
34-34: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
37-37: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
40-40: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
43-43: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
44-44: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
45-45: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
46-46: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
47-47: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
50-50: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
51-51: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
52-52: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
55-55: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
58-58: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
61-61: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
62-62: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
63-63: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
64-64: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
65-65: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
68-68: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
69-69: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
36-36: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
39-39: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
42-42: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
49-49: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
54-54: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
57-57: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
60-60: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
67-67: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines(MD022, blanks-around-headings)
37-37: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
40-40: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
43-43: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
50-50: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
55-55: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
58-58: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
61-61: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
68-68: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
69-69: null
Bare URL used(MD034, no-bare-urls)
📜 Review details
Configuration used: CodeRabbit UI
Review profile: ASSERTIVE
Plan: Pro
📒 Files selected for processing (3)
CHANGELOG.md
(2 hunks)build.gradle
(10 hunks)iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/MxParseUtils.java
(1 hunks)
🧰 Additional context used
🪛 LanguageTool
CHANGELOG.md
[grammar] ~20-~20: Did you mean “to Add”?
Context: ...he datetime to UTC offset if needed * Add support for Business Application Header...
(MISSING_TO_BEFORE_A_VERB)
🪛 Markdownlint (0.37.0)
CHANGELOG.md
3-3: Expected: h2; Actual: h4
Heading levels should only increment by one level at a time
(MD001, heading-increment)
4-4: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
5-5: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
6-6: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
7-7: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
8-8: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
9-9: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
12-12: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
17-17: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
18-18: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
19-19: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
20-20: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
23-23: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
24-24: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
3-3: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
11-11: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
16-16: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
22-22: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
26-26: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
4-4: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
12-12: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
17-17: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
23-23: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
37-37: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
40-40: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
43-43: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
44-44: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
45-45: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
46-46: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
47-47: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
50-50: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
51-51: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
52-52: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
55-55: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
58-58: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
61-61: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
62-62: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
63-63: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
64-64: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
65-65: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
36-36: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
39-39: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
42-42: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
49-49: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
54-54: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
57-57: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
60-60: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
37-37: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
40-40: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
43-43: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
50-50: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
55-55: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
58-58: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
61-61: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
🔇 Additional comments (4)
CHANGELOG.md (2)
20-20
: 🧹 Nitpick (assertive)Fix grammatical error in changelog entry.
Add the missing word "to" before "Add support".
- * Add support for Business Application Header version head.001.001.04 + * Added support for Business Application Header version head.001.001.04Likely invalid or redundant comment.
🧰 Tools
🪛 LanguageTool
[grammar] ~20-~20: Did you mean “to Add”?
Context: ...he datetime to UTC offset if needed * Add support for Business Application Header...(MISSING_TO_BEFORE_A_VERB)
🪛 Markdownlint (0.37.0)
20-20: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
61-65
: Verify migration impact on downstream dependencies.The migration to Java 11, Jakarta EE 10, and the replacement of
XMLGregorianCalendar
withjava.time
classes represent significant breaking changes. These changes may impact downstream dependencies and require coordination with dependent projects.Run the following script to identify potential migration impacts:
🧰 Tools
🪛 Markdownlint (0.37.0)
61-61: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
62-62: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
63-63: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
64-64: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
65-65: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
61-61: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
build.gradle (1)
43-43
: LGTM: Java 11 configuration is consistent.The Java version configuration is properly set across different tasks:
- JavaCompile task language version
- Jar task source and target compatibility
Also applies to: 89-90
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/MxParseUtils.java (1)
25-26
: Ensure completeness of the migration from javax to jakartaThese lines switch from
javax.xml.bind
tojakarta.xml.bind
. Please confirm that all references to the old JAX-B imports have been consistently updated throughout the codebase, and that the project’s build configuration is properly set up for Jakarta dependencies to avoid possible runtime conflicts.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 4
🔭 Outside diff range comments (3)
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/MxParseUtils.java (3)
Line range hint
80-86
: UpdatetargetClass
parameter inparseSAXSource
method to use genericsThe
targetClass
parameter should be declared asClass<?>
instead ofClass
to enhance type safety and eliminate raw type warnings.Apply this diff to update the method signature:
static Object parseSAXSource( - final SAXSource source, final Class targetClass, final Class<?>[] classes, final MxReadParams params) { + final SAXSource source, final Class<?> targetClass, final Class<?>[] classes, final MxReadParams params) {
Line range hint
122-131
: UpdatetargetClass
parameter inparse
method to use genericsSimilarly, the
targetClass
parameter in theparse
method should be updated toClass<?>
to improve type safety.Apply this diff:
static Object parse( - final Class targetClass, + final Class<?> targetClass, final String xml, final Class<?>[] classes, final String localName, final MxReadParams params) {
Line range hint
283-296
: Use try-with-resources to ensureXMLStreamReader
is properly closedIn the
parseComments
method, theXMLStreamReader
should be used within a try-with-resources statement to ensure it is always closed, preventing potential resource leaks.Apply this diff:
public static List<String> parseComments(final String xml) { Objects.requireNonNull(xml, "XML to parse must not be null"); Validate.notBlank(xml, "XML to parse must not be a blank string"); List<String> result = new ArrayList<>(); final XMLInputFactory factory = SafeXmlUtils.inputFactory(); - try { - XMLStreamReader reader = factory.createXMLStreamReader(new StringReader(MxParseUtils.makeXmlLenient(xml))); + try (XMLStreamReader reader = factory.createXMLStreamReader(new StringReader(MxParseUtils.makeXmlLenient(xml)))) { while (reader.hasNext()) { int event = reader.next(); if (event == XMLStreamConstants.COMMENT) { String comment = reader.getText(); if (comment != null) { result.add(comment.trim()); } } } } catch (XMLStreamException e) { log.log(Level.WARNING, "Error parsing XML comments", e); } return result; }
♻️ Duplicate comments (1)
build.gradle (1)
136-136
:⚠️ Potential issueMigration to Jakarta EE is incomplete.
The project still uses
javax.persistence-api
while also having Jakarta dependencies. This mixed usage could lead to compatibility issues.Replace the old javax dependency with its Jakarta equivalent:
-compileOnly 'javax.persistence:javax.persistence-api:2.2'
Also applies to: 155-156
📜 Review details
Configuration used: CodeRabbit UI
Review profile: ASSERTIVE
Plan: Pro
📒 Files selected for processing (3)
CHANGELOG.md
(2 hunks)build.gradle
(9 hunks)iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/MxParseUtils.java
(1 hunks)
🧰 Additional context used
🪛 LanguageTool
CHANGELOG.md
[grammar] ~26-~26: Did you mean “to Add”?
Context: ...he datetime to UTC offset if needed * Add support for Business Application Header...
(MISSING_TO_BEFORE_A_VERB)
🪛 Markdownlint (0.37.0)
CHANGELOG.md
3-3: Expected: h2; Actual: h4
Heading levels should only increment by one level at a time
(MD001, heading-increment)
4-4: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
7-7: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
10-10: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
11-11: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
12-12: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
13-13: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
14-14: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
15-15: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
18-18: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
23-23: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
24-24: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
25-25: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
26-26: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
29-29: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
30-30: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
3-3: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
6-6: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
9-9: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
17-17: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
22-22: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
28-28: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
32-32: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
4-4: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
7-7: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
10-10: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
18-18: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
23-23: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
29-29: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
43-43: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
46-46: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
49-49: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
50-50: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
51-51: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
52-52: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
53-53: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
56-56: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
57-57: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
58-58: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
61-61: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
64-64: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
67-67: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
68-68: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
69-69: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
70-70: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
71-71: Expected: 0; Actual: 2
Unordered list indentation
(MD007, ul-indent)
42-42: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
45-45: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
48-48: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
55-55: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
60-60: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
63-63: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
66-66: Expected: 1; Actual: 0; Below
Headings should be surrounded by blank lines
(MD022, blanks-around-headings)
43-43: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
46-46: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
49-49: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
56-56: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
61-61: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
64-64: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
67-67: null
Lists should be surrounded by blank lines
(MD032, blanks-around-lists)
🔇 Additional comments (4)
CHANGELOG.md (1)
67-71
: Verify the impact of Java 11 migration.The migration to Java 11 and Jakarta EE 10 involves significant changes:
- XMLGregorianCalendar replacement with java.time
- UTC timezone handling in headers
- Potential backward compatibility concerns
Run the following script to verify the migration impact:
✅ Verification successful
Java 11 migration successfully verified
The migration shows comprehensive adoption of java.time API across the codebase, with proper backward compatibility handling through dedicated adapters. The XMLGregorianCalendar references are intentionally maintained only for legacy system support.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Check for remaining Java 8 specific code and XMLGregorianCalendar usage # Test 1: Search for XMLGregorianCalendar usage echo "Checking for XMLGregorianCalendar usage..." rg "XMLGregorianCalendar" # Test 2: Search for java.time imports echo "Checking for java.time adoption..." rg "^import java.time." # Test 3: Check for any remaining javax.xml.datatype imports echo "Checking for old javax.xml.datatype imports..." rg "javax.xml.datatype"Length of output: 65901
🧰 Tools
🪛 Markdownlint (0.37.0)
67-67: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
68-68: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
69-69: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
70-70: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
71-71: Expected: 0; Actual: 2
Unordered list indentation(MD007, ul-indent)
67-67: null
Lists should be surrounded by blank lines(MD032, blanks-around-lists)
build.gradle (1)
150-150
: Verify dependency version compatibility.The upgrade of
pw-swift-core
toSRU2024-10.2.5
needs verification for compatibility with other dependencies.Run the following script to check for potential version conflicts:
Also applies to: 333-333
✅ Verification successful
Dependency version is consistent and properly configured
The
pw-swift-core:SRU2024-10.2.5
version is used consistently across the codebase, and there are no other Prowide dependencies that could cause version conflicts.🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Check for dependency version conflicts # Test 1: Search for pw-swift-core version references echo "Checking pw-swift-core versions..." rg "pw-swift-core:[^'\"]+" # Test 2: Check for any version property definitions echo "Checking version properties..." rg "^[[:space:]]*version[[:space:]]*="Length of output: 444
Script:
#!/bin/bash # Check for other Prowide dependencies echo "Checking other Prowide dependencies..." rg "com\.prowidesoftware:[^'\"]+"Length of output: 263
iso20022-core/src/main/java/com/prowidesoftware/swift/model/mx/MxParseUtils.java (2)
Line range hint
275-279
: Addition ofmakeXmlLenient
method improves XML parsing robustnessThe new
makeXmlLenient
method appropriately handles case sensitivity issues in XML declarations, enhancing the resilience of XML parsing.
27-28
: Verify alljavax.xml.bind
imports are replaced withjakarta.xml.bind
The import statements have been updated to
jakarta.xml.bind
, aligning with the migration to Java 11. Please ensure that there are no remaining references tojavax.xml.bind
in the codebase to prevent potential compatibility issues.Run the following script to identify any remaining
javax.xml.bind
imports:✅ Verification successful
All
javax.xml.bind
imports have been successfully migrated tojakarta.xml.bind
A comprehensive search of the codebase confirms that all XML bind imports are consistently using the jakarta namespace, with no remaining references to javax.xml.bind.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Find all occurrences of 'javax.xml.bind' imports in Java files. rg -t java 'import javax\.xml\.bind' -nLength of output: 41
Script:
#!/bin/bash # Verify both javax and jakarta XML bind usage comprehensively echo "=== Searching for jakarta.xml.bind imports ===" rg -t java 'import jakarta\.xml\.bind' -n echo -e "\n=== Searching for any javax.xml.bind string patterns ===" rg -t java 'javax\.xml\.bind' -n echo -e "\n=== Searching for any jakarta.xml.bind string patterns ===" rg -t java 'jakarta\.xml\.bind' -n echo -e "\n=== Searching for JAXB class references ===" rg -t java '(javax|jakarta)\.xml\.bind\.[A-Z]\w+' -nLength of output: 82543
* test fix attempt using fixed offset * test fix attempt using fixed offset * test fix attempt using fixed offset
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 16
🔭 Outside diff range comments (1)
.github/workflows/gradle.yml (1)
Line range hint
44-48
: Consider environment-aware memory settings.The fixed 7GB max heap size might be excessive and could cause failures in resource-constrained environments. Consider:
- Making memory settings configurable via workflow inputs
- Adding a step to validate the active Java version before build
Example validation step:
- name: Validate Java Version run: | java -version ./gradlew --version🧰 Tools
🪛 yamllint (1.35.1)
[error] 7-7: too many spaces inside brackets
(brackets)
[error] 7-7: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
📜 Review details
Configuration used: CodeRabbit UI
Review profile: ASSERTIVE
Plan: Pro
📒 Files selected for processing (4)
.github/workflows/gradle.yml
(1 hunks)iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java
(1 hunks)iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetDateTimeAdapterTest.java
(1 hunks)iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeAdapterTest.java
(1 hunks)
🧰 Additional context used
🪛 GitHub Actions: Java CI with Gradle
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeAdapterTest.java
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetDateTimeAdapterTest.java
[error] 95-95: Compilation error: cannot find symbol 'offset' in class OffsetDateTimeAdapterTest
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
🪛 yamllint (1.35.1)
.github/workflows/gradle.yml
[error] 9-9: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
🔇 Additional comments (8)
.github/workflows/gradle.yml (3)
Line range hint
22-37
: Well-structured JDK version management!The conditional JDK setup effectively manages both Java 11 for the new branch and Java 8 for legacy code, with proper handling of both push events and pull requests.
🧰 Tools
🪛 yamllint (1.35.1)
[error] 7-7: too many spaces inside brackets
(brackets)
[error] 7-7: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
Line range hint
50-71
: Excellent dependency graph configuration!The dependency submission job is well-configured with:
- Appropriate branch conditions to avoid legacy code alerts
- Clear documentation with reference links
- Smart exclusion of buildSrc from the dependency graph
🧰 Tools
🪛 yamllint (1.35.1)
[error] 7-7: too many spaces inside brackets
(brackets)
[error] 7-7: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
Line range hint
1-71
: Verify workflow with comprehensive test scenarios.Please ensure the workflow is tested with the following scenarios:
- PR from feature branch to SRU2024_v10 (should use Java 11)
- PR from feature branch to main (should use Java 8)
- Direct push to SRU2024_v10 (should use Java 11)
- Direct push to main (should use Java 8)
✅ Verification successful
Workflow configuration is correct but builds are failing
The workflow correctly handles Java versions for both branches:
- Java 11 for SRU2024_v10 branch
- Java 8 for main branch
However, builds are currently failing due to missing Jakarta Persistence dependencies, which is unrelated to the workflow configuration itself.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash # Description: Verify workflow configuration # Check if workflow is properly configured for both Java versions # Check workflow syntax gh workflow view "gradle.yml" # List recent workflow runs to verify Java version usage gh run list --workflow="gradle.yml" --limit 10 --json "headBranch,conclusion,displayTitle"Length of output: 1800
Script:
#!/bin/bash # Get detailed information about recent failed runs gh run list --workflow="gradle.yml" --limit 3 --json databaseId,headBranch,conclusion,displayTitle # For each failed run, get the job logs for run_id in $(gh run list --workflow="gradle.yml" --limit 2 --json databaseId --jq '.[].databaseId'); do echo "=== Run ID: $run_id ===" gh run view $run_id --log doneLength of output: 66442
🧰 Tools
🪛 yamllint (1.35.1)
[error] 7-7: too many spaces inside brackets
(brackets)
[error] 7-7: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
[error] 9-9: too many spaces inside brackets
(brackets)
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetDateTimeAdapterTest.java (3)
1-9
: Well-structured test class with appropriate imports!The class is properly set up using JUnit 5 with minimal and focused imports.
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
11-51
: 🧹 Nitpick (assertive)Consider using parameterized tests for better maintainability.
The test method contains multiple similar test cases that could be more maintainable using JUnit's
@ParameterizedTest
. This would make it easier to add new test cases and improve readability.Example refactor:
@ParameterizedTest @CsvSource({ "2021-09-19T12:13:14,2021-09-19T12:13:14${offset}", "2021-09-19T12:13:14.1,2021-09-19T12:13:14.1${offset}", // ... other test cases }) void testOffsetDateTime(String input, String expected) throws Exception { testDateTimeImpl(input, expected); }Also, consider adding test cases for invalid inputs to verify error handling.
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
1-97
: Verify source of pipeline warnings.The pipeline reports warnings about deprecated API usage and unsafe operations, but they're not visible in this file.
Run the following script to locate the source of warnings:
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[error] 95-95: Compilation error: cannot find symbol 'offset' in class OffsetDateTimeAdapterTest
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeAdapterTest.java (2)
1-23
: LGTM! Well-structured imports and licensing.The file follows best practices with appropriate copyright headers and minimal, specific imports using modern JUnit 5.
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
1-157
: Address compiler warnings as part of Java 11 migration.The pipeline shows warnings about deprecated API usage and unchecked operations. As part of the Java 8 to 11 migration, these should be addressed:
- Review and update any deprecated API usage
- Add appropriate type parameters or suppress warnings with justification where unchecked operations are unavoidable
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
...0022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java
Show resolved
Hide resolved
...0022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java
Show resolved
Hide resolved
return mx; | ||
} | ||
|
||
class TestCustomDateAdapter extends XmlAdapter<String, LocalDate> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🧹 Nitpick (assertive)
Address compiler warnings about deprecated API and unsafe operations.
To resolve the pipeline warnings:
- Add
@SuppressWarnings("deprecation")
if using deprecated JAXB APIs intentionally - Use generic type parameters with XmlAdapter to avoid unchecked operations
- class TestCustomDateAdapter extends XmlAdapter<String, LocalDate> {
+ @SuppressWarnings("deprecation")
+ private static class TestCustomDateAdapter extends XmlAdapter<String, LocalDate> {
- public class TestCustomDateTimeAdapter extends XmlAdapter<String, OffsetDateTime> {
+ @SuppressWarnings("deprecation")
+ private static class TestCustomDateTimeAdapter extends XmlAdapter<String, OffsetDateTime> {
Also applies to: 311-311
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
...0022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java
Outdated
Show resolved
Hide resolved
...0022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java
Outdated
Show resolved
Hide resolved
...22-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeAdapterTest.java
Show resolved
Hide resolved
@Test | ||
public void testUnmarshallFractionOfSeconds() throws Exception { | ||
OffsetTime offsetTime = adapter.unmarshal("12:50:08.123-03:00"); | ||
assertEquals(12, offsetTime.getHour()); | ||
assertEquals(50, offsetTime.getMinute()); | ||
assertEquals(8, offsetTime.getSecond()); | ||
assertEquals(123000000, offsetTime.getNano()); | ||
assertEquals(ZoneOffset.of("-03:00"), offsetTime.getOffset()); | ||
} | ||
|
||
@Test | ||
public void testUnmarshallNoFractionOfSeconds() throws Exception { | ||
OffsetTime offsetTime = adapter.unmarshal("12:50:08-03:00"); | ||
assertEquals(12, offsetTime.getHour()); | ||
assertEquals(50, offsetTime.getMinute()); | ||
assertEquals(8, offsetTime.getSecond()); | ||
assertEquals(ZoneOffset.of("-03:00"), offsetTime.getOffset()); | ||
} | ||
|
||
@Test | ||
public void testUnmarshallNoOffset() throws Exception { | ||
OffsetTime systemDateTime = OffsetTime.parse( | ||
"12:50:08" + ZoneOffset.systemDefault().getRules().getStandardOffset(Instant.now())); | ||
OffsetTime offsetTime = adapter.unmarshal("12:50:08"); | ||
assertEquals(12, offsetTime.getHour()); | ||
assertEquals(50, offsetTime.getMinute()); | ||
assertEquals(8, offsetTime.getSecond()); | ||
assertEquals(systemDateTime.getOffset(), offsetTime.getOffset()); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Add tests for error scenarios and avoid system timezone dependency.
While the happy path tests are well-structured, consider these improvements:
- Add tests for invalid input formats
- Make timezone-dependent tests deterministic
+ @Test
+ public void testUnmarshallInvalidFormat() {
+ assertThrows(Exception.class, () -> adapter.unmarshal("invalid"));
+ assertThrows(Exception.class, () -> adapter.unmarshal("25:00:00+01:00"));
+ assertThrows(Exception.class, () -> adapter.unmarshal("12:60:00+01:00"));
+ }
@Test
public void testUnmarshallNoOffset() throws Exception {
+ // Use fixed zone for deterministic testing
+ ZoneId fixedZone = ZoneId.of("UTC");
OffsetTime systemDateTime = OffsetTime.parse(
- "12:50:08" + ZoneOffset.systemDefault().getRules().getStandardOffset(Instant.now()));
+ "12:50:08" + fixedZone.getRules().getStandardOffset(Instant.now()));
Committable suggestion skipped: line range outside the PR's diff.
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
@Test | ||
public void testMarshallNoFractionOfSecondsAndWithFractionOfSeconds() throws Exception { | ||
ZoneOffset zoneOffSet = ZoneOffset.of("-03:00"); | ||
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS][XXX]"); | ||
|
||
OffsetTime offsetTime = LocalTime.parse("12:50:08", dateTimeFormatter).atOffset(zoneOffSet); | ||
assertEquals("12:50:08-03:00", adapter.marshal(offsetTime)); | ||
|
||
offsetTime = LocalTime.parse("12:50:08.123", dateTimeFormatter).atOffset(zoneOffSet); | ||
assertEquals("12:50:08.123-03:00", adapter.marshal(offsetTime)); | ||
} | ||
|
||
@Test | ||
public void testMarshallNoOffset() throws Exception { | ||
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS][XXX]"); | ||
ZoneOffset offset = ZoneId.systemDefault().getRules().getOffset(Instant.now()); | ||
OffsetTime offsetTime = LocalTime.parse("12:50:08", dateTimeFormatter).atOffset(offset); | ||
assertEquals("12:50:08" + offset, adapter.marshal(offsetTime)); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🧹 Nitpick (assertive)
Improve marshalling tests robustness.
Consider these enhancements:
- Extract DateTimeFormatter pattern as a constant
- Add null handling test
- Make timezone-dependent test deterministic
+ private static final String TIME_PATTERN = "HH:mm:ss[.SSS][XXX]";
+
@Test
public void testMarshallNoFractionOfSecondsAndWithFractionOfSeconds() throws Exception {
ZoneOffset zoneOffSet = ZoneOffset.of("-03:00");
- DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("HH:mm:ss[.SSS][XXX]");
+ DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern(TIME_PATTERN);
// ... rest of the test
}
+ @Test
+ public void testMarshallNull() {
+ assertThrows(IllegalArgumentException.class, () -> adapter.marshal(null));
+ }
Committable suggestion skipped: line range outside the PR's diff.
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
@Test | ||
public void testOffsetTime() throws Exception { | ||
String offset = systemOffset(); | ||
|
||
// DateTime without offset and with/without fractional seconds | ||
testTimeImpl("12:13:14", "12:13:14" + offset); | ||
testTimeImpl("12:13:14.1", "12:13:14.1" + offset); | ||
testTimeImpl("12:13:14.12", "12:13:14.12" + offset); | ||
testTimeImpl("12:13:14.123", "12:13:14.123" + offset); | ||
testTimeImpl("12:13:14.123456789", "12:13:14.123456789" + offset); | ||
|
||
// DateTime with offset and without fractional seconds | ||
testTimeImpl("12:13:14+01:00", "12:13:14+01:00"); | ||
testTimeImpl("12:13:14-01:00", "12:13:14-01:00"); | ||
testTimeImpl("12:13:14+00:00", "12:13:14+00:00"); | ||
testTimeImpl("12:13:14-00:00", "12:13:14+00:00"); | ||
testTimeImpl("12:13:14+08:30", "12:13:14+08:30"); | ||
testTimeImpl("12:13:14Z", "12:13:14+00:00"); | ||
|
||
// DateTime with offset and fractional seconds | ||
testTimeImpl("12:13:14.1+01:00", "12:13:14.1+01:00"); | ||
testTimeImpl("12:13:14.12-01:00", "12:13:14.12-01:00"); | ||
testTimeImpl("12:13:14.123+00:00", "12:13:14.123+00:00"); | ||
testTimeImpl("12:13:14.123+08:30", "12:13:14.123+08:30"); | ||
testTimeImpl("12:13:14.000+08:30", "12:13:14+08:30"); | ||
testTimeImpl("12:13:14.000Z", "12:13:14+00:00"); | ||
testTimeImpl("12:13:14.123Z", "12:13:14.123+00:00"); | ||
testTimeImpl("12:13:14.123456Z", "12:13:14.123456+00:00"); | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🧹 Nitpick (assertive)
Convert to parameterized tests for better maintainability.
The comprehensive test cases could be more maintainable using JUnit 5's @ParameterizedTest:
- @Test
- public void testOffsetTime() throws Exception {
+ @ParameterizedTest(name = "Time {0} should be formatted as {1}")
+ @CsvSource({
+ // Without offset
+ "12:13:14, 12:13:14${systemOffset}",
+ "12:13:14.1, 12:13:14.1${systemOffset}",
+ // With offset, no fractional seconds
+ "12:13:14+01:00, 12:13:14+01:00",
+ "12:13:14Z, 12:13:14+00:00",
+ // With offset and fractional seconds
+ "12:13:14.123Z, 12:13:14.123+00:00"
+ })
+ void testOffsetTimeFormats(String input, String expected) throws Exception {
+ testTimeImpl(input, expected);
+ }
Committable suggestion skipped: line range outside the PR's diff.
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
private void testTimeImpl(String value, String valueResult) throws Exception { | ||
OffsetTimeAdapter offsetTimeAdapter = new OffsetTimeAdapter(); | ||
OffsetTime OffsetTime = offsetTimeAdapter.unmarshal(value); | ||
String valueDateResult = offsetTimeAdapter.marshal(OffsetTime); | ||
assertEquals(valueResult, valueDateResult); | ||
} | ||
|
||
@Test | ||
public void testOffsetTimeWithNanosOnlyZero() throws Exception { | ||
// removable zeros | ||
testTimeImpl("17:30:33.0+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.00+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.000+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.0000+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.00000+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.000000+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.0000000+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.00000000+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.000000000+02:00", "17:30:33+02:00"); | ||
// testTimeImpl("17:30:33.0000000000+02:00", "17:30:33+02:00"); //Cannot parse more | ||
// than 9 characters in nano value | ||
|
||
// non removable zeros | ||
testTimeImpl("17:30:33.0+02:00", "17:30:33+02:00"); | ||
testTimeImpl("17:30:33.01+02:00", "17:30:33.01+02:00"); | ||
testTimeImpl("17:30:33.001+02:00", "17:30:33.001+02:00"); | ||
testTimeImpl("17:30:33.0001+02:00", "17:30:33.0001+02:00"); | ||
testTimeImpl("17:30:33.00001+02:00", "17:30:33.00001+02:00"); | ||
testTimeImpl("17:30:33.000001+02:00", "17:30:33.000001+02:00"); | ||
testTimeImpl("17:30:33.0000001+02:00", "17:30:33.0000001+02:00"); | ||
testTimeImpl("17:30:33.00000001+02:00", "17:30:33.00000001+02:00"); | ||
testTimeImpl("17:30:33.000000001+02:00", "17:30:33.000000001+02:00"); | ||
|
||
// UTC | ||
testTimeImpl("17:30:33.0Z", "17:30:33+00:00"); | ||
testTimeImpl("17:30:33.000000000Z", "17:30:33+00:00"); | ||
testTimeImpl("17:30:33.000000001Z", "17:30:33.000000001+00:00"); | ||
testTimeImpl("17:30:33.123456789Z", "17:30:33.123456789+00:00"); | ||
testTimeImpl("17:30:33Z", "17:30:33+00:00"); | ||
} | ||
|
||
private static String systemOffset() { | ||
ZoneOffset zoneOffset = ZoneId.systemDefault().getRules().getStandardOffset(Instant.now()); | ||
String offset = zoneOffset.toString(); | ||
if (offset.equals("Z")) { | ||
offset = "+00:00"; | ||
} | ||
return offset; | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🧹 Nitpick (assertive)
Document nano precision limitation and improve helper methods.
The nano precision tests are thorough, but consider these improvements:
- Document the 9-character nano value limitation
- Make systemOffset() more robust
+ /**
+ * Maximum precision for nano seconds is 9 characters.
+ * @see java.time.temporal.ChronoField#NANO_OF_SECOND
+ */
@Test
public void testOffsetTimeWithNanosOnlyZero() throws Exception {
// ... existing tests ...
}
private static String systemOffset() {
+ // Cache the offset to avoid repeated calculations
+ return ZoneId.systemDefault()
+ .getRules()
+ .getStandardOffset(Instant.now())
+ .toString()
+ .replace("Z", "+00:00");
}
📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.
private void testTimeImpl(String value, String valueResult) throws Exception { | |
OffsetTimeAdapter offsetTimeAdapter = new OffsetTimeAdapter(); | |
OffsetTime OffsetTime = offsetTimeAdapter.unmarshal(value); | |
String valueDateResult = offsetTimeAdapter.marshal(OffsetTime); | |
assertEquals(valueResult, valueDateResult); | |
} | |
@Test | |
public void testOffsetTimeWithNanosOnlyZero() throws Exception { | |
// removable zeros | |
testTimeImpl("17:30:33.0+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.00+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.0000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.00000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.000000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.0000000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.00000000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.000000000+02:00", "17:30:33+02:00"); | |
// testTimeImpl("17:30:33.0000000000+02:00", "17:30:33+02:00"); //Cannot parse more | |
// than 9 characters in nano value | |
// non removable zeros | |
testTimeImpl("17:30:33.0+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.01+02:00", "17:30:33.01+02:00"); | |
testTimeImpl("17:30:33.001+02:00", "17:30:33.001+02:00"); | |
testTimeImpl("17:30:33.0001+02:00", "17:30:33.0001+02:00"); | |
testTimeImpl("17:30:33.00001+02:00", "17:30:33.00001+02:00"); | |
testTimeImpl("17:30:33.000001+02:00", "17:30:33.000001+02:00"); | |
testTimeImpl("17:30:33.0000001+02:00", "17:30:33.0000001+02:00"); | |
testTimeImpl("17:30:33.00000001+02:00", "17:30:33.00000001+02:00"); | |
testTimeImpl("17:30:33.000000001+02:00", "17:30:33.000000001+02:00"); | |
// UTC | |
testTimeImpl("17:30:33.0Z", "17:30:33+00:00"); | |
testTimeImpl("17:30:33.000000000Z", "17:30:33+00:00"); | |
testTimeImpl("17:30:33.000000001Z", "17:30:33.000000001+00:00"); | |
testTimeImpl("17:30:33.123456789Z", "17:30:33.123456789+00:00"); | |
testTimeImpl("17:30:33Z", "17:30:33+00:00"); | |
} | |
private static String systemOffset() { | |
ZoneOffset zoneOffset = ZoneId.systemDefault().getRules().getStandardOffset(Instant.now()); | |
String offset = zoneOffset.toString(); | |
if (offset.equals("Z")) { | |
offset = "+00:00"; | |
} | |
return offset; | |
} | |
} | |
private void testTimeImpl(String value, String valueResult) throws Exception { | |
OffsetTimeAdapter offsetTimeAdapter = new OffsetTimeAdapter(); | |
OffsetTime OffsetTime = offsetTimeAdapter.unmarshal(value); | |
String valueDateResult = offsetTimeAdapter.marshal(OffsetTime); | |
assertEquals(valueResult, valueDateResult); | |
} | |
/** | |
* Maximum precision for nano seconds is 9 characters. | |
* @see java.time.temporal.ChronoField#NANO_OF_SECOND | |
*/ | |
@Test | |
public void testOffsetTimeWithNanosOnlyZero() throws Exception { | |
// removable zeros | |
testTimeImpl("17:30:33.0+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.00+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.0000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.00000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.000000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.0000000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.00000000+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.000000000+02:00", "17:30:33+02:00"); | |
// testTimeImpl("17:30:33.0000000000+02:00", "17:30:33+02:00"); //Cannot parse more | |
// than 9 characters in nano value | |
// non removable zeros | |
testTimeImpl("17:30:33.0+02:00", "17:30:33+02:00"); | |
testTimeImpl("17:30:33.01+02:00", "17:30:33.01+02:00"); | |
testTimeImpl("17:30:33.001+02:00", "17:30:33.001+02:00"); | |
testTimeImpl("17:30:33.0001+02:00", "17:30:33.0001+02:00"); | |
testTimeImpl("17:30:33.00001+02:00", "17:30:33.00001+02:00"); | |
testTimeImpl("17:30:33.000001+02:00", "17:30:33.000001+02:00"); | |
testTimeImpl("17:30:33.0000001+02:00", "17:30:33.0000001+02:00"); | |
testTimeImpl("17:30:33.00000001+02:00", "17:30:33.00000001+02:00"); | |
testTimeImpl("17:30:33.000000001+02:00", "17:30:33.000000001+02:00"); | |
// UTC | |
testTimeImpl("17:30:33.0Z", "17:30:33+00:00"); | |
testTimeImpl("17:30:33.000000000Z", "17:30:33+00:00"); | |
testTimeImpl("17:30:33.000000001Z", "17:30:33.000000001+00:00"); | |
testTimeImpl("17:30:33.123456789Z", "17:30:33.123456789+00:00"); | |
testTimeImpl("17:30:33Z", "17:30:33+00:00"); | |
} | |
private static String systemOffset() { | |
// Cache the offset to avoid repeated calculations | |
return ZoneId.systemDefault() | |
.getRules() | |
.getStandardOffset(Instant.now()) | |
.toString() | |
.replace("Z", "+00:00"); | |
} | |
} |
🧰 Tools
🪛 GitHub Actions: Java CI with Gradle
[warning] Some input files use or override a deprecated API
[warning] Some input files use unchecked or unsafe operations
…/adapters/MxWriteAdaptersTest.java Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
…/adapters/MxWriteAdaptersTest.java Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 1
📜 Review details
Configuration used: CodeRabbit UI
Review profile: ASSERTIVE
Plan: Pro
📒 Files selected for processing (3)
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java
(1 hunks)iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetDateTimeAdapterTest.java
(1 hunks)iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeAdapterTest.java
(1 hunks)
🔇 Additional comments (17)
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetDateTimeAdapterTest.java (6)
1-8
: LGTM! Well-structured test class.The class follows test best practices with appropriate visibility and minimal imports.
19-24
: Remove unused variable.The
dateTime
variable is declared but never used in the test method.
26-50
: LGTM! Comprehensive test coverage.The test cases are well-organized and cover:
- Date-time without offset (with/without fractional seconds)
- Date-time with offset (with/without fractional seconds)
- Various timezone offsets including UTC
52-57
: Optimize test helper method.Consider these improvements:
- Make
offsetDateTimeAdapter
a class field to avoid creating a new instance for each test.- Add null checks for the input parameters.
71-72
: Document limitation as a proper test case.Instead of commenting out the test case, document the limitation using a proper test case.
61-91
: LGTM! Thorough testing of nanosecond precision.The test cases comprehensively cover:
- Removable trailing zeros
- Non-removable significant zeros
- UTC representation
- Edge cases for nanosecond precision
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/MxWriteAdaptersTest.java (6)
33-33
: Make the field final for immutability.The
systemOffset
field should be declared asfinal
since it's not modified after initialization.- private final ZoneOffset systemOffset = ZoneOffset.ofHours(-3); + private final ZoneOffset systemOffset = ZoneOffset.ofHours(-3);
64-79
: Extract duplicated time assertions into a helper method.The time field assertions are duplicated across multiple test methods. Extract them into a helper method to improve maintainability.
Also applies to: 117-132, 171-186
61-61
: Standardize day field comparisons across tests.The tests use inconsistent ChronoField constants for day comparisons. Consider standardizing to
DAY_OF_MONTH
for consistency.Also applies to: 114-114, 168-168, 215-215
278-283
: Simplify the helper method by removing unnecessary variable.The intermediate variable can be removed by directly returning the result of
setDatesIntoMessage
.private MxPacs00800102 getMxPacs00800102WithDateTimeValuesFilled() { LocalDate localDate = LocalDate.parse("2021-10-19"); OffsetTime offsetTime = OffsetTime.parse("12:13:14" + systemOffset); OffsetDateTime offsetDateTime = OffsetDateTime.parse("2021-10-19T12:13:14" + systemOffset); - return setDatesIntoMessage(localDate, offsetTime, offsetDateTime); + return setDatesIntoMessage(localDate, offsetTime, offsetDateTime); }
298-320
: Standardize access modifiers and improve test adapters.
- The access modifiers are inconsistent between the two adapter classes.
- Using hardcoded values might not be flexible enough for all test scenarios.
- Missing
@SuppressWarnings("deprecation")
annotation for deprecated JAXB APIs.Consider:
- Making both classes either package-private or public.
- Allowing the test values to be configured through constructors.
- Adding the
@SuppressWarnings("deprecation")
annotation.- class TestCustomDateAdapter extends XmlAdapter<String, LocalDate> { + @SuppressWarnings("deprecation") + private static class TestCustomDateAdapter extends XmlAdapter<String, LocalDate> { + private final String marshalValue; + private final LocalDate unmarshalValue; + + TestCustomDateAdapter() { + this("foobar", LocalDate.now()); + } + + TestCustomDateAdapter(String marshalValue, LocalDate unmarshalValue) { + this.marshalValue = marshalValue; + this.unmarshalValue = unmarshalValue; + } @Override public LocalDate unmarshal(String v) { - return LocalDate.now(); + return unmarshalValue; } @Override public String marshal(LocalDate cal) { - return "foobar"; + return marshalValue; } } - public class TestCustomDateTimeAdapter extends XmlAdapter<String, OffsetDateTime> { + @SuppressWarnings("deprecation") + private static class TestCustomDateTimeAdapter extends XmlAdapter<String, OffsetDateTime> { // Similar changes for TestCustomDateTimeAdapter }
35-267
: Well-structured and comprehensive test coverage!The test suite effectively covers:
- Default adapters with/without fractional seconds
- UTC timezone handling
- Custom date/time patterns
- Custom adapters
- Business header date/time handling
iso20022-core/src/test/java/com/prowidesoftware/swift/model/mx/adapters/OffsetTimeAdapterTest.java (5)
24-27
: Consider improving test setup with @beforeeach.The adapter initialization could be improved for better test isolation and immutability.
48-56
: Add tests for error scenarios and avoid system timezone dependency.The test depends on system timezone which makes it non-deterministic.
58-76
: Improve marshalling tests robustness.Consider extracting the DateTimeFormatter pattern as a constant and adding null handling test.
78-107
: Convert to parameterized tests for better maintainability.The comprehensive test cases could be more maintainable using JUnit 5's @ParameterizedTest.
115-147
: Document nano precision limitation and improve helper methods.The nano precision tests are thorough, but consider documenting the 9-character nano value limitation.
private void testTimeImpl(String value, String valueResult) throws Exception { | ||
OffsetTimeAdapter offsetTimeAdapter = new OffsetTimeAdapter(); | ||
OffsetTime OffsetTime = offsetTimeAdapter.unmarshal(value); | ||
String valueDateResult = offsetTimeAdapter.marshal(OffsetTime); | ||
assertEquals(valueResult, valueDateResult); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🧹 Nitpick (assertive)
Improve helper method efficiency.
The testTimeImpl
method creates a new OffsetTimeAdapter
instance for each test case. Consider using the existing adapter field instead:
private void testTimeImpl(String value, String valueResult) throws Exception {
- OffsetTimeAdapter offsetTimeAdapter = new OffsetTimeAdapter();
- OffsetTime OffsetTime = offsetTimeAdapter.unmarshal(value);
- String valueDateResult = offsetTimeAdapter.marshal(OffsetTime);
+ OffsetTime OffsetTime = adapter.unmarshal(value);
+ String valueDateResult = adapter.marshal(OffsetTime);
assertEquals(valueResult, valueDateResult);
}
📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.
private void testTimeImpl(String value, String valueResult) throws Exception { | |
OffsetTimeAdapter offsetTimeAdapter = new OffsetTimeAdapter(); | |
OffsetTime OffsetTime = offsetTimeAdapter.unmarshal(value); | |
String valueDateResult = offsetTimeAdapter.marshal(OffsetTime); | |
assertEquals(valueResult, valueDateResult); | |
} | |
private void testTimeImpl(String value, String valueResult) throws Exception { | |
OffsetTime OffsetTime = adapter.unmarshal(value); | |
String valueDateResult = adapter.marshal(OffsetTime); | |
assertEquals(valueResult, valueDateResult); | |
} |
Summary by CodeRabbit
Release Notes for Version 10.2.5
New Features
LocalDate
,OffsetDateTime
,OffsetTime
,Year
, andYearMonth
.Major Changes
javax.xml.bind
tojakarta.xml.bind
namespace.XMLGregorianCalendar
withOffsetDateTime
for improved date-time handling.Dependency Updates
javax.persistence
andjavax.validation
with Jakarta EE counterparts.Enhancements
MxSwiftMessage
andAppHdrFactory
with better date-time handling.Bug Fixes