Вставьте новую строку Монго с помощью PowerShell - PullRequest
0 голосов
/ 27 апреля 2018

У меня есть сценарий PowerShell, который меня раздражал весь день. Я, наконец, дошел до того, что могу получить коллекцию, но получаю ошибку, которую не могу понять.

function Get-MongoDBCollection {
  Param(
    $database,
    $CollectionName,
    $settings = $null, #[MongoDB.Driver.MongoCollectionSetting]
    $returnType = [PSOBJECT]
  )
  $method = $database.GetType().GetMethod('GetCollection')
  $gericMethod = $method.MakeGenericMethod($returnType)
  $gericMethod.Invoke($database,[object[]]($CollectionName,$settings))
}

$dbName = "MyDatabaseName"
$collectionName = "MyCollectionName"

try {
   add-type -path 'C:\Program Files\MongoDB\Drivers\System.Runtime.InteropServices.RuntimeInformation.4.0.0\lib\net45\System.Runtime.InteropServices.RuntimeInformation.dll'
  Add-Type -Path "C:\Program Files\MongoDB\Drivers\MongoDB.Bson.2.6.0\lib\net45\MongoDB.Bson.dll"

   add-type -path "C:\Program Files\MongoDB\Drivers\DnsClient.1.0.7\lib\net45\DnsClient.dll";
   Add-Type -path "C:\Program Files\MongoDB\Drivers\MongoDB.Driver.Core.2.6.0\lib\net45\MongoDb.Driver.Core.dll"
   Add-Type -Path "C:\Program Files\MongoDB\Drivers\MongoDB.Driver.2.6.0\lib\net45\MongoDB.Driver.dll"
}
catch {
  $_;
  $_.Exception.LoaderExceptions

}

$connectionString = "mongodb://localhost:27018";
$mongoClient = new-object MongoDb.Driver.MongoClient($connectionString);

$mongoDatabase = $mongoclient.GetDatabase($dbName)
$mongoDatabase.GetCollection($collectionname)

$collection = Get-MongoDBCollection $mongodatabase "SharePoint" -returnType ([MongoDB.Bson.BsonDocument]);

$datafile = Get-Content -Raw -Path "D:\datafiles\86fba866-77ed-4f40-4637-08d57d2e25b4.json" #`| ConvertFrom-Json
[MongoDB.Bson.BsonDocument] $doc = [MongoDB.Bson.BsonDocument]::Parse($datafile);
$x = $collection.InsertOne($doc)

Сценарий берет содержимое файла, который содержит строку JSON, преобразует его в BsonDocument и затем пытается вставить его. Я получаю следующую ошибку.

Argument types do not match
At line:1 char:1
+ $collection.InsertOneAsync($doc)
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  + CategoryInfo          : OperationStopped: (:) [], ArgumentException
  + FullyQualifiedErrorId : System.ArgumentException

Что я здесь не так делаю?

1 Ответ

0 голосов
/ 02 июня 2018

Это жесткое печенье! По сути, powershell не поддерживает непатентованные средства тривиальным образом. Он поддерживает их, но сложным и трудным для понимания способом!

Я нашел этот фрагмент в другом переполнении стека несколько дней назад, но не могу найти его в данный момент. Это был неприятный поиск / трек для начала. Фрагмент оригинального автора, которому я хотел бы отдать должное, выглядит следующим образом:

###
# Usage:
# $Collection = Get-MongoDBCollection $database 'collectionName'
#  or
# $Collection = Get-MongoDBCollection $database 'collectionName' -returnType  ([MongoDB.Bson.BsonDocument])
function Get-MongoDBCollection {
   Param(
    $database,
    $CollectionName,
    $settings = $null, #[MongoDB.Driver.MongoCollectionSetting]
    $returnType = [PSOBJECT]
  )
  $method = $database.GetType().GetMethod('GetCollection')
  $gericMethod = $method.MakeGenericMethod($returnType)
  $gericMethod.Invoke($database,[object[]]($CollectionName,$settings))
}

А вот полное использование контекста, которое я использовал ниже. Я также должен загрузить зависимости DnsClient.dll самостоятельно. На протяжении всего процесса, как правило, модуль powershell не выдавал полезных ошибок.

