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

[API Proposal] Add cultureName constructors to GeneratedRegex #59492

Closed
stephentoub opened this issue Sep 22, 2021 · 28 comments · Fixed by #73490
Closed

[API Proposal] Add cultureName constructors to GeneratedRegex #59492

stephentoub opened this issue Sep 22, 2021 · 28 comments · Fixed by #73490
Labels
api-approved API was approved in API review, it can be implemented area-System.Text.RegularExpressions source-generator Indicates an issue with a source generator feature
Milestone

Comments

@stephentoub
Copy link
Member

stephentoub commented Sep 22, 2021

This issue has now been turned into an API proposal for the missing GeneratedRegex constructors that take in a cultureName. The previous description of the issue has been moved bellow

Background and motivation

Regex engines are able to do case insensitive searches when you use RegexOptions.IgnoreCase or when you use a pattern that signals that it is case-insensitive, like: (?i)abc. For better performance, our engines translate the passed in patterns into their case-sensitive equivalence so that we can benefit of some optimizations like using vectorized search, for example, a pattern (?i)abc will be translated into the equivalent [Aa][Bb][Cc].

Case conversions are not standard across all cultures, which means that when making that translation, it matters a lot which culture is used in order to produce the translation. For most of our engines (interpreted, Compiled, nonBacktracking) this translation happens at runtime, and will use the CultureInfo.CurrentCulture at runtime in order to perform this translation. The remaining engine, however, is different. Because it is the source generated engine, it means that this translation will happen at build-time of the application, which means that the culture won't necessarily be the same as the one you have at runtime. Today, the source generated engine will use the Culture at build-time to perform the translation, and there is no way to select a different culture, which is problematic, since even if you are able to change the culture of your machine if you are building the application, this won't be possible if the code gets built in a build server. This proposal is to add an extra string parameter to the GeneratedRegex attribute so that you can optionally select which culture should be used to perform the translation.

API Proposal

[AttributeUsage(AttributeTargets.Method, AllowMultiple = false, Inherited = false)]
public sealed class GeneratedRegexAttribute : System.Attribute
{
    public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex)] string pattern) { }
    public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex, "options")] string pattern, RegexOptions options) { }
+   public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex, "options")] string pattern, RegexOptions options, string cultureName) { }
    public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex, "options")] string pattern, RegexOptions options, int matchTimeoutMilliseconds) { }
+   public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex, "options")] string pattern, RegexOptions options, int matchTimeoutMilliseconds, string cultureName) { }

+   public string CultureName { get; }
}

API Usage

public partial class Foo
{
    [GeneratedRegex(@"(?i)abc", RegexOptions.None, "en-US")]
    public static partial Regex ThreeParameterRegex();

    [GeneratedRegex(@"abc", RegexOptions.IgnoreCase, /*matchTimeoutInMilliseconds*/ 5000, "en-US")]
    public static partial Regex FourParameterRegex();
}

Draft PR with the implementation

#73490

Original Issue Description

If a developer specifies RegexOptions.IgnoreCase to a regular expression (or uses the inline ignore casing option in the expression), the implementation(s) then need to do casing operations to determine whether things match. That's implemented today in two parts:

  1. At construction time, values are lowercased in the pattern.
  2. At match time, the text being compared is lowercased.

If the developer specifies RegexOptions.CultureInvariant, that's fine, as both (1) and (2) will be done with CultureInfo.InvariantCulture. But if CultureInvariant isn't specified, it's possible that the CultureInfo.CurrentCulture at the time of (1) will differ from the time of (2), in which case the lowercasing can be performed with different cultures, leading to problems. Note that this really only affects the instance methods on Regex. The static methods on Regex access a cache of Regex instances, and the culture used to create the Regex instances is included in the key to that cache. Also note that the docs talk about the current culture being used, but they don't specify current "when".

