Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

implement requireSourceFieldsUsedExcept #608

Conversation

nimatrueway
Copy link

@nimatrueway nimatrueway commented Sep 29, 2024

An implementation of requireSourceFieldsUsedExcept(_.field1, _.field2..) as described in #248

TODO/Future Work:

  • move usedSourceFields out from DerivationResult state and move it to ProductToProduct class
  • implement a vararg version of def ignoreUnusedField(selector, selectors*) as Mateus suggested
  • add FailOnSpecifiedUnused subclass for ActionOnUnused
    • implement a new setting to DSL def requireUsedField(selector, selectors*) as Mateus suggested
    • add RequireUsedField to TransformerOverrides to add to list of fields to be checked if FailOnSpecifiedUnused is enabled

@nimatrueway nimatrueway force-pushed the 248_sc3_RequireSourceFieldsExceptType branch 3 times, most recently from 151adba to 99f2e7b Compare September 29, 2024 23:17
to accommodate settings that are validation in nature, the first one being RequireAllSourceFieldsUsed(except)
to be able to track source field usage across the processing workflow without polluting the derivation code
@nimatrueway nimatrueway force-pushed the 248_sc3_RequireSourceFieldsExceptType branch 2 times, most recently from 8c71303 to 25eabf2 Compare September 29, 2024 23:24
@nimatrueway nimatrueway force-pushed the 248_sc3_RequireSourceFieldsExceptType branch from 25eabf2 to 13ebbaf Compare September 29, 2024 23:27
@nimatrueway nimatrueway marked this pull request as ready for review September 29, 2024 23:28
Copy link
Member

@MateuszKubuszok MateuszKubuszok left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hello, thank you for your PR!

I think it's really nice first iteration, showing that you really looked around the codebase 😄

In general, for the sake of backward compatibility and limiting the amount of new types, and flexibility I would like to suggest several things:

  • separation of the general policy (IgnoreUnusedFields/FailOnUnusedFields/FailOnUnusedFieldStrict/etc) from its exceptions (IgnoreField[Path, Cfg], FailOnUnusedField[Path, Cfg]) both in DSL, type-level config and config within macros
  • that would allow changing the policy globally or in the whole scope
  • allowed/rejected fields could be appended one by one, matching current conventions for other customizations
    • if needed there could be 3 possible overrides: ignore field, reject unused field, remove override for field

So that would mean:

  • storing field policy as a TransformerFlag similar to ImplicitTransformerPreference (allowing us to add more policiies in the future)
    • e.g. .unusedFieldPolicy(FailOnUnused)
  • adding exceptions to list... maybe requiring non-empty list of overrides
    • e.g. def ignoreUnusedField(selector, selectors*)
  • things would be stored in a slightly different way internally, mostly using runtimeOverrides: Vector[(Path, TransformerOverride)] which is already managing propagation of configs based on paths

Such DSL would require a but more keystrokes but it would better fit Chimney's existing conventions.

A.asCtor[runtime.PathList.List[?, ?]]
.map(A0 => A0.param_<[runtime.Path](0) -> A0.param_<[runtime.PathList](1))
}
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A list of paths would not be needed if instead this would apply a config, as many times as many paths there is in vararg. This would make less code to maintain in DSL and macros.

case _: Success[?] =>
updateState(_.appendToUsedSourceFields(field)).logSuccess(_ => s"source field $field usage registered")
case _: Failure => this
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wouldn't like to store the result of the fields check in DerivationResult - this should be handled by ProductToProduct rule only.

final case class RequireAllSourceFieldsUsedExcept(sourceFields: Set[String]) extends Verification {
override def toString: String = s"RequireAllSourceFieldsUsedExcept(sourceFields=${sourceFields.mkString(", ")})"
}
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wouldn't introduce a new type parameter the DSL - while users should not rely on it, and MiMa should not complain about it (type erasure) it would break source compatibility for all people who could accidentally save the inferred type somewhere.

Instead:

  • the field policy should be a separate flag, working similar to ImplicitTransformerPreference
  • this would allow to change the policy globally (via -Xmacro-settings) or for all derivations in scope with implicit TransformerConfiguration
  • then explicit allowance/rejection of a field would be a normal TransformerOverride following existing overrides conventions
  • it would work similarly to withFieldRenamed in that there would not be any runtime data related to it
  • then only ProductToProductRule would be modified to check what is the current policy and trace which fields were used, generating error if needed

