diff --git a/.github/actions/spelling/expect.txt b/.github/actions/spelling/expect.txt
index b5d725bbf0..e02b4b6b9b 100644
--- a/.github/actions/spelling/expect.txt
+++ b/.github/actions/spelling/expect.txt
@@ -38,6 +38,7 @@ attr
AType
AUrl
Authenticode
+azurefd
azurewebsites
bcp
BEFACEF
@@ -97,6 +98,7 @@ createmanifestmetadata
cswinrt
ctc
currentuser
+cxfsgwfxarb
DACL
datetimeoffset
Dbg
@@ -122,8 +124,10 @@ ecfrbrowse
ECustom
EFGH
EFile
+efileresource
endregion
ENDSESSION
+EPester
epth
EQU
errmsg
@@ -180,6 +184,7 @@ Howto
hre
hresults
hrow
+hwg
hwnd
IARP
IAttachment
@@ -350,6 +355,7 @@ pidlist
pkgmgr
pkindex
pkix
+pme
PMS
positionals
powershellgallery
@@ -423,6 +429,7 @@ SMTO
sortof
sourceforge
SOURCESDIRECTORY
+sourceversion
spamming
SPAPI
Srinivasan
@@ -445,6 +452,7 @@ Syncy
sysrefcomp
systemnotsupported
Tagit
+taskhostw
TCpp
tcs
TEMPDIRECTORY
@@ -458,6 +466,7 @@ timespan
timezone
Tlg
tombstoned
+TOperation
TOptions
TProgress
transitioning
diff --git a/Localization/Policies/de-DE/DesktopAppInstaller.adml b/Localization/Policies/de-DE/DesktopAppInstaller.adml
index f22b6b9668..5514a33447 100644
--- a/Localization/Policies/de-DE/DesktopAppInstaller.adml
+++ b/Localization/Policies/de-DE/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@ Wenn Sie diese Einstellung deaktivieren oder nicht konfigurieren, können Benutz
Wenn Sie diese Richtlinie aktivieren oder nicht konfigurieren, können Benutzer die Windows Package Manager CLI-Befehle und PowerShell-Cmdlets ausführen. (Vorausgesetzt, die Richtlinie "App-Installationsprogramm aktivieren" ist nicht deaktiviert).
Diese Richtlinie hat keinen Einfluss auf die Richtlinie "App-Installationsprogramm aktivieren".
+ Aktivieren der Windows-Paket-Manager-Konfiguration
+ Diese Richtlinie steuert, ob das Windows-Paket-Manager Konfigurationsfeature von Benutzern verwendet werden kann.
+
+Wenn Sie diese Einstellung aktivieren oder nicht konfigurieren, können Benutzer das Konfigurationsfeature Windows-Paket-Manager verwenden.
+
+Wenn Sie diese Einstellung deaktivieren, können Benutzer das Konfigurationsfeature Windows-Paket-Manager nicht verwenden.
diff --git a/Localization/Policies/es-ES/DesktopAppInstaller.adml b/Localization/Policies/es-ES/DesktopAppInstaller.adml
index 138e17c671..38c92029c1 100644
--- a/Localization/Policies/es-ES/DesktopAppInstaller.adml
+++ b/Localization/Policies/es-ES/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@ Si deshabilita esta directiva, los usuarios no podrán ejecutar la CLI de Admini
Si habilita o no configura esta directiva, los usuarios podrán ejecutar los comandos de la CLI de Administrador de paquetes de Windows y los cmdlets de PowerShell. (La directiva "Habilitar Instalador de aplicación" proporcionada no está deshabilitada).
Esta directiva no invalida la directiva "Habilitar Instalador de aplicación".
+ Habilitar configuración de Administrador de paquetes de Windows
+ Esta directiva controla si los usuarios pueden usar la característica de configuración Administrador de paquetes de Windows.
+
+Si habilita o no establece esta configuración, los usuarios podrán usar la característica de configuración Administrador de paquetes de Windows.
+
+Si deshabilita esta configuración, los usuarios no podrán usar la característica de configuración Administrador de paquetes de Windows.
diff --git a/Localization/Policies/fr-FR/DesktopAppInstaller.adml b/Localization/Policies/fr-FR/DesktopAppInstaller.adml
index 82374139c8..cb470a500e 100644
--- a/Localization/Policies/fr-FR/DesktopAppInstaller.adml
+++ b/Localization/Policies/fr-FR/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@ Si vous désactivez cette stratégie, les utilisateurs ne peuvent pas exécuter
Si vous activez ou ne configurez pas cette stratégie, les utilisateurs peuvent exécuter les commandes Gestionnaire de package Windows CLI et PowerShell. (La stratégie « Activer Programme d'installation d'application » fournie n’est pas désactivée).
Cette stratégie ne remplace pas la stratégie « Activer Programme d'installation d'application ».
+ Activer la configuration Gestionnaire de package Windows
+ Cette stratégie contrôle si la fonctionnalité de configuration Gestionnaire de package Windows peut être utilisée par les utilisateurs.
+
+Si vous activez ou ne configurez pas ce paramètre, les utilisateurs peuvent utiliser la fonctionnalité de configuration Gestionnaire de package Windows.
+
+Si vous désactivez ce paramètre, les utilisateurs ne peuvent pas utiliser la fonctionnalité de configuration Gestionnaire de package Windows.
diff --git a/Localization/Policies/it-IT/DesktopAppInstaller.adml b/Localization/Policies/it-IT/DesktopAppInstaller.adml
index ffdb7f9c96..1304355c59 100644
--- a/Localization/Policies/it-IT/DesktopAppInstaller.adml
+++ b/Localization/Policies/it-IT/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@ Se disabiliti questo criterio, gli utenti non potranno eseguire l'interfaccia de
Se abiliti o non configuri questo criterio, gli utenti potranno eseguire i comandi dell'interfaccia della riga di comando Gestione pacchetti Windows e i cmdlet di PowerShell. (Il criterio "Abilita Programma di installazione app" specificato non è disabilitato).
Questo criterio non esegue l'override del criterio "Abilita Programma di installazione app".
+ Abilita la configurazione di Gestione pacchetti
+ Questo criterio controlla se la funzionalità di configurazione Gestione pacchetti Windows può essere usata dagli utenti.
+
+Se si abilita o non si configura questa impostazione, gli utenti potranno usare la funzionalità di configurazione Gestione pacchetti Windows.
+
+Se si disabilita questa impostazione, gli utenti non potranno usare la funzionalità di configurazione Gestione pacchetti Windows.
diff --git a/Localization/Policies/ja-JP/DesktopAppInstaller.adml b/Localization/Policies/ja-JP/DesktopAppInstaller.adml
index 8988ae1869..bf3160c5a9 100644
--- a/Localization/Policies/ja-JP/DesktopAppInstaller.adml
+++ b/Localization/Policies/ja-JP/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@
このポリシーを有効にした場合、または構成しなかった場合、ユーザーは Windows パッケージ マネージャー CLI コマンドと PowerShell コマンドレットを実行できます ([アプリ インストーラーを有効にする] ポリシーが無効になっている場合に限る)。
このポリシーは、[アプリ インストーラーを有効にする] ポリシーをオーバーライドしません。
+ Windows パッケージ マネージャーの構成を有効にする
+ このポリシーでは、Windows パッケージ マネージャー構成機能をユーザーが使用できるかどうかを制御します。
+
+この設定を有効にした場合、または構成しなかった場合、ユーザーはWindows パッケージ マネージャー構成機能を使用できます。
+
+この設定を無効にすると、ユーザーはWindows パッケージ マネージャー構成機能を使用できなくなります。
diff --git a/Localization/Policies/ko-KR/DesktopAppInstaller.adml b/Localization/Policies/ko-KR/DesktopAppInstaller.adml
index 57735b08f0..3e0bc48966 100644
--- a/Localization/Policies/ko-KR/DesktopAppInstaller.adml
+++ b/Localization/Policies/ko-KR/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@
이 정책을 활성화하거나 구성하지 않으면 사용자는 Windows 패키지 관리자 CLI 명령 및 PowerShell cmdlet을 실행할 수 있습니다. ('앱 설치 프로그램 활성화' 정책이 비활성화되지 않은 경우).
이 정책은 '앱 설치 프로그램 활성화' 정책보다 우선 적용되지 않습니다.
+ Windows 패키지 관리자 구성 사용
+ 이 정책은 사용자가 Windows 패키지 관리자 구성 기능을 사용할 수 있는지 여부를 제어합니다.
+
+이 설정을 사용하거나 구성하지 않으면 사용자가 Windows 패키지 관리자 구성 기능을 사용할 수 있습니다.
+
+이 설정을 사용하지 않으면 사용자가 Windows 패키지 관리자 구성 기능을 사용할 수 없습니다.
diff --git a/Localization/Policies/pt-BR/DesktopAppInstaller.adml b/Localization/Policies/pt-BR/DesktopAppInstaller.adml
index 5526461bfa..98eaf861a3 100644
--- a/Localization/Policies/pt-BR/DesktopAppInstaller.adml
+++ b/Localization/Policies/pt-BR/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@ Se você desabilitar essa política, os usuários não poderão executar o Geren
Se você habilitar ou não configurar essa política, os usuários poderão executar os comandos Gerenciador de Pacotes do Windows CLI e os cmdlets do PowerShell. (A política "Habilitar Instalador de Aplicativo" fornecida não está desabilitada).
Esta política não substitui a política "Habilitar Instalador de Aplicativo".
+ Habilitar as Configurações do Gerenciador de Pacotes do Windows
+ Esta política controla se o Gerenciador de Pacotes do Windows de configuração pode ser usado pelos usuários.
+
+Se você habilitar ou não definir essa configuração, os usuários poderão usar o recurso Gerenciador de Pacotes do Windows configuração.
+
+Se você desabilitar essa configuração, os usuários não poderão usar o recurso Gerenciador de Pacotes do Windows configuração.
diff --git a/Localization/Policies/ru-RU/DesktopAppInstaller.adml b/Localization/Policies/ru-RU/DesktopAppInstaller.adml
index a7c05553b3..60d6c11a44 100644
--- a/Localization/Policies/ru-RU/DesktopAppInstaller.adml
+++ b/Localization/Policies/ru-RU/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@
Если эта политика включена или не настроена, пользователи смогут выполнять команды CLI Диспетчера пакетов Windows и командлеты PowerShell. (Указанная политика "Включить установщик приложений" не отключена.)
Эта политика не переопределяет политику "Включить установщик приложений".
+ Включить конфигурацию Диспетчера пакетов Windows
+ Эта политика определяет, Диспетчер пакетов Windows ли пользователи могут использовать эту Диспетчер пакетов Windows конфигурацию.
+
+Если этот параметр включен или не настроен, пользователи смогут использовать Диспетчер пакетов Windows конфигурации.
+
+Если этот параметр отключен, пользователи не смогут использовать Диспетчер пакетов Windows конфигурации.
diff --git a/Localization/Policies/zh-CN/DesktopAppInstaller.adml b/Localization/Policies/zh-CN/DesktopAppInstaller.adml
index f86c0a0e94..c0de8f0205 100644
--- a/Localization/Policies/zh-CN/DesktopAppInstaller.adml
+++ b/Localization/Policies/zh-CN/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@
如果启用或未配置此策略,用户将能够执行 Windows 程序包管理器 CLI 命令和 PowerShell cmdlet。(未禁用提供的“启用应用安装程序”策略)。
此策略不会替代“启用应用安装程序”策略。
+ 启用 Windows 程序包管理器配置
+ 此策略控制用户是否可以使用Windows 程序包管理器配置功能。
+
+如果启用或未配置此设置,则用户将能够使用Windows 程序包管理器配置功能。
+
+如果禁用此设置,则用户将无法使用Windows 程序包管理器配置功能。
diff --git a/Localization/Policies/zh-TW/DesktopAppInstaller.adml b/Localization/Policies/zh-TW/DesktopAppInstaller.adml
index 264cda03de..faca09c4db 100644
--- a/Localization/Policies/zh-TW/DesktopAppInstaller.adml
+++ b/Localization/Policies/zh-TW/DesktopAppInstaller.adml
@@ -103,6 +103,12 @@
如果您啟用或未設定此原則,使用者將可以執行 Windows 封裝管理員 CLI 命令和 PowerShell Cmdlet。(提供的 [啟用應用程式安裝程式] 原則將不會停用)。
此原則不會覆寫 [啟用應用程式安裝程式] 原則。
+ 啟用 Windows 封裝管理員設定
+ 此原則控制使用者是否可以使用Windows 封裝管理員設定功能。
+
+如果您啟用或未設定這個設定,使用者將可以使用Windows 封裝管理員設定功能。
+
+如果您停用這個設定,使用者將無法使用Windows 封裝管理員設定功能。
diff --git a/Localization/Resources/de-DE/winget.resw b/Localization/Resources/de-DE/winget.resw
index 699e76ab6e..fb08636cf3 100644
--- a/Localization/Resources/de-DE/winget.resw
+++ b/Localization/Resources/de-DE/winget.resw
@@ -1697,6 +1697,9 @@ Geben Sie eine Option für --source an, um den Vorgang fortzusetzen.
{0} Pakete verfügen über einen Pin, der vor dem Upgrade entfernt werden muss
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ Das Paket kann nicht mit winget aktualisiert werden. Verwenden Sie die vom Herausgeber bereitgestellte Methode zum Aktualisieren dieses Pakets.
+
Aktualisieren von Paketen auch dann, wenn sie über einen nicht blockierenden Pin verfügen
@@ -2074,4 +2077,7 @@ Geben Sie eine Option für --source an, um den Vorgang fortzusetzen.
Der Wert "--module-path" muss "currentuser", "allusers", "default" oder ein absoluter Pfad sein.
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ Aktivieren der Windows-Paket-Manager-Konfiguration
+
\ No newline at end of file
diff --git a/Localization/Resources/es-ES/winget.resw b/Localization/Resources/es-ES/winget.resw
index 5a8afbc041..567ea6a4c7 100644
--- a/Localization/Resources/es-ES/winget.resw
+++ b/Localization/Resources/es-ES/winget.resw
@@ -1697,6 +1697,9 @@ Especifique uno de ellos con la opción --source para continuar.
{0} paquetes tienen un pin que debe quitarse antes de la actualización
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ No se puede actualizar el paquete con winget. Use el método proporcionado por el publicador para actualizar este paquete.
+
Actualizar paquetes aunque tengan un PIN que no sea de bloqueo
@@ -2074,4 +2077,7 @@ Especifique uno de ellos con la opción --source para continuar.
El valor de `--module-path allusers` debe ser `currentuser`, `allusers`, `default` o una ruta de acceso absoluta.
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ Habilitar configuración de Administrador de paquetes de Windows
+
\ No newline at end of file
diff --git a/Localization/Resources/fr-FR/winget.resw b/Localization/Resources/fr-FR/winget.resw
index e928729a83..4a2303058c 100644
--- a/Localization/Resources/fr-FR/winget.resw
+++ b/Localization/Resources/fr-FR/winget.resw
@@ -1697,6 +1697,9 @@ Spécifiez l’un d’entre eux à l’aide de l’option --source pour continue
{0} package(s) ont une épingle qui doit être supprimée avant la mise à niveau
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ Impossible de mettre à niveau le package à l’aide de Winget. Utilisez la méthode fournie par l’éditeur pour mettre à niveau ce package.
+
Mettre à niveau les packages même s’ils ont une épingle non bloquante
@@ -2074,4 +2077,7 @@ Spécifiez l’un d’entre eux à l’aide de l’option --source pour continue
La valeur '--module-path' doit être 'currentuser', 'allusers', 'default' ou un chemin absolu.
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ Activer la configuration Gestionnaire de package Windows
+
\ No newline at end of file
diff --git a/Localization/Resources/it-IT/winget.resw b/Localization/Resources/it-IT/winget.resw
index 8b45f97e9e..260e04d2da 100644
--- a/Localization/Resources/it-IT/winget.resw
+++ b/Localization/Resources/it-IT/winget.resw
@@ -1697,6 +1697,9 @@ Specificarne uno utilizzando l'opzione --source per continuare.
{0} pacchetti hanno un PIN che deve essere rimosso prima dell'aggiornamento
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ Non è possibile aggiornare il pacchetto con WinGet. Usare il metodo fornito dal server di pubblicazione per l'aggiornamento di questo pacchetto.
+
Aggiorna i pacchetti anche se hanno un PIN non bloccante
@@ -2074,4 +2077,7 @@ Specificarne uno utilizzando l'opzione --source per continuare.
Il valore di '--module-path' deve essere 'currentuser', 'allusers', 'default' o un percorso assoluto.
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ Abilita la configurazione di Gestione pacchetti
+
\ No newline at end of file
diff --git a/Localization/Resources/ja-JP/winget.resw b/Localization/Resources/ja-JP/winget.resw
index 595bb21b82..cb56fd1077 100644
--- a/Localization/Resources/ja-JP/winget.resw
+++ b/Localization/Resources/ja-JP/winget.resw
@@ -1697,6 +1697,9 @@
{0} 個のパッケージには、アップグレードする前に削除する必要があるピンがあります
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ winget を使用してパッケージをアップグレードすることはできません。このパッケージをアップグレードするには、発行元から提供された方法を使用してください。
+
ブロックでないピンを持っている場合でもパッケージをアップグレードする
@@ -2074,4 +2077,7 @@
`--module-path` 値は、`currentuser`、`allusers`、`default`、または絶対パスである必要があります。
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ Windows パッケージ マネージャーの構成を有効にする
+
\ No newline at end of file
diff --git a/Localization/Resources/ko-KR/winget.resw b/Localization/Resources/ko-KR/winget.resw
index de853c03ff..8bad81dc58 100644
--- a/Localization/Resources/ko-KR/winget.resw
+++ b/Localization/Resources/ko-KR/winget.resw
@@ -1697,6 +1697,9 @@
{0} 패키지에 업그레이드 전에 제거해야 하는 PIN이 있습니다.
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ winget을 사용하여 패키지를 업그레이드할 수 없습니다. 게시자가 제공한 메서드를 사용하여 이 패키지를 업그레이드하세요.
+
차단되지 않는 PIN이 있는 경우에도 패키지 업그레이드
@@ -2074,4 +2077,7 @@
'--module-path' 값은 'currentuser', 'allusers', 'default' 또는 절대 경로여야 합니다.
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ Windows 패키지 관리자 구성 사용
+
\ No newline at end of file
diff --git a/Localization/Resources/pt-BR/winget.resw b/Localization/Resources/pt-BR/winget.resw
index 13d0f0769c..a1e191fb3d 100644
--- a/Localization/Resources/pt-BR/winget.resw
+++ b/Localization/Resources/pt-BR/winget.resw
@@ -1697,6 +1697,9 @@ Especifique um deles usando a opção --source para continuar.
{0} pacote(s) têm um pin que precisa ser removido antes da atualização
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ O pacote não pode ser atualizado usando winget. Use o método fornecido pelo editor para atualizar este pacote.
+
Atualizar pacotes mesmo que eles tenham um marcador sem bloqueio
@@ -2074,4 +2077,7 @@ Especifique um deles usando a opção --source para continuar.
O valor '--module-path' deve ser 'currentuser', 'allusers', 'default' ou um caminho absoluto.
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ Habilitar as Configurações do Gerenciador de Pacotes do Windows
+
\ No newline at end of file
diff --git a/Localization/Resources/ru-RU/winget.resw b/Localization/Resources/ru-RU/winget.resw
index db17620598..104d2e6d19 100644
--- a/Localization/Resources/ru-RU/winget.resw
+++ b/Localization/Resources/ru-RU/winget.resw
@@ -1697,6 +1697,9 @@
Несколько ({0}) пакетов используют закрепление, которое необходимо удалить перед обновлением
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ Невозможно обновить пакет с помощью winget. Для обновления этого пакета используйте метод, предоставленный издателем.
+
Обновление пакетов, даже если они используют неблокирующее закрепление
@@ -2074,4 +2077,7 @@
Для --module-path должно быть задано значение currentuser, allusers, default или абсолютный путь.
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ Включить конфигурацию Диспетчера пакетов Windows
+
\ No newline at end of file
diff --git a/Localization/Resources/zh-CN/winget.resw b/Localization/Resources/zh-CN/winget.resw
index 720e748757..d306342534 100644
--- a/Localization/Resources/zh-CN/winget.resw
+++ b/Localization/Resources/zh-CN/winget.resw
@@ -1697,6 +1697,9 @@
{0} 程序包拥有需要在升级前移除的包钉
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ 无法使用 winget 升级包。请使用发布者提供的方法升级此包。
+
即使程序包拥有非阻止性包钉,也要升级程序包
@@ -2074,4 +2077,7 @@
“--module-path”值必须是“currentuser”、“allusers”、“default”或绝对路径。
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ 启用 Windows 程序包管理器配置
+
\ No newline at end of file
diff --git a/Localization/Resources/zh-TW/winget.resw b/Localization/Resources/zh-TW/winget.resw
index bf877d133c..d01640b05b 100644
--- a/Localization/Resources/zh-TW/winget.resw
+++ b/Localization/Resources/zh-TW/winget.resw
@@ -1697,6 +1697,9 @@
{0} 個包裹具有更新前需要先移除的釘選
{Locked="{0}"} {0} is a placeholder that is replaced by an integer number of packages with pins that prevent upgrade
+
+ 無法使用 winget 升級封裝。請使用發行者提供的方法來升級此套件。
+
即使有非封鎖釘選仍要更新套件
@@ -2074,4 +2077,7 @@
'--module-path' 值必須是 'currentuser'、'allusers'、'default' 或絕對路徑。
{Locked="{--module-path}, {currentuser}, {allusers}, {default}}
+
+ 啟用 Windows 封裝管理員設定
+
\ No newline at end of file
diff --git a/doc/Settings.md b/doc/Settings.md
index 96faa510a3..1e56b714dd 100644
--- a/doc/Settings.md
+++ b/doc/Settings.md
@@ -272,4 +272,15 @@ You can enable the feature as shown below.
"experimentalFeatures": {
"configuration": true
},
+```
+
+### windowsFeature
+
+This feature enables the ability to enable Windows Feature dependencies during installation.
+You can enable the feature as shown below.
+
+```json
+ "experimentalFeatures": {
+ "windowsFeature": true
+ },
```
\ No newline at end of file
diff --git a/schemas/JSON/settings/settings.schema.0.2.json b/schemas/JSON/settings/settings.schema.0.2.json
index 279b9ed486..3911e4e8f7 100644
--- a/schemas/JSON/settings/settings.schema.0.2.json
+++ b/schemas/JSON/settings/settings.schema.0.2.json
@@ -241,6 +241,11 @@
"description": "Enable support for configuration",
"type": "boolean",
"default": false
+ },
+ "windowsFeature": {
+ "description": "Enable support for enabling Windows Feature(s)",
+ "type": "boolean",
+ "default": false
}
}
}
diff --git a/src/AppInstallerCLI.sln b/src/AppInstallerCLI.sln
index 9ec79883cd..924fc183e5 100644
--- a/src/AppInstallerCLI.sln
+++ b/src/AppInstallerCLI.sln
@@ -1577,13 +1577,9 @@ Global
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.Fuzzing|x86.ActiveCfg = Release|Any CPU
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.Fuzzing|x86.Build.0 = Release|Any CPU
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.PowerShell|Any CPU.ActiveCfg = Release|Any CPU
- {52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.PowerShell|Any CPU.Build.0 = Release|Any CPU
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.PowerShell|ARM64.ActiveCfg = Release|Any CPU
- {52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.PowerShell|ARM64.Build.0 = Release|Any CPU
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.PowerShell|x64.ActiveCfg = Release|Any CPU
- {52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.PowerShell|x64.Build.0 = Release|Any CPU
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.PowerShell|x86.ActiveCfg = Release|Any CPU
- {52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.PowerShell|x86.Build.0 = Release|Any CPU
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.Release|Any CPU.ActiveCfg = Release|Any CPU
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.Release|Any CPU.Build.0 = Release|Any CPU
{52EC37D6-088C-40D3-AD0B-BDE8F8DAF9EB}.Release|ARM64.ActiveCfg = Release|Any CPU
@@ -1616,14 +1612,14 @@ Global
{272B2B0E-40D4-4F0F-B187-519A6EF89B10}.Fuzzing|x64.Build.0 = Debug|Any CPU
{272B2B0E-40D4-4F0F-B187-519A6EF89B10}.Fuzzing|x86.ActiveCfg = Debug|Any CPU
{272B2B0E-40D4-4F0F-B187-519A6EF89B10}.Fuzzing|x86.Build.0 = Debug|Any CPU
- {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|Any CPU.ActiveCfg = Debug|Any CPU
- {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|Any CPU.Build.0 = Debug|Any CPU
- {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|ARM64.ActiveCfg = Debug|Any CPU
- {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|ARM64.Build.0 = Debug|Any CPU
- {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|x64.ActiveCfg = Debug|Any CPU
- {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|x64.Build.0 = Debug|Any CPU
- {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|x86.ActiveCfg = Debug|Any CPU
- {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|x86.Build.0 = Debug|Any CPU
+ {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|Any CPU.ActiveCfg = Release|Any CPU
+ {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|Any CPU.Build.0 = Release|Any CPU
+ {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|ARM64.ActiveCfg = Release|Any CPU
+ {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|ARM64.Build.0 = Release|Any CPU
+ {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|x64.ActiveCfg = Release|Any CPU
+ {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|x64.Build.0 = Release|Any CPU
+ {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|x86.ActiveCfg = Release|Any CPU
+ {272B2B0E-40D4-4F0F-B187-519A6EF89B10}.PowerShell|x86.Build.0 = Release|Any CPU
{272B2B0E-40D4-4F0F-B187-519A6EF89B10}.Release|Any CPU.ActiveCfg = Release|Any CPU
{272B2B0E-40D4-4F0F-B187-519A6EF89B10}.Release|Any CPU.Build.0 = Release|Any CPU
{272B2B0E-40D4-4F0F-B187-519A6EF89B10}.Release|ARM64.ActiveCfg = Release|Any CPU
diff --git a/src/AppInstallerCLICore/Workflows/DependenciesFlow.cpp b/src/AppInstallerCLICore/Workflows/DependenciesFlow.cpp
index 876b5ddaa5..2a92bb19ee 100644
--- a/src/AppInstallerCLICore/Workflows/DependenciesFlow.cpp
+++ b/src/AppInstallerCLICore/Workflows/DependenciesFlow.cpp
@@ -129,6 +129,11 @@ namespace AppInstaller::CLI::Workflow
void EnableWindowsFeaturesDependencies(Execution::Context& context)
{
+ if (!Settings::ExperimentalFeature::IsEnabled(Settings::ExperimentalFeature::Feature::WindowsFeature))
+ {
+ return;
+ }
+
const auto& rootDependencies = context.Get()->Dependencies;
if (rootDependencies.Empty())
diff --git a/src/AppInstallerCLICore/Workflows/DependencyNodeProcessor.cpp b/src/AppInstallerCLICore/Workflows/DependencyNodeProcessor.cpp
index 8749da1009..34a2668286 100644
--- a/src/AppInstallerCLICore/Workflows/DependencyNodeProcessor.cpp
+++ b/src/AppInstallerCLICore/Workflows/DependencyNodeProcessor.cpp
@@ -25,14 +25,14 @@ namespace AppInstaller::CLI::Workflow
if (matches.empty())
{
- error << Resource::String::DependenciesFlowNoMatches;
+ error << Resource::String::DependenciesFlowNoMatches << std::endl;
return DependencyNodeProcessorResult::Error;
}
if (matches.size() > 1)
{
auto dependencyNodeId = Utility::LocIndString{ Utility::Normalize(dependencyNode.Id()) };
- error << Resource::String::DependenciesFlowSourceTooManyMatches(dependencyNodeId);
+ error << Resource::String::DependenciesFlowSourceTooManyMatches(dependencyNodeId) << std::endl;
AICLI_LOG(CLI, Error, << "Too many matches for package " << dependencyNode.Id());
return DependencyNodeProcessorResult::Error;
}
@@ -60,17 +60,17 @@ namespace AppInstaller::CLI::Workflow
// as we won't keep searching for dependencies for installed packages
return DependencyNodeProcessorResult::Skipped;
}
-
+
if (!m_nodePackageLatestVersion)
{
- error << Resource::String::DependenciesFlowPackageVersionNotFound(Utility::LocIndView{ Utility::Normalize(packageId) });
+ error << Resource::String::DependenciesFlowPackageVersionNotFound(Utility::LocIndView{ Utility::Normalize(packageId) }) << std::endl;
AICLI_LOG(CLI, Error, << "Latest available version not found for package " << packageId);
return DependencyNodeProcessorResult::Error;
}
if (!dependencyNode.IsVersionOk(Utility::Version(m_nodePackageLatestVersion->GetProperty(PackageVersionProperty::Version))))
{
- error << Resource::String::DependenciesFlowNoMinVersion(Utility::LocIndView{ Utility::Normalize(packageId) });
+ error << Resource::String::DependenciesFlowNoMinVersion(Utility::LocIndView{ Utility::Normalize(packageId) }) << std::endl;
AICLI_LOG(CLI, Error, << "No suitable min version found for package " << packageId);
return DependencyNodeProcessorResult::Error;
}
@@ -80,7 +80,7 @@ namespace AppInstaller::CLI::Workflow
if (m_nodeManifest.Installers.empty())
{
- error << Resource::String::DependenciesFlowNoInstallerFound(Utility::LocIndView{ Utility::Normalize(m_nodeManifest.Id) });
+ error << Resource::String::DependenciesFlowNoInstallerFound(Utility::LocIndView{ Utility::Normalize(m_nodeManifest.Id) }) << std::endl;
AICLI_LOG(CLI, Error, << "Installer not found for manifest " << m_nodeManifest.Id << " with version" << m_nodeManifest.Version);
return DependencyNodeProcessorResult::Error;
}
@@ -98,7 +98,7 @@ namespace AppInstaller::CLI::Workflow
{
auto manifestId = Utility::LocIndString{ Utility::Normalize(m_nodeManifest.Id) };
auto manifestVersion = Utility::LocIndString{ m_nodeManifest.Version };
- error << Resource::String::DependenciesFlowNoSuitableInstallerFound(manifestId, manifestVersion);
+ error << Resource::String::DependenciesFlowNoSuitableInstallerFound(manifestId, manifestVersion) << std::endl;
AICLI_LOG(CLI, Error, << "No suitable installer found for manifest " << m_nodeManifest.Id << " with version " << m_nodeManifest.Version);
return DependencyNodeProcessorResult::Error;
}
diff --git a/src/AppInstallerCLIE2ETests/ConfigureCommand.cs b/src/AppInstallerCLIE2ETests/ConfigureCommand.cs
index 5b4fa2cc55..7153e511fb 100644
--- a/src/AppInstallerCLIE2ETests/ConfigureCommand.cs
+++ b/src/AppInstallerCLIE2ETests/ConfigureCommand.cs
@@ -154,6 +154,23 @@ public void ConfigServerUnexpectedExit()
FileAssert.DoesNotExist(targetFilePath);
}
+ ///
+ /// Resource name case insensitive test.
+ ///
+ [Test]
+ public void ResourceCaseInsensitive()
+ {
+ TestCommon.EnsureModuleState(Constants.SimpleTestModuleName, present: false);
+
+ var result = TestCommon.RunAICLICommand(CommandAndAgreementsAndVerbose, TestCommon.GetTestDataFile("Configuration\\ResourceCaseInsensitive.yml"));
+ Assert.AreEqual(0, result.ExitCode);
+
+ // The configuration creates a file next to itself with the given contents
+ string targetFilePath = TestCommon.GetTestDataFile("Configuration\\ResourceCaseInsensitive.txt");
+ FileAssert.Exists(targetFilePath);
+ Assert.AreEqual("Contents!", System.IO.File.ReadAllText(targetFilePath));
+ }
+
private void DeleteTxtFiles()
{
// Delete all .txt files in the test directory; they are placed there by the tests
diff --git a/src/AppInstallerCLIE2ETests/FeaturesCommand.cs b/src/AppInstallerCLIE2ETests/FeaturesCommand.cs
index ce6abf9b0a..04770c4690 100644
--- a/src/AppInstallerCLIE2ETests/FeaturesCommand.cs
+++ b/src/AppInstallerCLIE2ETests/FeaturesCommand.cs
@@ -53,6 +53,7 @@ public void EnableExperimentalFeatures()
WinGetSettingsHelper.ConfigureFeature("experimentalArg", true);
WinGetSettingsHelper.ConfigureFeature("experimentalCmd", true);
WinGetSettingsHelper.ConfigureFeature("directMSI", true);
+ WinGetSettingsHelper.ConfigureFeature("windowsFeature", true);
var result = TestCommon.RunAICLICommand("features", string.Empty);
Assert.True(result.StdOut.Contains("Enabled"));
}
diff --git a/src/AppInstallerCLIE2ETests/Helpers/TestCommon.cs b/src/AppInstallerCLIE2ETests/Helpers/TestCommon.cs
index 6f4fe24bfa..8746cae1af 100644
--- a/src/AppInstallerCLIE2ETests/Helpers/TestCommon.cs
+++ b/src/AppInstallerCLIE2ETests/Helpers/TestCommon.cs
@@ -916,7 +916,8 @@ public static void EnsureModuleState(string moduleName, bool present, string rep
else
{
string path = customPath;
- if (location == TestModuleLocation.WinGetModulePath)
+ if (location == TestModuleLocation.WinGetModulePath ||
+ location == TestModuleLocation.Default)
{
path = wingetModulePath;
}
diff --git a/src/AppInstallerCLIE2ETests/InstallCommand.cs b/src/AppInstallerCLIE2ETests/InstallCommand.cs
index 0a6927f844..186e8a7515 100644
--- a/src/AppInstallerCLIE2ETests/InstallCommand.cs
+++ b/src/AppInstallerCLIE2ETests/InstallCommand.cs
@@ -16,6 +16,15 @@ namespace AppInstallerCLIE2ETests
///
public class InstallCommand : BaseCommand
{
+ ///
+ /// One time setup.
+ ///
+ [OneTimeSetUp]
+ public void OneTimeSetup()
+ {
+ WinGetSettingsHelper.ConfigureFeature("windowsFeature", true);
+ }
+
///
/// Set up.
///
diff --git a/src/AppInstallerCLIE2ETests/TestData/Configuration/ResourceCaseInsensitive.yml b/src/AppInstallerCLIE2ETests/TestData/Configuration/ResourceCaseInsensitive.yml
new file mode 100644
index 0000000000..32cec1bade
--- /dev/null
+++ b/src/AppInstallerCLIE2ETests/TestData/Configuration/ResourceCaseInsensitive.yml
@@ -0,0 +1,9 @@
+properties:
+ configurationVersion: 0.2
+ resources:
+ - resource: xE2ETestResource/e2efileresource
+ directives:
+ repository: AppInstallerCLIE2ETestsRepo
+ settings:
+ Path: ${WinGetConfigRoot}\ResourceCaseInsensitive.txt
+ Content: Contents!
diff --git a/src/AppInstallerCLITests/WindowsFeature.cpp b/src/AppInstallerCLITests/WindowsFeature.cpp
index 4f4d1a308c..df84e23207 100644
--- a/src/AppInstallerCLITests/WindowsFeature.cpp
+++ b/src/AppInstallerCLITests/WindowsFeature.cpp
@@ -24,6 +24,9 @@ TEST_CASE("InstallFlow_WindowsFeatureDoesNotExist", "[windowsFeature]")
TestCommon::TempFile installResultPath("TestExeInstalled.txt");
+ TestCommon::TestUserSettings testSettings;
+ testSettings.Set(true);
+
std::ostringstream installOutput;
TestContext context{ installOutput, std::cin };
auto previousThreadGlobals = context.SetForCurrentThread();
@@ -57,6 +60,9 @@ TEST_CASE("InstallFlow_FailedToEnableWindowsFeature", "[windowsFeature]")
TestCommon::TempFile installResultPath("TestExeInstalled.txt");
+ TestCommon::TestUserSettings testSettings;
+ testSettings.Set(true);
+
std::ostringstream installOutput;
TestContext context{ installOutput, std::cin };
auto previousThreadGlobals = context.SetForCurrentThread();
@@ -92,6 +98,9 @@ TEST_CASE("InstallFlow_FailedToEnableWindowsFeature_Force", "[windowsFeature]")
TestCommon::TempFile installResultPath("TestExeInstalled.txt");
+ TestCommon::TestUserSettings testSettings;
+ testSettings.Set(true);
+
// Override with arbitrary DISM api error (DISMAPI_E_DISMAPI_NOT_INITIALIZED) and make windows feature discoverable.
HRESULT dismErrorResult = 0xc0040001;
LocIndString testFeatureDisplayName = LocIndString{ "Test Windows Feature"_liv };
@@ -139,6 +148,9 @@ TEST_CASE("InstallFlow_RebootRequired", "[windowsFeature]")
TestCommon::TempFile installResultPath("TestExeInstalled.txt");
+ TestCommon::TestUserSettings testSettings;
+ testSettings.Set(true);
+
// Override with reboot required HRESULT.
auto mockDismHelperOverride = TestHook::MockDismHelper_Override();
auto setEnableFeatureOverride = TestHook::SetEnableWindowsFeatureResult_Override(ERROR_SUCCESS_REBOOT_REQUIRED);
@@ -173,6 +185,9 @@ TEST_CASE("InstallFlow_RebootRequired_Force", "[windowsFeature]")
TestCommon::TempFile installResultPath("TestExeInstalled.txt");
+ TestCommon::TestUserSettings testSettings;
+ testSettings.Set(true);
+
// Override with reboot required HRESULT.
auto mockDismHelperOverride = TestHook::MockDismHelper_Override();
auto setEnableFeatureOverride = TestHook::SetEnableWindowsFeatureResult_Override(ERROR_SUCCESS_REBOOT_REQUIRED);
diff --git a/src/AppInstallerCommonCore/Downloader.cpp b/src/AppInstallerCommonCore/Downloader.cpp
index cb22039a53..dfff527c8a 100644
--- a/src/AppInstallerCommonCore/Downloader.cpp
+++ b/src/AppInstallerCommonCore/Downloader.cpp
@@ -1,6 +1,5 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
-
#include "pch.h"
#include "Public/AppInstallerErrors.h"
#include "Public/AppInstallerRuntime.h"
@@ -17,6 +16,9 @@
using namespace AppInstaller::Runtime;
using namespace AppInstaller::Settings;
using namespace AppInstaller::Filesystem;
+using namespace winrt::Windows::Web::Http;
+using namespace winrt::Windows::Web::Http::Headers;
+using namespace winrt::Windows::Web::Http::Filters;
namespace AppInstaller::Utility
{
@@ -141,6 +143,37 @@ namespace AppInstaller::Utility
return result;
}
+ std::map GetHeaders(std::string_view url)
+ {
+ AICLI_LOG(Core, Verbose, << "Retrieving headers from url: " << url);
+
+ HttpBaseProtocolFilter filter;
+ filter.CacheControl().ReadBehavior(HttpCacheReadBehavior::MostRecent);
+
+ HttpClient client(filter);
+ client.DefaultRequestHeaders().Connection().Clear();
+ client.DefaultRequestHeaders().Append(L"Connection", L"close");
+ client.DefaultRequestHeaders().UserAgent().ParseAdd(Utility::ConvertToUTF16(Runtime::GetDefaultUserAgent().get()));
+
+ winrt::Windows::Foundation::Uri uri{ Utility::ConvertToUTF16(url) };
+ HttpRequestMessage request(HttpMethod::Head(), uri);
+
+ HttpResponseMessage response = client.SendRequestAsync(request, HttpCompletionOption::ResponseHeadersRead).get();
+
+ THROW_HR_IF(
+ MAKE_HRESULT(SEVERITY_ERROR, FACILITY_HTTP, response.StatusCode()),
+ response.StatusCode() != HttpStatusCode::Ok);
+
+ std::map result;
+
+ for (const auto& header : response.Headers())
+ {
+ result.emplace(Utility::FoldCase(static_cast(Utility::ConvertToUTF8(header.Key()))), Utility::ConvertToUTF8(header.Value()));
+ }
+
+ return result;
+ }
+
std::optional> DownloadToStream(
const std::string& url,
std::ostream& dest,
diff --git a/src/AppInstallerCommonCore/ExperimentalFeature.cpp b/src/AppInstallerCommonCore/ExperimentalFeature.cpp
index 33766e9102..066c380d00 100644
--- a/src/AppInstallerCommonCore/ExperimentalFeature.cpp
+++ b/src/AppInstallerCommonCore/ExperimentalFeature.cpp
@@ -40,6 +40,8 @@ namespace AppInstaller::Settings
return userSettings.Get();
case ExperimentalFeature::Feature::DirectMSI:
return userSettings.Get();
+ case ExperimentalFeature::Feature::WindowsFeature:
+ return userSettings.Get();
default:
THROW_HR(E_UNEXPECTED);
}
@@ -69,6 +71,8 @@ namespace AppInstaller::Settings
return ExperimentalFeature{ "Argument Sample", "experimentalArg", "https://aka.ms/winget-settings", Feature::ExperimentalArg };
case Feature::DirectMSI:
return ExperimentalFeature{ "Direct MSI Installation", "directMSI", "https://aka.ms/winget-settings", Feature::DirectMSI };
+ case Feature::WindowsFeature:
+ return ExperimentalFeature{ "Windows Feature Dependencies", "windowsFeature", "https://aka.ms/winget-settings", Feature::WindowsFeature };
default:
THROW_HR(E_UNEXPECTED);
}
diff --git a/src/AppInstallerCommonCore/HttpStream/HttpClientWrapper.cpp b/src/AppInstallerCommonCore/HttpStream/HttpClientWrapper.cpp
index cb5ea7843b..3f5a536b8e 100644
--- a/src/AppInstallerCommonCore/HttpStream/HttpClientWrapper.cpp
+++ b/src/AppInstallerCommonCore/HttpStream/HttpClientWrapper.cpp
@@ -4,6 +4,7 @@
#include "pch.h"
#include "Public/AppInstallerStrings.h"
#include "HttpClientWrapper.h"
+#include "Public/AppInstallerRuntime.h"
using namespace winrt::Windows::Foundation;
using namespace winrt::Windows::Security::Cryptography;
@@ -32,6 +33,7 @@ namespace AppInstaller::Utility::HttpStream
instance->m_httpClient.DefaultRequestHeaders().Connection().Clear();
instance->m_httpClient.DefaultRequestHeaders().Append(L"Connection", L"Keep-Alive");
+ instance->m_httpClient.DefaultRequestHeaders().UserAgent().ParseAdd(Utility::ConvertToUTF16(Runtime::GetDefaultUserAgent().get()));
co_await instance->PopulateInfoAsync();
diff --git a/src/AppInstallerCommonCore/Public/AppInstallerDownloader.h b/src/AppInstallerCommonCore/Public/AppInstallerDownloader.h
index 9b96d18504..8434c6e981 100644
--- a/src/AppInstallerCommonCore/Public/AppInstallerDownloader.h
+++ b/src/AppInstallerCommonCore/Public/AppInstallerDownloader.h
@@ -7,6 +7,7 @@
#include
#include
+#include