¿Equivalente al comando * Nix 'which' en PowerShell?

405

¿Cómo le pregunto a PowerShell dónde está algo?

Por ejemplo, "qué bloc de notas" y devuelve el directorio desde donde se ejecuta notepad.exe de acuerdo con las rutas actuales.

Chris en desarrollo
fuente

Respuestas:

392

El primer alias que hice una vez que comencé a personalizar mi perfil en PowerShell fue 'which'.

New-Alias which get-command

Para agregar esto a su perfil, escriba esto:

"`nNew-Alias which get-command" | add-content $profile

La `n al comienzo de la última línea es asegurar que comenzará como una nueva línea.

halr9000
fuente
1
Puedes ponerlo en tu script de perfil. Más sobre perfiles - msdn.microsoft.com/en-us/library/bb613488(VS.85).aspx
Steven Murawski
6262
me gusta correr: Get-Command <command> | Format-Table Path, Nameasí puedo obtener la ruta donde también se encuentra el comando.
jrsconfitto
44
¿Hay alguna manera de tener la ruta todo el tiempo sin escribir '| Ruta de tabla de formato, nombre '?
Guillaume
10
Si desea el comportamiento estilo Unix de darle la ruta, necesitará canalizar la salida de get-command select -expandproperty Path.
Casey
55
Use (gcm <command>).definitionpara obtener solo las rutas. gcmes el alias predeterminado para Get-Command. También puede utilizar comodines, por ejemplo: (gcm win*.exe).definition.
Sachin Joseph
165

Aquí hay un equivalente real * nix, es decir, proporciona una salida de estilo * nix.

Get-Command <your command> | Select-Object -ExpandProperty Definition

Simplemente reemplácelo con lo que esté buscando.

PS C:\> Get-Command notepad.exe | Select-Object -ExpandProperty Definition
C:\Windows\system32\notepad.exe

Cuando lo agregue a su perfil, querrá usar una función en lugar de un alias porque no puede usar alias con tuberías:

function which($name)
{
    Get-Command $name | Select-Object -ExpandProperty Definition
}

Ahora, cuando vuelva a cargar su perfil, puede hacer esto:

PS C:\> which notepad
C:\Windows\system32\notepad.exe
petrsnd
fuente
22
Uso esta sintaxis alternativa: "(Bloc de notas Get-Command) .definition"
Yann
2
@ B00merang Su sintaxis es excelente, definitivamente más concisa, pero desafortunadamente, incluso con la tubería eliminada, no se puede agregar como un alias a menos que incluya el nombre del programa que está buscando.
petrsnd
44
Esta es una publicación antigua, pero en caso de que Google envíe a alguien aquí (como yo), esta respuesta funciona con más tipos de comandos Powershell que la respuesta aceptada. Por ejemplo, tengo un alias llamado oktaque apunta a un script de Powershell llamado okta.ps1que no está en mi $PATH. El uso de la respuesta aceptada devuelve el nombre del script ( okta -> okta.ps1). Esto está bien pero no me dice la ubicación de okta.ps1. Sin embargo, usar esta respuesta me da todo el camino ( C:\Users\blah\etc\scripts\okta.ps1). Entonces +1 de mi parte.
skye --- capitán
88

Usualmente solo escribo:

gcm notepad

o

gcm note*

gcm es el alias predeterminado para Get-Command.

En mi sistema, gcm note * salidas:

[27] » gcm note*

CommandType     Name                                                     Definition
-----------     ----                                                     ----------
Application     notepad.exe                                              C:\WINDOWS\notepad.exe
Application     notepad.exe                                              C:\WINDOWS\system32\notepad.exe
Application     Notepad2.exe                                             C:\Utils\Notepad2.exe
Application     Notepad2.ini                                             C:\Utils\Notepad2.ini

Obtiene el directorio y el comando que coincide con lo que está buscando.

David Mohundro
fuente
es un poco desordenado, pero mucho más limpio que las funciones personalizadas y las divisiones arbitrarias
DevelopingChris
1
Cuando escribo "gcm notepad" en mi símbolo del sistema de PowerShell, solo obtengo las dos primeras columnas y una tercera columna llamada 'ModuleName' que está vacía. ¿Sabes cómo forzarlo a enumerar la columna 'Definición' de forma predeterminada?
Piyush Soni
3
@PiyushSoni probablemente se deba a una versión actualizada de PowerShell. Siempre puede mostrar las otras columnas haciendo algo como gcm note* | select CommandType, Name, Definition. Sin embargo, si lo ejecuta a menudo, probablemente debería envolverlo en una función.
David Mohundro
40

Prueba este ejemplo:

(Get-Command notepad.exe).Path
thesqldev
fuente
2
Agregue más código o explicación para que el OP pueda entenderlo mejor. Gracias.
sshashank124
3
Gracias por agregar menos código para que pueda recordar esto por una vez: P
albertjan
1
¡Esto es lo que quería! También funciona con gcm:(gcm py.exe).path
Bill Agee
7

Mi propuesta para la función Which:

function which($cmd) { get-command $cmd | % { $_.Path } }

PS C:\> which devcon

C:\local\code\bin\devcon.exe
VortiFred
fuente
Esta es una mejor respuesta que la aceptada. Le permite agregar los sufijos de posprocesamiento sugeridos anteriormente para proporcionar una mejor salida; un alias no.
BobHy
5

Un partido rápido y sucio con Unix whiches

New-Alias which where.exe

Pero devuelve varias líneas si existen, entonces se convierte en

function which {where.exe command | select -first 1}
Chris F Carroll
fuente
1
where.exe wheredebería decirteC:\Windows\System32\where.exe
Chris F Carroll
1
where.exees equivalente a which -a, ya que devolverá todos los ejecutables coincidentes, no solo el primero que se ejecutará. Es decir, where.exe notepadda c:\windows\notepad.exey c:\windows\system32\notepad.exe. Así que esto es particularmente no adecuado para la forma $(which command). (Otro problema es que imprimirá un mensaje de error agradable y útil si no se encuentra el comando, que tampoco se expandirá bien $(), eso se puede remediar /Q, pero no como un alias.)
Jeroen Mostert
punto a favor. Edité la respuesta, pero sí, ya no es una solución tan ordenada
Chris F Carroll
1
Tenga en cuenta que whereparece buscar la variable PATH del sistema y no la variable PATH del shell actual. Ver esta pregunta
Leonardo
3

Esto parece hacer lo que quieres (lo encontré en http://huddledmasses.org/powershell-find-path/ ):

Function Find-Path($Path, [switch]$All = $false, [Microsoft.PowerShell.Commands.TestPathType]$type = "Any")
## You could comment out the function stuff and use it as a script instead, with this line:
#param($Path, [switch]$All = $false, [Microsoft.PowerShell.Commands.TestPathType]$type = "Any")
   if($(Test-Path $Path -Type $type)) {
      return $path
   } else {
      [string[]]$paths = @($pwd);
      $paths += "$pwd;$env:path".split(";")

      $paths = Join-Path $paths $(Split-Path $Path -leaf) | ? { Test-Path $_ -Type $type }
      if($paths.Length -gt 0) {
         if($All) {
            return $paths;
         } else {
            return $paths[0]
         }
      }
   }
   throw "Couldn't find a matching path of type $type"
}
Set-Alias find Find-Path
Nicholas
fuente
Pero no es realmente "cuál", ya que funciona con cualquier archivo (tipo) y no encuentra cmdlets, funciones o alias
Jaykul
3

Comprueba este PowerShell que .

El código provisto allí sugiere esto:

($Env:Path).Split(";") | Get-ChildItem -filter notepad.exe
tzot
fuente
2
Sé que han pasado años, pero mi ruta tenía "% systemroot% \ system32 \ ..." y PowerShell no expande esa variable de entorno y arroja errores al hacer esto.
TessellatingHeckler
3

Me gusta Get-Command | Format-List, o más corto, usar alias para los dos y solo para powershell.exe:

gcm powershell | fl

Puedes encontrar alias como este:

alias -definition Format-List

La finalización de pestañas funciona con gcm.

js2010
fuente
2

Pruebe el wherecomando en Windows 2003 o posterior (o Windows 2000 / XP si ha instalado un Kit de recursos).

Por cierto, esto recibió más respuestas en otras preguntas:

¿Hay un equivalente de 'cual' en Windows?

¿PowerShell equivalente al whichcomando Unix ?

Anónimo
fuente
44
wherealias al Where-Objectcomando en Powershell, por lo que escribir where <item>en un indicador de Powershell no produce nada. Por lo tanto, esta respuesta es completamente incorrecta: como se señaló en la respuesta aceptada en la primera pregunta vinculada, para obtener el DOS where, debe escribir where.exe <item>.
Ian Kemp
0

Tengo esta whichfunción avanzada en mi perfil de PowerShell:

function which {
<#
.SYNOPSIS
Identifies the source of a PowerShell command.
.DESCRIPTION
Identifies the source of a PowerShell command. External commands (Applications) are identified by the path to the executable
(which must be in the system PATH); cmdlets and functions are identified as such and the name of the module they are defined in
provided; aliases are expanded and the source of the alias definition is returned.
.INPUTS
No inputs; you cannot pipe data to this function.
.OUTPUTS
.PARAMETER Name
The name of the command to be identified.
.EXAMPLE
PS C:\Users\Smith\Documents> which Get-Command

Get-Command: Cmdlet in module Microsoft.PowerShell.Core

(Identifies type and source of command)
.EXAMPLE
PS C:\Users\Smith\Documents> which notepad

C:\WINDOWS\SYSTEM32\notepad.exe

(Indicates the full path of the executable)
#>
    param(
    [String]$name
    )

    $cmd = Get-Command $name
    $redirect = $null
    switch ($cmd.CommandType) {
        "Alias"          { "{0}: Alias for ({1})" -f $cmd.Name, (. { which cmd.Definition } ) }
        "Application"    { $cmd.Source }
        "Cmdlet"         { "{0}: {1} {2}" -f $cmd.Name, $cmd.CommandType, (. { if ($cmd.Source.Length) { "in module {0}" -f $cmd.Source} else { "from unspecified source" } } ) }
        "Function"       { "{0}: {1} {2}" -f $cmd.Name, $cmd.CommandType, (. { if ($cmd.Source.Length) { "in module {0}" -f $cmd.Source} else { "from unspecified source" } } ) }
        "Workflow"       { "{0}: {1} {2}" -f $cmd.Name, $cmd.CommandType, (. { if ($cmd.Source.Length) { "in module {0}" -f $cmd.Source} else { "from unspecified source" } } ) }
        "ExternalScript" { $cmd.Source }
        default          { $cmd }
    }
}
Jeff Zeitlin
fuente
0

Utilizar:

function Which([string] $cmd) {
  $path = (($Env:Path).Split(";") | Select -uniq | Where { $_.Length } | Where { Test-Path $_ } | Get-ChildItem -filter $cmd).FullName
  if ($path) { $path.ToString() }
}

# Check if Chocolatey is installed
if (Which('cinst.bat')) {
  Write-Host "yes"
} else {
  Write-Host "no"
}

O esta versión, llamando al comando where original.

Esta versión también funciona mejor, porque no se limita a los archivos bat:

function which([string] $cmd) {
  $where = iex $(Join-Path $env:SystemRoot "System32\where.exe $cmd 2>&1")
  $first = $($where -split '[\r\n]')
  if ($first.getType().BaseType.Name -eq 'Array') {
    $first = $first[0]
  }
  if (Test-Path $first) {
    $first
  }
}

# Check if Curl is installed
if (which('curl')) {
  echo 'yes'
} else {
  echo 'no'
}
Jerome
fuente
0

Si desea un comando que acepte la entrada de la tubería o como parámetro, debe intentar esto:

function which($name) {
    if ($name) { $input = $name }
    Get-Command $input | Select-Object -ExpandProperty Path
}

copie y pegue el comando en su perfil ( notepad $profile).

Ejemplos:

 echo clang.exe | which
C:\Program Files\LLVM\bin\clang.exe

 which clang.exe
C:\Program Files\LLVM\bin\clang.exe
Amin
fuente