PSMultipartFormData.psm1
<# .SYNOPSIS MultiPartFormData - A PowerShell module for creating and handling multipart form data for HTTP/REST requests. .DESCRIPTION A module to provide functionality for creating multipart form data used in HTTP requests, including adding form fields, files, and objects. It supports the conversion of file names to MIME types and encoding file content. .EXPORTS - Import-FileAsMultipartFormData - New-MultipartFormData .FUNCTIONS - Import-FileAsMultipartFormData Imports a file and returns a MultiPartFormData object with the file encoded as part of the form data. PARAMETERS: - FilePath (String, Mandatory): The path of the file to be added. - New-MultipartFormData Creates a new instance of the MultiPartFormData class. PARAMETERS: - None .EXAMPLE Creating and using multipart form data: $formData = New-MultipartFormData $formData.AddField("name", "John Doe") $formData.AddFile("file", "example.txt", "text/plain", "File content here") $formData.GetBody() # Output: The complete multipart form data body .EXAMPLE Importing a file as multipart form data: $multipart = Import-FileAsMultipartFormData -FilePath "C:\example.txt" $multipart.GetBody() # Output: Multipart form data body with the file included .NOTES Author: Kieron Morris/t3hn3rd (kjm@kieronmorris.me) Created: 2025-03-12 Version: 1.0.1 License: Apache License 2.0 Contributors: - Kieron Morris/t3hn3rd Dependencies: - PSMimeTypes License: Apache License 2.0 .LINK Github: https://github.com/t3hn3rd/PSMultipartFormData #> Import-Module "PSMimeTypes" <# .SYNOPSIS MultiPartFormData - A class for constructing and managing multipart form data for HTTP requests. .DESCRIPTION The `MultiPartFormData` class provides methods for adding form fields, files, and objects to a multipart form data body. It supports generating a unique boundary for each instance, and encoding files as part of the form data. .PROPERTIES - bodyLines (PSObject[]): Stores the lines of the multipart form data body. - boundary (string): A unique boundary used to separate parts in the multipart form data. - LF (string): The line feed characters used in the form data. .METHODS - AddField (string, string): Adds a form field with the specified name and value to the body. - AddFile (string, string, string, PSObject): Adds a file with the specified name, filename, MIME type, and content. - AddFile (string): Adds a file from the provided file path to the form data. - AddObject (string, PSObject): Adds a PowerShell object as a JSON-encoded field to the form data. - GetBody (): Returns the complete multipart form data body as a string. - GetBoundary (): Returns the unique boundary for the multipart form data. .EXAMPLE Add form fields and files to multipart form data $formData = [MultiPartFormData]::new() $formData.AddField("name", "John Doe") $formData.AddFile("file", "example.txt", "text/plain", "File content here") $formData.GetBody() .EXAMPLE Add a file from a path $formData.AddFile("C:\path\to\file.txt") $formData.GetBody() .EXAMPLE Add an object as a JSON field $obj = [PSCustomObject]@{ key = "value" } $formData.AddObject("jsonField", $obj) $formData.GetBody() #> class MultiPartFormData { hidden [PSObject[]]$bodyLines hidden [string]$boundary hidden [string]$LF MultiPartFormData() { $this.bodyLines = @() $this.boundary = [System.Guid]::NewGuid().ToString() $this.LF = "`r`n" } [MultiPartFormData] AddField([string]$name, [string]$value) { if($value) { $this.bodyLines += "--$($this.boundary)" $this.bodyLines += "Content-Disposition: form-data; name=`"$name`"" + $this.LF $this.bodyLines += "$value" } return $this } [MultiPartFormData] AddFile([string]$name, [string]$filename, [string]$mime, [PSObject]$fileContent) { $this.bodyLines += "--$($this.boundary)" $this.bodyLines += "Content-Disposition: form-data; name=`"$name`"; filename=`"$filename`"" $this.bodyLines += "Content-Type: $mime" + $this.LF $this.bodyLines += $fileContent return $this } [MultiPartFormData] AddFile([string]$FilePath) { if($FilePath) { $fileBytes = [System.IO.File]::ReadAllBytes($FilePath) $fileEnc = [System.Text.Encoding]::GetEncoding('ISO-8859-1').GetString($fileBytes) $Filename = Split-Path $FilePath -Leaf $MIME = Convert-FileNameToMimeType -FileName $Filename $this.AddFile('file', $Filename, $MIME, $fileEnc) } return $this } [MultiPartFormData] AddObject([string]$name, [PSObject]$object) { if($Object) { $this.AddField($name, ($object | ConvertTo-Json -Depth 100 -Compress)) } return $this } [string] GetBody() { if ($this.bodyLines.Count -eq 0) { return "" } return ($this.bodyLines -join $this.LF) + $this.LF + "--$($this.boundary)--$($this.LF)" } [string] GetBoundary() { return $this.boundary } } <# .SYNOPSIS Imports a file as part of a multipart form data object. .DESCRIPTION The `Import-FileAsMultipartFormData` function reads a file from the specified path, converts it to a byte array, encodes it in ISO-8859-1, determines its MIME type based on the file extension, and adds it to a `MultiPartFormData` object. .INPUTS - FilePath (String, Mandatory): The path to the file that will be imported into the multipart form data. .OUTPUTS [MultiPartFormData] - A `MultiPartFormData` object with the file added as part of the form data. .EXAMPLE Import a file as multipart form data multipartFormData = Import-FileAsMultipartFormData -FilePath "C:\example.txt" multipartFormData.GetBody() #> Function Import-FileAsMultipartFormData { param ( [string]$FilePath ) $fileBytes = [System.IO.File]::ReadAllBytes($FilePath) $fileEnc = [System.Text.Encoding]::GetEncoding('ISO-8859-1').GetString($fileBytes) $Filename = Split-Path $FilePath -Leaf $MIME = Convert-FileNameToMimeType -FileName $Filename $MPFD = [MultiPartFormData]::new() return $MPFD.AddFile('file', $Filename, $MIME, $fileEnc) } <# .SYNOPSIS Creates a new instance of the `MultiPartFormData` class. .DESCRIPTION The `New-MultipartFormData` function instantiates a new `MultiPartFormData` object, which can be used to build multipart form data for HTTP requests. This object allows for adding fields, files, and objects to the form data. .OUTPUTS [MultiPartFormData] - A new instance of the `MultiPartFormData` class. .EXAMPLE Create a new multipart form data object $multipartFormData = New-MultipartFormData $multipartFormData.AddField("name", "John Doe") $multipartFormData.GetBody() #> function New-MultipartFormData { return [MultiPartFormData]::new() } Export-ModuleMember -Function Import-FileAsMultipartFormData, New-MultipartFormData |