We have a few options to fix this:

  1. Keep the current discrepancy. Using a different culture to match from the culture when the Regex was created is buggy.
  2. Use the culture that was present at the time of construction. This would mean the Regex would cache that CultureInfo instance (or TextInfo) instance, and use it for match operations rather than having those match operations each consult CultureInfo.CurrentCulture. (This could even have a small positive impact on perf, as it means we'd be accessing a field rather than a thread static on each match operation.)
  3. Use the culture that's present at the time of match. This means we couldn't do at construction time many of the optimizations that are currently done, and we'd need to come up with some caching scheme where we lazily compute the necessary data and find the right set of data to use at match time. In the extreme this could include maintaining a cache per culture of all the regex's state and then finding the right state to use as part of the match, essentially a limited version of what we do in the static methods. There is very likely to be a negative perf impact from this.

We should investigate whether the discrepancy actually matters, e.g. in practice, do all regexes where this could matter:

  • not use IgnoreCase
  • use InvariantCulture
  • only ever access it from the same culture in which it was constructed or where casing rules are the same as the culture in which it was constructed
  • have a pattern that's not susceptible to issues (e.g. is)

My gut is we should do (2), primarily as it's explainable and simple. We're already doing some such computation at construction time, so it's unlikely a robust dependency could have been taken on the use of the culture only at match time.

If we do (2), there are then additional complications for the regex source generator. All the computation that normally happens in the Regex constructor happens at build time with the regex source generator, which means CultureInfo.CurrentCulture is determined at the time of build, whatever the culture is for the process doing the build / generating the source code. We have a few options there:

  1. Keep that design, using whatever the current culture is at build. That's probably not ideal.
  2. Always use invariant culture in the source generator, regardless of whether RegexOptions.CultureInvariant is specified.
  3. Fail to generate code if casing might be required (either RegexOptions.IgnoreCase or the inline i option is specified somewhere in the expression) and RegexOptions.CultureInvariant wasn't specified
  4. Add overloads to the RegexGenerator attribute constructor that accept a string cultureName, which is then used. Either get rid of the other overloads, or have them do one of the other options.

I'm tempted to say we should do (4), and if you use a ctor that doesn't take a culture, fall back to (3).

cc: @GrabYourPitchforks, @tarekgh

Related:
#36147
#58958

@ghost
Copy link

ghost commented Sep 22, 2021

Tagging subscribers to this area: @eerhardt, @dotnet/area-system-text-regularexpressions
See info in area-owners.md if you want to be subscribed.

Issue Details

If a developer specifies RegexOptions.IgnoreCase to a regular expression, the implementation(s) then need to do casing operations to determine whether things match. That's implemented today in two parts:

  1. At construction time, values are lowercased in the pattern.
  2. At match time, the text being compared is lowercased.

If the developer specifies RegexOptions.CultureInvariant, that's fine, as both (1) and (2) will be done with CultureInfo.InvariantCulture. But if CultureInvariant isn't specified, it's possible that the CultureInfo.CurrentCulture at the time of (1) will differ from the time of (2), in which case the lowercasing can be performed with different cultures, leading to problems. Note that this really only affects the instance methods on Regex. The static methods on Regex access a cache of Regex instances, and the culture used to create the Regex instances is included in the key to that cache. Also note that the docs talk about the current culture being used, but they don't specify current "when".

We have a few options to fix this:

  1. Keep the current discrepancy. Using a different culture to match from the culture when the Regex was created is buggy.
  2. Use the culture that was present at the time of construction. This would mean the Regex would cache that CultureInfo instance (or TextInfo) instance, and use it for match operations rather than having those match operations each consult CultureInfo.CurrentCulture. (This could even have a small positive impact on perf, as it means we'd be accessing a field rather than a thread static on each match operation.)
  3. Use the culture that's present at the time of match. This means we couldn't do at construction time many of the optimizations that are currently done, and we'd need to come up with some caching scheme where we lazily compute the necessary data and find the right set of data to use at match time. In the extreme this could include maintaining a cache per culture of all the regex's state and then finding the right state to use as part of the match, essentially a limited version of what we do in the static methods. There is very likely to be a negative perf impact from this.

We should investigate whether the discrepancy actually matters, e.g. in practice, do all regexes where this could matter:

  • not use IgnoreCase
  • use InvariantCulture
  • only ever access it from the same culture in which it was constructed or where casing rules are the same as the culture in which it was constructed
  • have a pattern that's not susceptible to issues (e.g. is)

My gut is we should do (2).

If we do (2), there are then additional complications for the regex source generator. All the computation that normally happens in the Regex constructor happens at build time with the regex source generator, which means CultureInfo.CurrentCulture is determined at the time of build, whatever the culture is for the process doing the build / generating the source code. We have a few options there:

  1. Keep that design, using whatever the current culture is at build. That's probably not ideal.
  2. Always use invariant culture in the source generator, regardless of whether RegexOptions.CultureInvariant is specified.
  3. Fail to generate code if casing might be required (either RegexOptions.IgnoreCase or the inline i option is specified somewhere in the expression) and RegexOptions.CultureInvariant wasn't specified
  4. Add overloads to the RegexGenerator attribute constructor that accept a string cultureName, which is then used. Either get rid of the other overloads, or have them do one of the other options.

I'm tempted to say we should do (4), and if you use a ctor that doesn't take a culture, fall back to (3).

Author: stephentoub
Assignees: -
Labels:

area-System.Text.RegularExpressions

Milestone: 7.0.0

@dotnet-issue-labeler dotnet-issue-labeler bot added the untriaged New issue has not been triaged by the area owner label Sep 22, 2021
@stephentoub stephentoub removed the untriaged New issue has not been triaged by the area owner label Sep 22, 2021
@danmoseley
Copy link
Member

cc @veanes @olsaarik as this is of course relevant to all engines.

@joperezr
Copy link
Member

With all of the IgnoreCase work that happened during this release, some of the behavior here changed, so some of the statements in the original description of the issue are no longer accurate. Here is the current state of things:

  • We no longer lowercase the pattern nor the input at any time. Instead, we use a new Regex casing table to make some transformations in the pattern such that for any pattern that has case equivalences, we transform it to instead become a set of all possible values that the pattern could be considered equal given the selected culture. For example, we would transform a case-insensitive pattern of: abc to a case-sensitive [Aa][Bb][Cc].
  • The culture being used in order to compute these case equivalences (when not using the source generator) will be either the current culture at runtime, or the invariant culture if RegexOptions.CultureInvariant was passed.
  • For the source generator case, the culture being used will be either the current culture at build-time, or invariant culture if RegexOptions.CultureInvariant was passed.

I think the only action item here we may want to consider (and for which it is probably already late in case of 7.0) would be to add an option to the RegexGenerator attribute so it could optionally also take a particular culture to be used for computing the case equivalences.

I'm punting this to Future for now, as having this extra option would be considered new API, and we are currently not taking any more new-APIs for 7.0.

@joperezr joperezr modified the milestones: 7.0.0, Future Jul 18, 2022
@stephentoub
Copy link
Member Author

stephentoub commented Jul 18, 2022

I think the only action item here we may want to consider (and for which it is probably already late in case of 7.0) would be to add an option to the RegexGenerator attribute so it could optionally also take a particular culture to be used for computing the case equivalences.

I think we need to decide now, in .NET 7, whether the source generator defaults to invariant. Additional API could be used to override that.

Having the generated code be based on whatever ambient culture the roslyn process is running as is unfortunate.

@stephentoub
Copy link
Member Author

(And if we need to get in an additional ctor to the attribute in 7 in order to ensure a good story, we can do that.)

@joperezr joperezr modified the milestones: Future, 7.0.0 Jul 18, 2022
@joperezr
Copy link
Member

TBH, the case mappings we use are basically the same across all cultures except for a handful of characters (all related to i and turkish i's). For that reason, I think it is fine if we want to default to current culture, and if someone really want it to be invariant they can control it with the surface we have today as they can pass in RegexOptions.CultureInvariant.

@stephentoub
Copy link
Member Author

all related to i and turkish i's

i is a fairly important letter. :-)

My main concern, though, is determinism, reproducibility, and expectations. If you compile the exact same C# project on two different machines and they produce binaries with different semantics without having opted in to that, that seems concerning.

@joperezr
Copy link
Member

that's a fair point. I'm fine with defaulting to InvariantCulture on the source generator. If we are going to do so, it probably does make sense to also introduce an additional constructor that lets you select the culture at build time.

@stephentoub
Copy link
Member Author

To be clear, I'm not dictating we must change it... I just want us to have the conversation in .NET 7 and explicitly decide what to do rather than punting, at which point the decision is made for us.

@jzabroski
Copy link
Contributor

Document it on docs.microsoft.com for Regular Expressions RegexOptions flags as an FYI for the IgnoreCase and CultureInvariant flags. Done.

In general, I think that table needs some sprucing up. Sometimes its better to focus efforts on documentation than writing more code. If people can read the docs, they can suggest enhancements. At present, there is the defensible argument that it has always worked this way (at worst).

It doesn't sound like there are any explicit user stories for this. That said, it would be nice if there was a nuget package that contained StringTestData and contained some example double wide characters that would cause a string reader that is using per-byte logic to fail on a double wide character across one encoding but succeed on another. I have had use for such test data construction in the past, and it surprises me there isn't really a great library for that anywhere, especially as an English-only speaker. I know ParseCombiningCharacters, etc. exists but the API is also very low-level and most engineers I ask about this are unaware how to use that API, even. - My own need for this was unrelated to regular expressions but rather handling Oracle long SQL text concatenations that would otherwise fail at 4,000 chars - Hope I'm making sense.

@stephentoub
Copy link
Member Author

At present, there is the defensible argument that it has always worked this way

That the culture employed by the regex depends on the culture the Roslyn compiler server process was running with when the user's assembly was compiled in whatever build lab or cloud build environment was used?

@jzabroski
Copy link
Contributor

jzabroski commented Aug 1, 2022

Oof. I did not consider that. Defaulting to InvariantCulture makes sense, but there should be a way to procedurally embed a different culture. I had discussed in a separate issue a way to pass serializable "Cargo" via C# 10 generic attributes, but I don't think generic attributes are planned as part of .NET 7? I heard they were a preview and pushed to C# 11 and whatever .NET major version that coincides with. Either way, you could pass a culture Func via some higher order means, and call it once per generator. The .NET Framework Design Guidelines may not have strong opinions on this, but it might be better than coupling the generator to the user's build machine.

EDIT: this was my hacky idea: #4525 (comment)

@stephentoub
Copy link
Member Author

but there should be a way to procedurally embed a different culture

As opposed to declaratively that the additional overload accepting a culture would support? Why? What's the scenario for that?

@jzabroski
Copy link
Contributor

I was thinking it would be more discoverable to the API end-user. If the RegexGeneratorAttribute has a parameter to control that, they would be more aware of the potential to embed the build machine's culture info at time of generation. But, when you ask the question the way you did, I do feel a bit silly proposing that.

So, to sum it up, I saw this issue as two issues:

  1. Policy - what is the right default, what optimizations are safe to use under that default, whether the default is consistent with existing non-generator API usage
  2. Mechanism - which also entails how end-users discover how the API works and whether the surface area of the API should change. The existing Regex class constructor does not accept a CultureInfo. The only knob is on RegexOptions, which seems less discoverable.

@jzabroski
Copy link
Contributor

@stephentoub I re-read this whole thread and I can see why you didn't follow my logic. My poignant feedback is:

  1. Add overloads to the RegexGenerator attribute constructor that accept a string cultureName, which is then used. Either get rid of the other overloads, or have them do one of the other options.

I'm tempted to say we should do (4), and if you use a ctor that doesn't take a culture, fall back to (3).

I don't like passing in a string for the culture name. You can customize cultures. Just my opinion, but if this is the API for the next 10 years, it would be ideal if the Regex constructor took in an actual CultureInfo, not some string cultureName.

I suppose the problem with passing in a CultureInfo is that someone can pass in a reference to the current thread's CultureInfo (at time of construction) and the Regex could get throw an InvalidOperationException if run on other threads using the spawning thread's CurrentCulture CultureInfo value. Not exactly a probably scenario, but possible, and would violate the documented .NET Standard thread safety attributes of the Regex class. The only way I could see to guarantee thread safety in this "doesn't throw if access across threads" sense is if a ThreadLocal<CultureInfo> was used.

Hopefully this is a bit more constructive, albeit you're much smarter than me, so probably already thought of all this.

@stephentoub
Copy link
Member Author

I suppose the problem with passing in a CultureInfo is that someone can pass in a reference to the current thread's CultureInfo

We're talking about the source generator here. The problem with passing in a CultureInfo is this is all at build time; there's no user code being executed.

@tarekgh
Copy link
Member

tarekgh commented Aug 3, 2022

In addition to @stephentoub reply:

I suppose the problem with passing in a CultureInfo is that someone can pass in a reference to the current thread's CultureInfo (at time of construction) and the Regex could get throw an InvalidOperationException

The documentation can be a little bit misleading but what you mentioned here is not true. The CurrentCulture is not tied to specific thread and can be used in any other threads.

@jzabroski
Copy link
Contributor

We're talking about the source generator here. The problem with passing in a CultureInfo is this is all at build time; there's no user code being executed.

I see. (4) with fallback to (3) is the ideal, in my opinion. If in the future .NET makes Attribute metadata more robust, the RegexGenerator can be extended to add a CultureInfo parameter.

Just wondering, is the full proposal for (4) to:

  1. Add string cultureName as a ctor parameter to Regex, RegexGeneratorAttribute
    or
  2. Add CultureInfo cultureInfo to Regex ctor, add string cultureName to RegexGeneratorAttribute which is effectively equivalent to new CultureInfo(cultureName) but an implementation detail (it may be acceptable to peer through the BCP-47 language tag to see if it's Turkish or Azerbayan and not construct any culture in some cases)?

Assuming documentation explicitly calls out cultureName must be a BCP-47 tag.

and, if not already considered:

  1. If you're compiling on a system where PlatformDetection suggests it's using Globalization Invariant settings, do you throw a CultureNotFoundException? How does someone write unit tests on their regex and run them on a docker instance with Globalization Invariant enabled and specifies a non-ICU tag? (Probably a minor concern - just thinking this through). https://docs.microsoft.com/en-us/dotnet/core/compatibility/globalization/6.0/culture-creation-invariant-mode In general, you couldn't do those tests today on such a docker instance, so I dont think it's a big deal unless it would cause the generator to eagerly load the ICU package via BCP-47 tag and crash the generator with the FailFast.

@joperezr
Copy link
Member

joperezr commented Aug 3, 2022

Ok so after thinking a bit more about this I agree with @stephentoub that we should change to default to InvariantCulture. I don't think that it is a good experience to have the exact same source code produce two different behaviors depending on where it gets built, especially when a lot of times you don't control where is your code getting built (for example when you use build servers that may be geo-distributed). I'll push a PR that makes this change because we do want this to be part of .NET 7.

As for the GeneratedRegex constructor that allows to pass in a culture, do we still think there is time to consider that API, or would you be ok with pushing this consideration for vNext?

@joperezr
Copy link
Member

joperezr commented Aug 3, 2022

Just wondering, is the full proposal for (4) to:
Add string cultureName as a ctor parameter to Regex, RegexGeneratorAttribute
or
Add CultureInfo cultureInfo to Regex ctor, add string cultureName to RegexGeneratorAttribute which is effectively equivalent to new CultureInfo(cultureName) but an implementation detail (it may be acceptable to peer through the BCP-47 language tag to see if it's Turkish or Azerbayan and not construct any culture in some cases)?

I guess we are not sure yet as we haven't designed that API just yet, but the high-level idea would be that the source generated regex defaults to use InvariantCulture mappings, and we would also provide some version of the attribute in the future where you can select a different one if you wanted to.

How does someone write unit tests on their regex and run them on a docker instance with Globalization Invariant enabled and specifies a non-ICU tag

Regex actually has it's own globalization casing data, so if you set the AppContext swith for using invariant globalization, we detect that and use our own invariant culture mappings.

@jzabroski
Copy link
Contributor

The documentation can be a little bit misleading but what you mentioned here is not true. The CurrentCulture is not tied to specific thread and can be used in any other threads.
-- @tarekgh

cf - guess the docs should be updated? It does use AsyncLocal 🤷 https://github.com/dotnet/runtime/blob/main/src/libraries/System.Private.CoreLib/src/System/Globalization/CultureInfo.cs#L378-L397

would you be ok with pushing this consideration for vNext?
-- @joperezr

I am always pro hardening code vs. hitting moving targets. I would move it to vNext, and document the default culture change as a breaking change to the library. Given the number of bug fixes .NET 7 addresses with things that actually have globalization casing scenarios and how busted it was before, I think this is probably an acceptable breaking change? What would actually be a real breaking change a customer would complain about?

@joperezr
Copy link
Member

joperezr commented Aug 3, 2022

The source generator engine is new in .NET 7, so this particular API we are talking about is not a breaking change. The overall changes we did with IgnoreCase during this release do have some breaking changes against different versions, but we have tried keeping them to the minimum, and for cases where we are breaking, we are doing so in order to "correct" behavior (for example, previously we would check the culture twice, once when the regex is created and once at match time and that could lead to inconsistencies, that is now fixed, but it will be breaking for people that relied on that inconsistency.)

@stephentoub
Copy link
Member Author

stephentoub commented Aug 3, 2022

do we still think there is time to consider that API

There's still time. We just need to do it asap if we're going to.

My suggestion would be:

  • Source generator defaults to invariant.
  • We add a ctor that takes the culture name. Probably two, actually, one with timeout and one without.
  • If you specify the culture name, we use that instead of invariant. If you also explicitly specify RegexOptions.InvariantCulture, we error out due to the explicit conflict.
  • The auto-fixer looks to see whether InvariantCulture is specified. If it is, it removes it, since it's the default. If it's not, it looks at the options and expression to see if anything about it is ignore case, and if it is, it switches to the ctor that accepts the culture and uses the string for whatever culture the user is currently running as. That way, the fixer is very explicit about any behavior changes it may be causing.

@GrabYourPitchforks, @tarekgh, any opinions about this direction?

@jaredpar
Copy link
Member

jaredpar commented Aug 3, 2022

@stephentoub

My main concern, though, is determinism, reproducibility, and expectations. If you compile the exact same C# project on two different machines and they produce binaries with different semantics without having opted in to that, that seems concerning.

100% agree, this is my main concern as well. It's a very subtle, but significant break for determinism to do this. Enough so that I've added this API, and a few others, to the list of "banned APIs" we'll be checking generators against with an analyzer we have in the works. Essentially APIs we know to be dangerous in generators / analyzers that we want to push customers to stop using because it impacts these type of scenarios.

My suggestion would be:

This seems very reasonable to me.

@joperezr
Copy link
Member

joperezr commented Aug 3, 2022

Makes sense, I will send out a PR soon that addresses all @stephentoub's points above. @jaredpar that means there is no need on adding this API to the "banned APIs" list 😄

@jaredpar
Copy link
Member

jaredpar commented Aug 3, 2022

Oh the culture APIs are still being added to the ban list for reference of future customers. 😄

There are a lot of APIs that are effectively banned in the compiler and an analyzer is the best way to communicate that to generator / analyzer authors. Some of the banned APIs are more obvious, like don't do file IO, but others like culture are more subtle. I hadn't even thought of culture as a specific problem until reading this thread.

@jzabroski

This comment was marked as outdated.

@ghost ghost added the in-pr There is an active PR which will close this issue when it is merged label Aug 5, 2022
@joperezr joperezr added api-ready-for-review API is ready for review, it is NOT ready for implementation blocking Marks issues that we want to fast track in order to unblock other important work labels Aug 5, 2022
@joperezr joperezr changed the title Determine if/how to handle CultureInfo better in Regex (and RegexGenerator) [API Proposal] Add cultureName constructors to GeneratedRegex Aug 5, 2022
@bartonjs
Copy link
Member

bartonjs commented Aug 9, 2022

Video

  • Looks good as proposed
namespace System.Text.RegularExpressions;

[AttributeUsage(AttributeTargets.Method, AllowMultiple = false, Inherited = false)]
public sealed class GeneratedRegexAttribute : System.Attribute
{
    public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex)] string pattern) { }
    public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex, "options")] string pattern, RegexOptions options) { }
+   public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex, "options")] string pattern, RegexOptions options, string cultureName) { }
    public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex, "options")] string pattern, RegexOptions options, int matchTimeoutMilliseconds) { }
+   public GeneratedRegexAttribute([StringSyntax(StringSyntaxAttribute.Regex, "options")] string pattern, RegexOptions options, int matchTimeoutMilliseconds, string cultureName) { }

+   public string CultureName { get; }
}

@bartonjs bartonjs added api-approved API was approved in API review, it can be implemented and removed blocking Marks issues that we want to fast track in order to unblock other important work api-ready-for-review API is ready for review, it is NOT ready for implementation labels Aug 9, 2022
@ghost ghost removed the in-pr There is an active PR which will close this issue when it is merged label Aug 11, 2022
@ghost ghost locked as resolved and limited conversation to collaborators Sep 10, 2022
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
api-approved API was approved in API review, it can be implemented area-System.Text.RegularExpressions source-generator Indicates an issue with a source generator feature
Projects
None yet
Development

Successfully merging a pull request may close this issue.

7 participants