#PS7Now! PowerShell 7 Is Here
Part of #PSBlogWeek, this article is one of many from several community members and PowerShell bloggers, like me, that focus on a given topic.
The topic of this #PSBlogWeek is PowerShell 7.
I was incredibly flattered when Jeff asked me to participate in this #PSBlogWeek. Though I’m relatively new to the blogging scene, I’ve been using Windows PowerShell well over 10 years. Most recently and beyond this blog, I’ve participated in several IronScripter challenges, contributed a chapter on soft skills in the PowerShell Conference Book, Volume 2, and been part of conversations within the PowerShell community.
But enough about me, I’m sure you want #PS7Now!
With the official release of PowerShell 7, we wanted to cover some of the changes that demonstrate the efficacy of adopting the newest, fastest, and best PowerShell.
In this article, we will be looking at the JSON cmdlets - ConvertFrom-Json, ConvertTo-Json, and the new addition Test-Json*.
Test-Json was technically introduced in PowerShell Core 6.2.
        PowerShell 7 Changes to JSON Cmdlets
The Convert*-Json cmdlets were introduced with Windows PowerShell 3.0 in late 2012.
Since the release of Windows PowerShell 5.1 in early 2017, there have been several improvements to the cmdlets, including updates to the underlying dependencies.
Now, without further adieu, let’s check them out.
ConvertFrom-Json
Comparing the syntax for the ConvertFrom-Json cmdlets from Windows PowerShell 5.1 and PowerShell 7, we can see that the new cmdlet has three new parameters.
|  |  | 
We won’t be focusing on the existing parameters (which, in reality is just one), but we will examine each of the new ones in greater detail in the next few sections.
-AsHashtable
Originally introduced in PowerShell Core 6.0 and updated in later releases, this switch parameter allows the cmdlet to overcome a few limitations of outputting converted JSON to a [PsCustomObject].
Specifically, a [PsCustomObject] has the following limitations:
- Property names cannot be empty
- Property names are case insensitive
- Slower than a [Hashtable]to add new properties
- Slower than a [Hashtable]to search
What was the original behavior for the cmdlet in 5.1?
|  |  | 
Despite using valid JSON input, the older cmdlet would stop your code in its tracks. No bueno.
Now, what about the new cmdlet behavior?
|  |  | 
This returns a very succinct and descriptive error (due to the new default $ErrorView of ConciseView in PowerShell 7).
As suggested by the error message, the -AsHashtable switch will come to the rescue!
|  |  | 
The addition of this parameter makes ConvertFrom-Json play nicer with valid JSON and will give you a way to speed up your code when dealing with large datasets by manipulating the hashtable instead.
-Depth
Introduced in PowerShell Core 6.2, this parameter allows you to set the maximum depth of JSON input.
It was named to align with a similar parameter of ConvertTo-Json.
In 5.1, if you attempted to convert a greater depth than 101, you would get a ConvertFrom-Json : RecursionLimit exceeded. (606) error and a sea of red in your console.
In PowerShell 7, Get-Help -Name ConvertFrom-Json -Full reveals that -Depth parameter accepts type [Int32] and has a default value of 1024.
This is already a great improvement over the older cmdlet.
A discussion in issue #3182, which continued into pull request #8199, focused on increasing the default value.
The decision was to add the parameter to allow the user to exceed the default depth, up to [int]::MaxValue.
Now let’s see it in action.
|  |  | 
This fails with the below error.
|  |  | 
But by specifying an appropriate -Depth, the command will convert the JSON input correctly.
|  |  | 
If you consistently deal with JSON having a depth larger than 1024, you should consider using $PSDefaultParameterValues near the beginning of your scripts.
Here is an example of doubling the default maximum depth.
|  |  | 
-NoEnumerate
The last parameter for ConvertFrom-Json we are going to examine is -NoEnumerate.
From the Microsoft documentation:
Specifies that output is not enumerated.
Setting this parameter causes arrays to be sent as a single object instead of sending every element separately. This guarantees that JSON can be round-tripped via ConvertTo-Json.
Previous to PowerShell 7, the default behavior for the ConvertFrom-Json cmdlet was to not enumerate arrays by default.
This lead to confusion as it went against the behavior of how other cmdlets sent multiple objects through the pipeline.
Consider the following:
|  |  | 
The array, i.e. collection, of two integers should be seen as having two members in the output, but that is not the case.
The new behavior is to unwrap collections by default.
To handle the previous behavior, the -NoEnumerate switch was added to the cmdlet, which aligns to the implementation in the Write-Output object.
Let’s perform the same actions as the example above:
|  |  | 
Incidentally, I believe the issue referenced above was the most discussed for the ConvertFrom-Json cmdlet.
Your voice matters!
ConvertTo-Json
Now, let’s compare the syntax for the ConvertTo-Json cmdlets from Windows PowerShell 5.1 and PowerShell 7.
|  |  | 
This cmdlet also has three new parameters (though, I’m sure it was just a coincidence). Likewise, we will examine each of these new parameters in the following sections.
-EnumsAsStrings
JSON is used heavily in serialization, which essentially is translating a complex object to a simple object (typically a string representation) and vice versa. Serialization is used extensively in web applications and APIs.
Enum Backgrounder
An enumerated type, or enum, is a data type that enables a variable to be a set of predefined constants.
The value of the enum is a zero-based index, beginning with the first item.
For example, if you wanted to define a selection of car types in a script, you could use the following to create the enum then retrieve its value:
|  |  | 
The -EnumsAsStrings parameter instructs the cmdlet to output enums as their string representations, so as to ensure the data remains meaningful.
Continuing with the Enum example above, the following statements demonstrate this usefulness of this switch.
|  |  | 
-AsArray
The -AsArray switch, suggested in issue #6327, instructs the cmdlet to wrap the output object in array brackets.
This guarantees that the pipeline input can be treated as an array, whether it’s a single item or not.
|  |  | 
-EscapeHandling
The last parameter for ConvertTo-Json that we will cover is -EscapeHandling which was introduced in PowerShell Core 6.2.
While the default behavior remains unchanged, this parameter allows the user to properly escape non-ASCII and HTML characters.
Possible values with an example for each are:
|  |  | 
Introducing Test-Json
The last JSON cmdlet that we will examine is the Test-Json.
It allows you to validate JSON input against proper syntax and against a defined JSON Schema.
Before we discuss the Test-Json cmdlet, let’s take a short detour to gain a better understanding of JSON Schema.
JSON Schema
For some of you, this will be the first time that you’re hearing about JSON Schema.
In fact, I had worked with JSON for a while before realizing, just last year, that there is an IETF JSON Schema draft. This draft serves to define the structure of a given JSON object type.
Prior to this, the contents of a JSON object were at the discretion of the developer or scripter. Within small teams, there could be some differences between two objects that ultimately refer to the same type.
Let’s take an example of a Person object.
|  |  | 
This Person object is fairly simple. Perhaps too simple.
Where do you put the first name? Or last name? Nickname?
Writing unit tests or even code against the moving target of the previous Person object in this team would be tedious and prone to failures or bugs.
Consider the following new Person object JSON schema.
|  |  | 
This schema will ensure the team will use consistent property names and data types for each property. Actually, there is an error I introduced purposefully that we will discover shortly.
Validate JSON Basic Syntax with Test-Json
As I mentioned previously, the Test-Json cmdlet has two primary functions.
The first is to validate the syntax of the JSON input.
Before this cmdlet, the only PowerShell way to validate JSON was to use ConvertFrom-Json | ConvertTo-Json in the pipeline.
For reasons gleaned from the sections above for both of these cmdlets, this method was often fallible.
|  |  | 
Ah! I completely forgot to enclose the required field age in double quotes.
|  |  | 
After making the change above, let’s see the updated output.
|  |  | 
Much better.
Validate JSON Schema with Test-Json
Continuing with the Person schema that we defined above, let’s focus on the second function of the Test-Json cmdlet.
That is, we will test a JSON object against the Person schema that we have defined.
|  |  | 
Looks like I forgot to include my age in the object. Let’s correct that.
|  |  | 
Though the JSON object is validated correctly against the schema, an age of 1000 is highly unlikely. We can adjust the schema to handle real world data.
|  |  | 
Let’s try that again.
|  |  | 
We don’t have to correct my age in the example.
And that’s how you validate JSON objects in PowerShell 7, both for syntax and against a predefined schema.
In my blog post on Writing Windows Events with Smart EventData, I mention using EventData schema for each event type that you want to write.
Using JSON Schema and the Test-Json cmdlet would help your team with documentation and implementation of consistent Smart EventData.
Community Input
If you’re interested in seeing how much the PowerShell community has shaped the present (and future) of PowerShell, check out the PowerShell GitHub BI Community Dashboard page with with Pull Requests and Issues By Community and Microsoft.
#PS7Now #PSBlogWeek Contributors
Be sure to watch for more #PS7Now! #PSBlogWeek articles from my fellow contributors and myself. And be sure to follow us on Twitter and add our blogs to your feed reader. We can help you on your PowerShell enlightenment journey, along with many others in the PowerShell community.
| Author | Blog | |
|---|---|---|
| Adam Bertram | @adbertram | https://adamtheautomator.com/ | 
| Dave Carroll | @thedavecarroll | https://thedavecarroll.com/ | 
| Josh Duffney | @joshduffney | https://duffney.io/ | 
| Dan Franciscus | @danfranciscus | https://winsysblog.com/ (old cert) | 
| Jeff Hicks | @jeffhicks | https://jdhitsolutions.com/blog/ | 
| Mike Kanakos | @MikeKanakos | https://commandline.ninja/ | 
| Josh King | @WindosNZ | https://toastit.dev/ | 
| Thomas Lee | @doctordns | https://tfl09.blogspot.com/ | 
| Tommy Maynard | @thetommymaynard | https://tommymaynard.com/ | 
| Jonathan Medd | @jonathanmedd | https://www.jonathanmedd.net/ | 
| Prateek Singh | @singhprateik | https://ridicurious.com/ (bad link) | 
Summary
As you can see, there have been a great number of improvements with just these cmdlets. Imagine all of the other commands and their improvements. It’s a great time for scripters of all experience levels.
Thank you for your interest in the JSON cmdlets in PowerShell 7. And thank you for being part of the community. You are the reason we do what we do.
I hope you’ve found this interesting or informative. If you have any comments or questions, please post them below.
Thanks for reading!
And, if you haven’t already, begin your journey with PowerShell 7 now!