object PathList {
final class Empty extends PathList
final class List[Head <: Path, Tail <: PathList] extends PathList
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not needed when the Path is applied to Overrides for every selector in vararg

private val preventImplicitSummoningForTypes: Option[(??, ??)] = None
private val preventImplicitSummoningForTypes: Option[(??, ??)] = None,
/** Stores all verification settings provided by user */
verifications: Vector[Verification] = Vector.empty
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Instead of having a separate value, it should be stored in runtimeOverrides under the Path key, but with a new subtype e.g. sealed trait ForFieldPolicy extends TransformerOverride

@@ -76,6 +76,11 @@ final case class NotSupportedTransformerDerivation(
)(val fromType: String, val toType: String)
extends TransformerDerivationError

final case class UnusedButRequiredToUseSourceFields(
unused: Set[String]
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

unused: Set[String] -> unusedFields: ListSet[String]

@@ -108,6 +113,8 @@ object TransformerDerivationError {
| Please eliminate total/partial ambiguity from implicit scope or use ${MAGENTA}enableImplicitConflictResolution$RESET/${MAGENTA}withFieldComputed$RESET/${MAGENTA}withFieldComputedPartial$RESET to decide which one should be used.""".stripMargin
case NotSupportedTransformerDerivation(exprPrettyPrint) =>
s" derivation from $exprPrettyPrint: $fromType to $toType is not supported in Chimney!"
case UnusedButRequiredToUseSourceFields(unusedFields) =>
s" field(s) $MAGENTA${unusedFields.mkString(", ")}$RESET of $MAGENTA${fromType}$RESET are required to be used in the transformation but are not used!"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe are required to be used in the transformation but are not used -> were required to be used in the transformation but are not used?

* @param selectorFrom
* exception fields that are not required to be used in the transformation
* @return
*/
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

* @return
*   [[io.scalaland.chimney.dsl.TransformerInto]]
*
* @since 1.5.0

// expected output:
// Target(a = "value")
```

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Snippet cannot at the same time check successful output and fail compilation, it would have to be 2 snippets:

  • one with // expected error:
  • one with expected output:

without it, snipped checking will fail.

@nimatrueway
Copy link
Author

Hello, thank you for your PR!

I think it's really nice first iteration, showing that you really looked around the codebase 😄
...

My pleasure! Thank you so much for the quick review and good ideas. I initially had a similar thought in #601, it's nice to hear it wasn't too far off.

I'll start developing the API change 🤓🚧

@nimatrueway
Copy link
Author

Hey @MateuszKubuszok, I addressed the architectural change you proposed (and almost all your comments). Namely I split the API into a flag and a setting to override configuration per field.

// implementation:done -> tracks all fields and forces compile error if any left unused
.enableUnusedFieldPolicy(FailOnUnused)

// implementation:done -> untracks a single field
// TODO: how do I implement the vararg version without introducing PathList type? could use your direction here
.withIgnoreUnusedField(_.b)

To avoid a API change in the future and preserve backward compatibility, it's probably best to fix the API in this PR. So I tried to formulate your vision. Here's the domain terms and phrases I imagined makes sense for this feature. Please share if you have any suggestion.

  • I would like to track certain fields and fail transformation derivation if any are unused.
  • I would like to track certain fields and spit out warning during transformation derivation if any are unused.
  • I would like to track all fields of the source type by default.
  • I would like to track none of fields of the source type by default.
  • I would like to ignore certain fields if unused.
  • I would like to track certain fields if unused.

In that case I imagined the flag to be parameterized on two aspects fail/warn on unused and whether to track all fields or none of the fields by default.

// TODO: does this make sense? isn't it too verbose?
.enableUnusedFieldPolicy(FailOnUnused, TrackAll)
.enableUnusedFieldPolicy(WarnOnUnused, TrackNone)

// TODO: to manually track a bunch of fields
.withTrackUnusedFields(_.b, _.c)

@MateuszKubuszok
Copy link
Member

MateuszKubuszok commented Oct 2, 2024

Hmm, the way I see it we have 2 possible policies for now:

  • ignore any unused field
  • fail the compilation on any field which was not explicitly marked as unused

and both would additionally fail the compilation if someone marked a field as unused but it was required.

In some more-or-less distant future, after #115, we might have additional policies:

  • ignore unused fields in fallback, failing on unused fields in the main value
  • fail the compilation for for any field, either in main value or fallback, what wasn't explicitly marked as unused
    • this could be problematic as I cannot think of a good DSL to mark such field as explicitly unused

Additionally, we have to keep in mind that:

  • even though we say field, actually right now Chimney makes no distinction between vals defined like:

    case class Foo(a: Int, b: String)

    and:

    case class Foo() {
      val a: Int = 0
      val b: String = "text"
    }

    and, some people would probably prefer to make the distinction which they want to ignore, but that distinction would have to be implemented somewhere in the future, so for now, we would have to keep the distinction in mind when naming things, e.g. by using name like "unused source values"

  • these flags are global, so whatever name we use, should not collide with whatever users' might use in their codebases, verbose names are less of an issue

  • we would have to add similar thing for target subtypes (when it's an enum/a subtype), so we have to make sure that the name indicates which is which

  • ATM Chimney does not support showing warnings - it's either info for logs, or error for reasons why macro cannot be expanded, we might add it in the future, as something like policyReportingStrategy(PolicyCheckFailureIsWarning)/policyReportingStrategy(PolicyCheckFailureIsError) but that would increase the scope of this PR too much.


Currently I think the API what would work the best would be something like:

// sealed abstract class UnusedFieldPolicy
// object FailOnIgnoredSourceVal extends UnusedFieldPolicy

src.into[Target]
.enableUnusedFieldPolicy(FailOnIgnoredSourceVal) // 1 value for now
.disableUnusedFieldPolicy // ignores unused fields, the default
.withFieldUnused(_.foo.bar.baz) // expect that field is unused:
                                // - fail compilation if it is needed
                                // - do NOT treat field as ignored when UnusedFieldPolicy is enabled
                                // - be na noop when UnusedFieldPolicy is disabled and field is not used
.transform

Then we could test it similarly to other flags:

  • in both TotalTransformerProductSpec and PartialTransformerProductSpec:
  • group("flag .enableUnusedFieldPolicy")
    • test("should be disabled by default")
    • test("should inform user which unused fields failed")
    • test("should ignore explicitly unused fields")
    • test("should not allow marking required fields as unused")
  • group("flag .disableUnusedFieldPolicy")
  • test("should disable globally enabled .enableUnusedFieldPolicy")
  • tests would have to check that nesting case classes would also work

Adding:

  • distinction between warnings and errors
  • distinction between case class fields and other vals
  • support for UnusedTargetSubtypePolicy

would be a job for another PR, and not necessarily something to do soon. I think that even this change would require some refactoring to how TransformationContext works with regard to Paths in source, so I did a little refactoring in #612 and #613 (sorry for the conflicts 😅 )

@nimatrueway
Copy link
Author

Hmm, the way I see it we have 2 possible policies for now:

...

Thanks for sharing your insights Mateus, I skimmed through your comment twice and we are aligned. I need to carve up sometime on the weekend to hack it. 🚀

@MateuszKubuszok
Copy link
Member

Hey, once again thank you for your effort, I am sorry if you invested some more time to this, since I needed this functionality for another WIP PR (#614) and I had to started implementing it there. I am not sure how long it will take to finish that other PR, but it would be hard now to switch to another implementation.

@nimatrueway
Copy link
Author

nimatrueway commented Jan 17, 2025

Hey, once again thank you for your effort, I am sorry if you invested some more time to this, since I needed this functionality for another WIP PR (#614) and I had to started implementing it there. I am not sure how long it will take to finish that other PR, but it would be hard now to switch to another implementation.

Sorry man, I switched job around the same time as I was working on this PR. Now, I'm working in Ruby/Sorbet shop and slogging to catch up to this new world. (Having to deal with Ruby, I can't put into words how much I missed Scala)

Best of luck to you!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants