This is about what can happen when things go wrong and what not to do so that things don’t go wrong.
In case you didn’t know, I work at Quest Software on the PowerGUI Team. My position is rather unique because I work exclusively with Windows PowerShell, and I don’t think there are too many jobs out there that offer that experience. The main focus of my job is to create and extend PowerPacks, which are collections of PowerShell scripts that are packaged together to define an extension to the customizable PowerGUI Admin Console. The scripts in the PowerPacks I create range anywhere from the extremely simple (one cmdlet or function call) to the very complicated (a script that calls into one or more functions that span many thousands of lines of PowerShell script).
Among others, one of the PowerPacks that I manage is the Active Directory PowerPack. The Active Directory PowerPack provides a management interface to Active Directory, and it uses the Quest AD cmdlets in its PowerShell scripts to do the actual management of Active Directory. For quite a few reasons1, I need to explicitly indicate which connection should be used when you use that PowerPack to do something with Active Directory, regardless of what it is you are doing. This means I am absolutely dependent on being able to determine what connection should be used in any script that is included in the PowerPack. For scripts that are used at the beginning of a pipeline I look for the appropriate connection information that is stored globally. For other scripts that are used later in a pipeline with Quest AD cmdlets I look at the Connection property that is attached to the objects coming down the pipeline. This model has worked very well until recently when something unexpected happened.
Almost two weeks ago, just after version 1.2 of the Quest AD cmdlets was released, it came to my attention that the Connection property that I am so dependent on was not included on objects in that release. For those of you who are not developers, this type of change (changing a public interface in an SDK) is very bad news. It is called a breaking change, and the name is very fitting – the Active Directory PowerPack started breaking all over the place for users who upgraded their Quest AD cmdlets to version 1.2!
Amid a flurry of email discussions between myself and the Quest AD cmdlets team I spent the next week and a bit putting together an update to the Active Directory PowerPack so that it would work despite these changes, basically putting the Connection property there myself if it isn’t there. At the same time, the Quest AD cmdlets team quickly started work on a 1.2.1 patch release to put the Connection property back in so that scripts outside of PowerGUI that use that property wouldn’t be affected either. The patch for the Quest AD cmdlets is not available yet, but the updated Active Directory PowerPack, and the updated Network PowerPack which also used the Quest AD cmdlets a fair amount and was therefore susceptible to this change, are now posted in the PowerPack library and available for download. If you use these PowerPacks, I strongly recommend you download and import the most recent releases of each of these PowerPacks. Their document page in the PowerPack library contains all of the information you need to upgrade your PowerPacks and get started using the new versions.
Now that I have finished dealing with the impact of a breaking change like this one and the unpleasant work that ensues, I thought it would be worthwhile to share with others what exactly consititutes a breaking change in a PowerShell cmdlet or advanced function. The following is a list of 13 rules that you should adhere to when updating your cmdlets or advanced functions if you want to avoid introducing breaking changes:
- Do not remove any published parameters that are included in your published cmdlets or advanced functions.
- Do not rename any published parameters that are included in your published cmdlets or advanced functions unless you add an alias to the parameter that is the old parameter name.
- Do not reduce or remove pipeline support for any published parameters in your published cmdlets or advanced functions that support pipeline input.
- Do not remove support for $null input for any published parameters in your published cmdlets or advanced functions that support $null input.
- Do not modify the default value for any published parameters in your published cmdlets or advanced functions unless that modification will not change how the command functions in existing scripts.
- Do not add a required flag to any parameter in published parameter sets in your published cmdlets or advanced functions.
- Do not modify the position of any published parameters in published parameter sets in your published cmdlets or advanced functions.
- Do not replace the associated object type with an incompatible type for any published parameters in your published cmdlets or advanced functions.
- Do not remove wildcard support for any published parameters in your published cmdlets or advanced functions that have wildcard support.
- Do not remove support for building the value from the remaining arguments in any published parameters in your published cmdlets or advanced functions that support building the value in this manner.
- Do not remove any of your published parameters from published parameter sets in your published cmdlets or advanced functions.
- Do not increase validation restrictions for any of your published parameters in your published cmdlets or advanced functions that have validation restrictions.
- Do not modify or remove aliases for any published parameters in your published cmdlets or advanced functions.
Additionally you need to pay attention to the objects that are returned by your cmdlets or advanced functions so that you don’t introduce breaking changes on those either. The following rules should be adhered to in order to avoid breaking changes on objects returned by your cmdlets or advanced functions:
- Do not remove a public property or method.
- Do not replace the types associated with a public property or method with incompatible types.
- Do not modify the order of the parameters in a public method.
- Do not add required parameters in a public method.
Not following these rules means risking breaking production scripts, risking making your customers unhappy, and many other potential undesirable consequences. Please, if you are seriously trying to add value to the PowerShell ecosystem by extending it with additional commands, pay attention to these rules and don’t introduce breaking changes when you update them!
Thanks for listening!
|1||Reasons why the connection must be explicitly indicated in the Active Directory PowerPack:|
|Share this post:|