You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			113 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
		
		
			
		
	
	
			113 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
|   | # Breaking Changes
 | ||
|  | 
 | ||
|  | We take compatibility in .NET Framework and .NET Core extremely seriously. | ||
|  | 
 | ||
|  | Although .NET Core can be deployed app local, we are engineering it such that | ||
|  | portable libraries can target it and still run on .NET Framework as well. This | ||
|  | means that the behavior of .NET Framework can constrain the implementation of | ||
|  | any overlapping APIs in .NET Core. | ||
|  | 
 | ||
|  | Below is a summary of some documentation we have internally about what kinds of | ||
|  | things constitute breaking changes, how we categorize them, and how we decide | ||
|  | what we're willing to take. | ||
|  | 
 | ||
|  | Note that these rules only apply to APIs that have shipped in a previous RTM | ||
|  | release. New APIs still under development can be modified but we are still | ||
|  | cautious not to disrupt the ecosystem unnecessarily when prerelease APIs change. | ||
|  | 
 | ||
|  | To help triage breaking changes, we classify them in to four buckets: | ||
|  | 
 | ||
|  | 1. Public Contract | ||
|  | 2. Reasonable Grey Area | ||
|  | 3. Unlikely Grey Area | ||
|  | 4. Clearly Non-Public | ||
|  | 
 | ||
|  | ## Bucket 1: Public Contract
 | ||
|  | *Clear [violation of public contract][breaking-change].* | ||
|  | 
 | ||
|  | Examples: | ||
|  | * Renaming or removing of a public type, member, or parameter | ||
|  | * Changing the value of a public constant or enum member | ||
|  | * Sealing a type that wasn't sealed | ||
|  | * Making a virtual member abstract | ||
|  | * Adding an interface to the set of base types of an interface | ||
|  | * Removing a type or interface from the set of base types | ||
|  | * Changing the return type of a member | ||
|  | * ...or any other [incompatible change][breaking-change] to the shape of an API | ||
|  | 
 | ||
|  | [breaking-change]: breaking-change-rules.md#source-and-binary-compatibility-changes | ||
|  | 
 | ||
|  | ## Bucket 2: Reasonable Grey Area
 | ||
|  | *[Change of behavior][behavioral-changes] that customers would have reasonably | ||
|  | depended on.* | ||
|  | 
 | ||
|  | Examples: | ||
|  | 
 | ||
|  | * Throwing a new/different exception type in an existing common scenario | ||
|  | * An exception is no longer thrown | ||
|  | * A different behavior is observed after the change for an input | ||
|  | * decreasing the range of accepted values within a given parameter | ||
|  | * A new instance field is added to a type (impacts serialization) | ||
|  | * Change in timing/order of events (even when not specified in docs) | ||
|  | * Change in parsing of input and throwing new errors (even if parsing behavior | ||
|  |   is not specified in the docs) | ||
|  | 
 | ||
|  | These require judgment: how predictable, obvious, consistent was the behavior? | ||
|  | 
 | ||
|  | [behavioral-changes]: breaking-change-rules.md#behavioral-changes | ||
|  | 
 | ||
|  | ## Bucket 3: Unlikely Grey Area
 | ||
|  | *Change of behavior that customers could have depended on, but probably | ||
|  | wouldn't.* | ||
|  | 
 | ||
|  | Examples: | ||
|  | 
 | ||
|  | * Correcting behavior in a subtle corner case | ||
|  | 
 | ||
|  | As with changes in bucket 2, these require judgment: what is reasonable and | ||
|  | what's not? | ||
|  | 
 | ||
|  | ## Bucket 4: Clearly Non-Public
 | ||
|  | *Changes to surface area or behavior that is clearly internal or non-breaking | ||
|  | in theory, but breaks an app.* | ||
|  | 
 | ||
|  | Examples: | ||
|  | 
 | ||
|  | * Changes to internal API that break private reflection | ||
|  | 
 | ||
|  | It is impossible to evolve a code base without making such changes, so we don't | ||
|  | require up-front approval for these, but we will sometimes have to go back and | ||
|  | revisit such change if there's too much pain inflicted on the ecosystem through | ||
|  | a popular app or library. | ||
|  | 
 | ||
|  | This bucket is painful for the machine-wide .NET Framework, but we do have much | ||
|  | more latitude here in .NET Core. | ||
|  | 
 | ||
|  | ## What This Means for Contributors
 | ||
|  | 
 | ||
|  | * All bucket 1, 2, and 3 breaking changes require talking to the repo owners | ||
|  |   first: | ||
|  |     - We generally **don't accept** change proposals that are in bucket #1. | ||
|  |     - We **might accept** change proposals that are in #2 and #3 after a | ||
|  |       risk-benefit analysis. See below for more details. | ||
|  |     - We **usually accept** changes that are in bucket #4 | ||
|  | * If you're not sure which bucket applies to a given change, contact us. | ||
|  | 
 | ||
|  | ### Risk-Benefit Analysis
 | ||
|  | 
 | ||
|  | For buckets #2 and #3 we apply a risk-benefit analysis. It doesn't matter if the | ||
|  | old behavior is "wrong", we still need to think through the implications. This | ||
|  | can result in one of the following outcomes:  | ||
|  | 
 | ||
|  | * **Accepted with compat switch**. Depending on the estimated customer impact, | ||
|  |   we may decide to add a compat switch that allows consumers to bring back the | ||
|  |   old behavior if necessary. | ||
|  | 
 | ||
|  | * **Accepted**. In some minor cases, we may decide to accept the change if the | ||
|  |   benefit is large and the risk is super low or if the risk is moderate and a | ||
|  |   compat switch isn't viable. | ||
|  | 
 | ||
|  | * **Rejected**. If the risk is too high and/or the improvement too minor, we may | ||
|  |   decide not to accept the change proposal at all. We can help identify | ||
|  |   alternatives such as introducing a new API and obsoleting the old one. |