2011-12-07 1 views
5

Sto provando a convertire un hashtable in un oggetto json da utilizzare in un servizio Web con PowerShell 2.0.Come posso convertire un hastable in una stringa json in powershell?

$testhash = @{ 
    Name = 'John Doe' 
    Age = 10 
    Amount = 10.1 
    MixedItems = (1,2,3,"a") 
    NestedHash = @{ 
     nestedkey = "nextedvalue" 
    } 
} 

function toJson($obj){ 

    $ms = New-Object IO.MemoryStream 
    $type = $obj.getType() 
    [Type[]]$types = ($obj | select -expand PsTypeNames | Select -unique) + [type]'System.Management.Automation.PSObject' 
    $js = New-Object System.Runtime.Serialization.Json.DataContractJsonSerializer $type, $types, ([int]::MaxValue), $false, $null, $false 
    $js.writeObject($ms, $obj) | out-null 
    $utf8.GetString($ms.ToArray(), 0, $ms.Length) 
    $ms.Dispose() | out-null 
} 

toJson $testhash 
'[{"Key":"Name","Value":"John Doe"},{"Key":"Age","Value":10},{"Key":"Amount","Value":10.1},{"Key":"NestedHash","Value":[{"__type":"KeyValuePairOfanyTypeanyType:#System.Collections.Generic","key":"nestedkey","value":"nextedvalue"}]},{"Key":"MixedItems","Value":[1,2,3,"a"]}]' 

sto usando DataContractJsonSerializer constructor in un modo che dovrebbe sopprimere le informazioni sul tipo, ma è ovviamente non. Sono anche divertito dall'estrarre coppie di chiavi e valori, ma mi piacerebbe che non lo facesse neanche. Che cosa sto facendo di sbagliato?

risposta

3

mi sono adattato lo script da here come di seguito:

$testhash = @{ 
    Name = 'John Doe' 
    Age = 10 
    Amount = 10.1 
    MixedItems = (1,2,3,"a") 
    NestedHash = @{ 
     nestedkey = "nextedvalue" 
    } 
} 

function Read-Stream { 
PARAM(
    [Parameter(Position=0,ValueFromPipeline=$true)]$Stream 
) 
process { 
    $bytes = $Stream.ToArray() 
    [System.Text.Encoding]::UTF8.GetString($bytes,0,$bytes.Length) 
}} 

function New-Json { 
[CmdletBinding()] 
param([Parameter(ValueFromPipeline=$true)][HashTable]$InputObject) 
begin { 
    $ser = @{} 
    $jsona = @() 
} 
process { 
    $jsoni = 
    foreach($input in $InputObject.GetEnumerator() | Where { $_.Value }) { 
     if($input.Value -is [Hashtable]) { 
     '"'+$input.Key+'": ' + (New-JSon $input.Value) 
     } else { 
     $type = $input.Value.GetType() 
     if(!$Ser.ContainsKey($Type)) { 
      $Ser.($Type) = New-Object System.Runtime.Serialization.Json.DataContractJsonSerializer $type 
     } 
     $stream = New-Object System.IO.MemoryStream 
     $Ser.($Type).WriteObject($stream, $Input.Value) 
     '"'+$input.Key+'": ' + (Read-Stream $stream) 
     } 
    } 

    $jsona += "{`n" +($jsoni -join ",`n")+ "`n}" 
} 
end { 
    if($jsona.Count -gt 1) { 
     "[$($jsona -join ",`n")]" 
    } else { 
     $jsona 
    } 
}} 

$testHash | New-Json 
+0

Questo mi sembra una cosa la funzione Serialize dovrebbe fare su di essa la propria, no? In entrambi i casi - questo funziona quasi perfettamente - fallisce su due livelli di hash annidati. – reconbot

+0

@wizard - Dovrebbe essere per semplici hashtable, penso. Puoi sembrare aroudn, ci sono già script come ho collegato a fare l'analisi di JSON. – manojlds

+0

Ho finito per usare http://poshcode.org/2930 che ha problemi ma funziona abbastanza bene - un giorno PowerShell 3.0 sarà disponibile e guarderò indietro a queste ore sprecate e sarò turbato – reconbot

16

Ok, quindi manojlds risposto per v2 quindi mi limiterò a vomitare v3 equivalente qui:

PS> @{name="oisin"; age=37} | convertto-json 
{ 
    "age": 37, 
    "name": "oisin" 
} 

Piuttosto un po 'più pulito , destra?

PowerShell 3.0 CTP2: http://www.microsoft.com/download/en/details.aspx?id=27548

+2

Vorrei poter usare powershell v3 =) – reconbot

+0

Così pulito e semplice !!! – lantrix

+0

C'è un bel po 'di _power_ in questo _shell_ :) – raghav710