I converted two json files with the Get-LeafProperty from this post Powershell Selecting NoteProperty Type Objects From Object
This maked it easy to merge 2 json file which include different general.apps[] at least it was easy to merge the psobject created by the Get-LeafProperty.
Now I want to go back from the merged psobjects to the origninal json format.
I have a psobject like this which I like to convert to json.
`
[path] : [value]
general.apps[0].name : admin
general.apps[0].storageAccount.sku.name : Standard_LRS
general.apps[0].storageAccount.sku.tier : Standard
general.apps[0].hostingPlan.sku.name : Y1
general.apps[0].hostingPlan.sku.tier : Dynamic
general.apps[0].appconfig.AzureWebJobsDisableHomepage : True
general.apps[0].appconfig.AzureWebJobsStorage__accountName : ${Prefix}${tier}admin
general.apps[0].appconfig.CosmosDbConnectionStringOrManagedIdentity : AccountEndpoint=https://${Prefix}${tier}cosmosdb.documents.azure.com:443
general.apps[0].appconfig.cpo-blackbox-authorization-token :
general.apps[0].appconfig.DatabaseCacheRefreshMinutes :
general.apps[0].appconfig.HttpCallMaxSeconds :
general.apps[0].appconfig.EvgCpoOcpiUrl : https://${Prefix}${tier}${staging}cpoocpi.azurewebsites.net/api/
general.apps[0].appconfig.EvgCtrlWpsUrl : https://${Prefix}${tier}${staging}ctrlwps.azurewebsites.net/api/
general.apps[0].appconfig.EvgPingUrl : https://${Prefix}${tier}${staging}ping.azurewebsites.net/api/
general.apps[0].appconfig.Logging___DebugAsInformation :
general.apps[0].appconfig.Logging___TraceAsInformation :
general.apps[0].appconfig.NegotiatePostfix : /webpubsub/v100/negotiate
general.apps[0].appconfig.OcpiAuthenticationFailureDelayBaseMilliseconds :
general.apps[0].appconfig.OcpiEvgCountryCode :
general.apps[0].appconfig.OcpiLocationSuppressEvents : True
general.apps[0].appconfig.OcpiEvgPartyId :
general.apps[0].appconfig.OcpiSessionTokenMaxAgeSeconds :
general.apps[0].appconfig.OCPIv211___GetPagingLimit :
general.apps[0].appconfig.OCPPv16_OCPIv211___Cdr___Disconnected___TimerCheckConnection :
general.apps[0].appconfig.OCPPv16_OCPIv211___Cdr___Pump___TimerCheckConnection :
general.apps[0].appconfig.OCPPv16_OCPIv211___Cdr___Pump___TimerResponse :
general.apps[0].appconfig.OCPPv16_OCPIv211___LocationEvse___Disconnected___TimerCheckConnection :
general.apps[0].appconfig.OCPPv16_OCPIv211___LocationEvse___Pump___TimerCheckConnection :
general.apps[0].appconfig.OCPPv16_OCPIv211___LocationEvse___Pump___TimerResponse :
general.apps[0].appconfig.OCPPv16_OCPIv211___SessionRecovery___TimerDelayRecoveryRepeatSec :
general.apps[0].appconfig.OCPPv16_OCPIv211___SessionRecovery___TimerDelayRecoveryShotSec :
general.apps[0].appconfig.OCPPv16_OCPIv211___Statemachine___Default___TimerExitSec :
general.apps[0].appconfig.OCPPv16_OCPIv211___Statemachine___Default___TimerRecoverySec :
general.apps[0].appconfig.OCPPv16_OCPIv211___Statemachine___StartSessionState___TimerExitSec :
general.apps[0].appconfig.OCPPv16_OCPIv211___Statemachine___StartSessionRecoveryState___TimerExitSec :
general.apps[0].appconfig.OCPPv16_OCPIv211___Statemachine___WaitPatchSessionState___TimerExitSec :
general.apps[0].appconfig.OCPPv16_OCPIv211___Statemachine___WaitPatchSessionState___TimerRecoverySec :
general.apps[0].appconfig.PingUrls :
general.apps[0].appconfig.SCALE_CONTROLLER_LOGGING_ENABLED :
general.apps[0].appconfig.ServiceBusConnectionString__fullyQualifiedNamespace :
general.apps[0].appconfig.ServiceBusQueueList : cpoocpi:ctrlwps
general.apps[0].appconfig.ServiceBusName :
general.apps[0].appconfig.StatemachineCheckAgeCronSchedule :
general.apps[0].appconfig.StatemachineCheckAgeMinAgeDays :
general.apps[0].appconfig.StatemachineEngineDatabase :
general.apps[0].appconfig.StatemachineEngineContainerData :
general.apps[0].appconfig.StatemachineEngineContainerOperations :
general.apps[0].appconfig.StatemachineEngineContainerOperationsLeases :
general.apps[0].appconfig.StatemachineEngineQueueTimers :
general.apps[0].appconfig.StatemachineEngineServiceBusQueueOperations :
general.apps[0].appconfig.TimerKeepAliveCronSchedule : 0 * * * * *
general.apps[0].appconfig.TimerOcpiSessionTokenCleanupCronSchedule :
general.apps[0].appconfig.TimerOcpiV211GetCdrsCronSchedule :
general.apps[0].appconfig.TimerOcpiV211GetLocationsCronSchedule :
general.apps[0].appconfig.TimerTriggerCronSchedule :
general.apps[0].appconfig.WebPubSubEndpoint :
general.apps[0].appconfig.WebPubSubIdentityObjectId :
general.apps[0].appconfig.WebPubSubConnectionString :
general.apps[0].appconfig.WebPubSubHub :
general.apps[0].appconfig.WebPubSubHubLogging :
general.apps[0].appconfig.WsHostname : ${Prefix}${tier}admin.azurewebsites.net
general.apps[1].name : blackboxtestapi
general.apps[1].storageAccount.sku.name : Standard_LRS
general.apps[1].storageAccount.sku.tier : Standard
general.apps[1].hostingPlan.sku.name : Y1
general.apps[1].hostingPlan.sku.tier : Dynamic
general.apps[1].appconfig.AzureWebJobsDisableHomepage : True
general.apps[1].appconfig.AzureWebJobsStorage__accountName : ${Prefix}${tier}blackboxtestapi
general.apps[1].appconfig.CosmosDbConnectionStringOrManagedIdentity :
general.apps[1].appconfig.cpo-blackbox-authorization-token :
general.apps[1].appconfig.DatabaseCacheRefreshMinutes :
general.apps[1].appconfig.HttpCallMaxSeconds :
general.apps[1].appconfig.EvgCpoOcpiUrl :
general.apps[1].appconfig.EvgCtrlWpsUrl :
general.apps[1].appconfig.EvgPingUrl :
general.apps[1].appconfig.Logging___DebugAsInformation :
general.apps[1].appconfig.Logging___TraceAsInformation :
general.apps[1].appconfig.NegotiatePostfix :
general.apps[1].appconfig.OcpiAuthenticationFailureDelayBaseMilliseconds :
general.apps[1].appconfig.OcpiEvgCountryCode :
general.apps[1].appconfig.OcpiEvgPartyId :
general.apps[1].appconfig.OcpiLocationSuppressEvents : True
general.apps[1].appconfig.OcpiSessionTokenMaxAgeSeconds :
general.apps[1].appconfig.OCPIv211___GetPagingLimit :
general.apps[1].appconfig.OCPPv16_OCPIv211___Cdr___Disconnected___TimerCheckConnection :
general.apps[1].appconfig.OCPPv16_OCPIv211___Cdr___Pump___TimerCheckConnection :
general.apps[1].appconfig.OCPPv16_OCPIv211___Cdr___Pump___TimerResponse :
general.apps[1].appconfig.OCPPv16_OCPIv211___LocationEvse___Disconnected___TimerCheckConnection :
general.apps[1].appconfig.OCPPv16_OCPIv211___LocationEvse___Pump___TimerCheckConnection :
general.apps[1].appconfig.OCPPv16_OCPIv211___LocationEvse___Pump___TimerResponse :
general.apps[1].appconfig.OCPPv16_OCPIv211___SessionRecovery___TimerDelayRecoveryRepeatSec :
general.apps[1].appconfig.OCPPv16_OCPIv211___SessionRecovery___TimerDelayRecoveryShotSec :
general.apps[1].appconfig.OCPPv16_OCPIv211___Statemachine___Default___TimerExitSec :
general.apps[1].appconfig.OCPPv16_OCPIv211___Statemachine___Default___TimerRecoverySec :
general.apps[1].appconfig.OCPPv16_OCPIv211___Statemachine___StartSessionState___TimerExitSec :
general.apps[1].appconfig.OCPPv16_OCPIv211___Statemachine___StartSessionRecoveryState___TimerExitSec :
general.apps[1].appconfig.OCPPv16_OCPIv211___Statemachine___WaitPatchSessionState___TimerExitSec :
general.apps[1].appconfig.OCPPv16_OCPIv211___Statemachine___WaitPatchSessionState___TimerRecoverySec :
general.apps[1].appconfig.PingUrls :
general.apps[1].appconfig.SCALE_CONTROLLER_LOGGING_ENABLED :
general.apps[1].appconfig.ServiceBusConnectionString__fullyQualifiedNamespace :
general.apps[1].appconfig.ServiceBusQueueList :
general.apps[1].appconfig.ServiceBusName :
general.apps[1].appconfig.StatemachineCheckAgeCronSchedule :
general.apps[1].appconfig.StatemachineCheckAgeMinAgeDays :
general.apps[1].appconfig.StatemachineEngineDatabase :
general.apps[1].appconfig.StatemachineEngineContainerData :
general.apps[1].appconfig.StatemachineEngineContainerOperations :
general.apps[1].appconfig.StatemachineEngineContainerOperationsLeases :
general.apps[1].appconfig.StatemachineEngineQueueTimers :
general.apps[1].appconfig.StatemachineEngineServiceBusQueueOperations :
general.apps[1].appconfig.TimerKeepAliveCronSchedule : 0 * * * * *
general.apps[1].appconfig.TimerOcpiSessionTokenCleanupCronSchedule :
general.apps[1].appconfig.TimerOcpiV211GetLocationsCronSchedule :
general.apps[1].appconfig.TimerOcpiV211GetCdrsCronSchedule :
general.apps[1].appconfig.TimerTriggerCronSchedule :
general.apps[1].appconfig.WebPubSubConnectionString :
general.apps[1].appconfig.WebPubSubEndpoint :
general.apps[1].appconfig.WebPubSubIdentityObjectId :
general.apps[1].appconfig.WebPubSubHub : blackboxtest
general.apps[1].appconfig.WebPubSubHubLogging : logging
`
I want to convert this object to json. ConvertTo-Json gives me this =>
`
{
"general.apps[0].name": "admin",
"general.apps[0].storageAccount.sku.name": "Standard_LRS",
"general.apps[0].storageAccount.sku.tier": "Standard",
"general.apps[0].hostingPlan.sku.name": "Y1",
"general.apps[0].hostingPlan.sku.tier": "Dynamic",
<snip>
`
But I want to get something like this.
`
{
"general": {
"apps":[
{
"name": "admin",
"storageAccount":
{
"sku": {
"name": "Standard_LRS",
"tier": "Standard"
},
"queues": []
},
"hostingPlan":{
"sku": {
"name": "Y1",
"tier": "Dynamic"
}
},
"insight": {},
"appconfig": {
"AzureWebJobsDisableHomepage": true,
"AzureWebJobsStorage__accountName": "${Prefix}${tier}admin",
<snip>
`
I can run a loop over the Path variable and split it on '.' and try to create path again. I wonder if there is no easier solution.
I solved it differently. Instead of merging via the Get-Leafeproperty, I use as script that merges the json directly. The next function merges complex json just fine.