PS: Это любительский PowerShell, используется только .ps, потому что мне нужно было интерфейс Office365! Нет обещаний лучшей практики!

#########
# Globals
##

$mongoDbDriverPath = "Z:\Work\mb-rule-watch\lib\net45"
$dbName = "mbRules"
$collectionName = "Mailboxes"

#########
# Load o365 credentials/modules
##

  Import-Module MsOnline

  if( ! $credential ){
     Write-Host "Requesting Credentials - Use o365 Admin Account"
     $credential = Get-Credential
     Connect-MsolService -Credential $credential
  }

  # Prep remote session connection
  if( ! $session ){
     $session = New-PSSession `
        -ConfigurationName Microsoft.Exchange `
        -ConnectionUri https://outlook.office365.com/powershell-liveid/ `
        -Credential $credential `
        -Authentication Basic `
        -AllowRedirection

     # import commands from Microsoft Exchange Server shell
     Import-PSSession $session
  }

###########
# Functions
##

  ###
  # Usage:
  # $Collection = Get-MongoDBCollection $database 'collectionName'
  #  or
  # $Collection = Get-MongoDBCollection $database 'collectionName' -returnType  ([MongoDB.Bson.BsonDocument])
  function Get-MongoDBCollection {
    Param(
        $database,
        $CollectionName,
        $settings = $null, #[MongoDB.Driver.MongoCollectionSetting]
        $returnType = [PSOBJECT]
    )
    $method = $database.GetType().GetMethod('GetCollection')
    $gericMethod = $method.MakeGenericMethod($returnType)
    $gericMethod.Invoke($database,[object[]]($CollectionName,$settings))
  }

###########
# MAIN
##
try
{
  # Load mongo driver
  Add-Type -Path "$($mongoDbDriverPath)\DnsClient.dll"
  Add-Type -Path "$($mongoDbDriverPath)\MongoDB.Bson.dll"
  Add-Type -Path "$($mongoDbDriverPath)\MongoDB.Driver.Core.dll"
  Add-Type -Path "$($mongoDbDriverPath)\MongoDB.Driver.dll"

  # Connect to mongo
  $client = new-object -TypeName MongoDB.Driver.MongoClient -ArgumentList "mongodb://localhost"

  # Get DB handle
  [MongoDB.Driver.IMongoDatabase] $db = $client.GetDatabase( $dbName );

  # Aquire Collection handle with brute force generic hacks Via a PS god on stackoverflow.
  $collection = Get-MongoDBCollection $db $collectionName -returnType ([MongoDB.Bson.BsonDocument])

  #foreach( $mbx in $( Get-Mailbox -ResultSize Unlimited -identity example_user_id ) ){
  foreach( $mbx in $( Get-Mailbox -ResultSize Unlimited ) ){

    $identityStr = $mbx.identity

    $rules =  Get-InboxRule -Mailbox $identityStr

    # convert some huge ints (>Mongo Int64) to strings
    foreach( $rule in $rules ){
      $rule.RuleIdentity = "" + $rule.RuleIdentity + ""
    }

    # Json Stringify
    $rules_json = ConvertTo-Json $rules

    # If the mailbox had rules
    if( $rules_json ){

      write-host( "Inserting rules for: " + $identityStr )

      # Cache results to FS this time.
      echo $rules_json > var\rules\$identityStr.rules.json

      try{

        # Type convert/parse our json string
        $document = new-object -TypeName MongoDB.Bson.BsonDocument
        $document = [MongoDb.Bson.BsonDocument]::Parse( '{ "_username":"' + $identityStr + '","rules": ' + $rules_json + '}' );

        # Insert the JSON document
        $collection.InsertOne( $document )

      } catch {

        Write-Host "JSON parse or mongo insert failure"

        foreach( $x IN $_.Exception ){
          foreach( $msg IN $x ){
            Write-Error $msg
          }
        }

      }
    }
  }
}
catch
{
  Write-Host "Script errors occured"
  if( $_.Exception.LoaderExceptions ){
    Write-Host "!!Dependency Loads Failed!!"
    foreach( $msg IN $_.Exception.LoaderExceptions ){
      Write-Error $msg
    }
  } else {
    foreach( $x IN $_.Exception ){
      foreach( $msg IN $x ){
        Write-Error $msg
      }
    }
  }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...