Deserialization
Informations de base
La sérialisation est comprise comme la méthode de conversion d'un objet dans un format pouvant être préservé, dans le but de stocker l'objet ou de le transmettre dans le cadre d'un processus de communication. Cette technique est couramment utilisée pour garantir que l'objet puisse être recréé ultérieurement, en conservant sa structure et son état.
La désérialisation, en revanche, est le processus qui contrecarre la sérialisation. Il consiste à prendre des données structurées dans un format spécifique et à les reconstruire en un objet.
La désérialisation peut être dangereuse car elle permet potentiellement aux attaquants de manipuler les données sérialisées pour exécuter un code malveillant ou provoquer un comportement inattendu dans l'application lors du processus de reconstruction de l'objet.
PHP
En PHP, des méthodes magiques spécifiques sont utilisées lors des processus de sérialisation et de désérialisation :
__sleep
: Appelée lorsqu'un objet est sérialisé. Cette méthode doit renvoyer un tableau contenant les noms de toutes les propriétés de l'objet qui doivent être sérialisées. Elle est couramment utilisée pour valider les données en attente ou effectuer des tâches de nettoyage similaires.__wakeup
: Appelée lorsqu'un objet est désérialisé. Elle est utilisée pour rétablir les connexions à la base de données qui auraient pu être perdues lors de la sérialisation et effectuer d'autres tâches de réinitialisation.__unserialize
: Cette méthode est appelée à la place de__wakeup
(si elle existe) lorsqu'un objet est désérialisé. Elle offre plus de contrôle sur le processus de désérialisation par rapport à__wakeup
.__destruct
: Cette méthode est appelée lorsqu'un objet est sur le point d'être détruit ou lorsque le script se termine. Elle est généralement utilisée pour des tâches de nettoyage, comme la fermeture des gestionnaires de fichiers ou des connexions à la base de données.__toString
: Cette méthode permet à un objet d'être traité comme une chaîne de caractères. Elle peut être utilisée pour lire un fichier ou d'autres tâches basées sur les appels de fonctions à l'intérieur, fournissant ainsi une représentation textuelle de l'objet.
Si vous regardez les résultats, vous pouvez voir que les fonctions __wakeup
et __destruct
sont appelées lorsque l'objet est désérialisé. Notez que dans plusieurs tutoriels, vous trouverez que la fonction __toString
est appelée lors de la tentative d'impression d'un attribut, mais apparemment cela n'arrive plus.
La méthode __unserialize(array $data)
est appelée au lieu de __wakeup()
si elle est implémentée dans la classe. Cela vous permet de désérialiser l'objet en fournissant les données sérialisées sous forme de tableau. Vous pouvez utiliser cette méthode pour désérialiser les propriétés et effectuer les tâches nécessaires lors de la désérialisation.
Vous pouvez lire un exemple PHP expliqué ici : https://www.notsosecure.com/remote-code-execution-via-php-unserialize/, ici https://www.exploit-db.com/docs/english/44756-deserialization-vulnerability.pdf ou ici https://securitycafe.ro/2015/01/05/understanding-php-object-injection/
PHP Deserial + Autoload Classes
Vous pourriez abuser de la fonctionnalité d'autoload PHP pour charger des fichiers PHP arbitraires et plus encore :
PHP - Deserialization + Autoload ClassesSérialisation de valeurs référencées
Si pour une raison quelconque vous souhaitez sérialiser une valeur comme une référence à une autre valeur sérialisée, vous pouvez :
PHPGGC (ysoserial pour PHP)
PHPGGC peut vous aider à générer des charges utiles pour abuser des désérialisations PHP.
Notez que dans plusieurs cas, vous ne pourrez pas trouver de moyen d'abuser d'une désérialisation dans le code source de l'application mais vous pourriez être en mesure d'abuser du code des extensions PHP externes.
Donc, si possible, vérifiez le phpinfo()
du serveur et recherchez sur internet (et même sur les gadgets de PHPGGC) des gadgets possibles que vous pourriez exploiter.
Désérialisation des métadonnées phar://
Si vous avez trouvé une LFI qui se contente de lire le fichier et de ne pas exécuter le code PHP à l'intérieur, par exemple en utilisant des fonctions telles que file_get_contents(), fopen(), file() ou file_exists(), md5_file(), filemtime() ou filesize(). Vous pouvez essayer d'abuser d'une désérialisation se produisant lors de la lecture d'un fichier en utilisant le protocole phar. Pour plus d'informations, lisez l'article suivant:
phar:// deserializationPython
Pickle
Lorsque l'objet est dépicklé, la fonction __reduce__ sera exécutée. Lorsqu'exploité, le serveur pourrait renvoyer une erreur.
Pour plus d'informations sur l'évasion des prisons de pickle, consultez :
Bypass Python sandboxesYaml & jsonpickle
La page suivante présente la technique pour abuser d'une désérialisation non sécurisée dans les bibliothèques python yamls et se termine par un outil qui peut être utilisé pour générer une charge utile de désérialisation RCE pour Pickle, PyYAML, jsonpickle et ruamel.yaml :
Python Yaml DeserializationPollution de classe (Pollution de prototype Python)
Class Pollution (Python's Prototype Pollution)NodeJS
Fonctions magiques JS
JS n'a pas de fonctions "magiques" comme PHP ou Python qui vont être exécutées juste pour créer un objet. Mais il a certaines fonctions qui sont fréquemment utilisées même sans les appeler directement telles que toString
, valueOf
, toJSON
.
En abusant d'une désérialisation, vous pouvez compromettre ces fonctions pour exécuter un autre code (potentiellement en abusant des pollutions de prototype) vous pourriez exécuter du code arbitraire lorsqu'elles sont appelées.
Une autre manière "magique" d'appeler une fonction sans l'appeler directement est en compromettant un objet retourné par une fonction asynchrone (promesse). Parce que, si vous transformez cet objet retourné en une autre promesse avec une propriété appelée "then" de type fonction, elle sera exécutée simplement parce qu'elle est retournée par une autre promesse. Suivez ce lien pour plus d'informations.
Pollution de __proto__
et prototype
__proto__
et prototype
Si vous souhaitez en savoir plus sur cette technique, consultez le tutoriel suivant:
NodeJS - __proto__ & prototype PollutionCette bibliothèque permet de sérialiser des fonctions. Exemple:
L'objet sérialisé ressemblera à :
Vous pouvez voir dans l'exemple que lorsqu'une fonction est sérialisée, le drapeau _$$ND_FUNC$$_
est ajouté à l'objet sérialisé.
À l'intérieur du fichier node-serialize/lib/serialize.js
, vous pouvez trouver le même drapeau et comment le code l'utilise.
Comme vous pouvez le voir dans le dernier morceau de code, si le drapeau est trouvé, eval
est utilisé pour désérialiser la fonction, donc en gros l'entrée utilisateur est utilisée à l'intérieur de la fonction eval
.
Cependant, simplement sérialiser une fonction ne l'exécutera pas, car il serait nécessaire qu'une partie du code appelle y.rce
dans notre exemple, ce qui est très improbable.
Quoi qu'il en soit, vous pourriez simplement modifier l'objet sérialisé en ajoutant des parenthèses afin d'exécuter automatiquement la fonction sérialisée lorsque l'objet est désérialisé.
Dans le prochain morceau de code, remarquez la dernière parenthèse et comment la fonction unserialize
exécutera automatiquement le code:
Comme indiqué précédemment, cette bibliothèque récupérera le code après _$$ND_FUNC$$_
et l'exécutera en utilisant eval
. Par conséquent, pour auto-exécuter du code, vous pouvez supprimer la création de la fonction et la dernière parenthèse et simplement exécuter un JS oneliner comme dans l'exemple suivant:
Vous pouvez trouver ici plus d'informations sur la manière d'exploiter cette vulnérabilité.
Un aspect notable de funcster est l'inaccessibilité des objets intégrés standard; ils se trouvent en dehors de la portée accessible. Cette restriction empêche l'exécution de code qui tente d'invoquer des méthodes sur des objets intégrés, entraînant des exceptions telles que "ReferenceError: console is not defined"
lorsque des commandes comme console.log()
ou require(something)
sont utilisées.
Malgré cette limitation, la restauration d'un accès complet au contexte global, y compris à tous les objets intégrés standard, est possible grâce à une approche spécifique. En exploitant directement le contexte global, il est possible de contourner cette restriction. Par exemple, l'accès peut être rétabli en utilisant l'extrait suivant:
Pour plus d'informations, consultez cette source.
Le package serialize-javascript est conçu exclusivement à des fins de sérialisation, sans aucune capacité de désérialisation intégrée. Les utilisateurs sont responsables de mettre en œuvre leur propre méthode de désérialisation. Une utilisation directe de eval
est suggérée par l'exemple officiel pour désérialiser des données sérialisées:
Si cette fonction est utilisée pour désérialiser des objets, vous pouvez facilement l'exploiter :
Pour plus d'informations, consultez cette source.
Bibliothèque Cryo
Sur les pages suivantes, vous trouverez des informations sur la manière d'abuser de cette bibliothèque pour exécuter des commandes arbitraires :
Java - HTTP
En Java, les rappels de désérialisation sont exécutés pendant le processus de désérialisation. Cette exécution peut être exploitée par des attaquants qui créent des charges malveillantes déclenchant ces rappels, pouvant entraîner l'exécution potentielle d'actions nuisibles.
Empreintes digitales
Boîte Blanche
Pour identifier les vulnérabilités potentielles de sérialisation dans le code source, recherchez :
Les classes qui implémentent l'interface
Serializable
.L'utilisation des fonctions
java.io.ObjectInputStream
,readObject
,readUnshare
.
Portez une attention particulière à :
XMLDecoder
utilisé avec des paramètres définis par des utilisateurs externes.La méthode
fromXML
deXStream
, en particulier si la version de XStream est inférieure ou égale à 1.46, car elle est sujette à des problèmes de sérialisation.ObjectInputStream
associé à la méthodereadObject
.L'implémentation de méthodes telles que
readObject
,readObjectNodData
,readResolve
oureadExternal
.ObjectInputStream.readUnshared
.L'utilisation générale de
Serializable
.
Boîte Noire
Pour les tests en boîte noire, recherchez des signatures ou "Magic Bytes" spécifiques indiquant des objets sérialisés Java (provenant de ObjectInputStream
) :
Motif hexadécimal :
AC ED 00 05
.Motif Base64 :
rO0
.En-têtes de réponse HTTP avec
Content-type
défini surapplication/x-java-serialized-object
.Motif hexadécimal indiquant une compression préalable :
1F 8B 08 00
.Motif Base64 indiquant une compression préalable :
H4sIA
.Fichiers Web avec l'extension
.faces
et le paramètrefaces.ViewState
. Découvrir ces motifs dans une application Web devrait inciter à un examen détaillé comme décrit dans le post sur la désérialisation de Java JSF ViewState.
Vérifier si vulnérable
Si vous voulez apprendre comment fonctionne une exploitation de désérialisation Java, vous devriez jeter un œil à Désérialisation Java de base, Désérialisation Java DNS, et Charge utile CommonsCollection1.
Test de la boîte blanche
Vous pouvez vérifier s'il y a une application installée avec des vulnérabilités connues.
Vous pourriez essayer de vérifier toutes les bibliothèques connues pour être vulnérables et pour lesquelles Ysoserial peut fournir une exploitation. Ou vous pourriez vérifier les bibliothèques indiquées sur Java-Deserialization-Cheat-Sheet. Vous pourriez également utiliser gadgetinspector pour rechercher des chaînes de gadgets potentielles qui peuvent être exploitées. Lors de l'exécution de gadgetinspector (après l'avoir construit), ne vous souciez pas des tonnes d'avertissements/erreurs qu'il traverse et laissez-le terminer. Il écrira toutes les découvertes sous gadgetinspector/gadget-results/gadget-chains-year-month-day-hore-min.txt. Veuillez noter que gadgetinspector ne créera pas d'exploit et il peut indiquer des faux positifs.
Test en boîte noire
En utilisant l'extension Burp gadgetprobe, vous pouvez identifier quelles bibliothèques sont disponibles (et même les versions). Avec ces informations, il pourrait être plus facile de choisir une charge utile pour exploiter la vulnérabilité.
Lisez ceci pour en savoir plus sur GadgetProbe.
GadgetProbe est axé sur les désérialisations de ObjectInputStream
.
En utilisant l'extension Burp Java Deserialization Scanner, vous pouvez identifier les bibliothèques vulnérables exploitables avec ysoserial et les exploiter.
Lisez ceci pour en savoir plus sur Java Deserialization Scanner.
Java Deserialization Scanner est axé sur les désérialisations de ObjectInputStream
.
Vous pouvez également utiliser Freddy pour détecter les vulnérabilités de désérialisation dans Burp. Ce plugin détectera les vulnérabilités non seulement liées à ObjectInputStream
mais aussi aux vulnérabilités des bibliothèques de désérialisation Json et Yml. En mode actif, il essaiera de les confirmer en utilisant des charges utiles de sommeil ou de DNS.
Vous pouvez trouver plus d'informations sur Freddy ici.
Test de sérialisation
Tout ne consiste pas à vérifier si une bibliothèque vulnérable est utilisée par le serveur. Parfois, vous pourriez être en mesure de modifier les données à l'intérieur de l'objet sérialisé et contourner certaines vérifications (peut-être vous accorder des privilèges administratifs à l'intérieur d'une application web). Si vous trouvez un objet sérialisé Java envoyé à une application web, vous pouvez utiliser SerializationDumper pour afficher dans un format plus lisible par l'homme l'objet de sérialisation qui est envoyé. Savoir quelles données vous envoyez faciliterait leur modification et le contournement de certaines vérifications.
Exploitation
ysoserial
L'outil principal pour exploiter les désérialisations Java est ysoserial (téléchargez ici). Vous pouvez également envisager d'utiliser ysoseral-modified qui vous permettra d'utiliser des commandes complexes (avec des tubes par exemple).
Notez que cet outil est axé sur l'exploitation de ObjectInputStream
.
Je commencerais par utiliser la charge utile "URLDNS" avant une charge utile RCE pour tester si l'injection est possible. Quoi qu'il en soit, notez que peut-être la charge utile "URLDNS" ne fonctionne pas mais une autre charge utile RCE oui.
Lors de la création d'une charge utile pour java.lang.Runtime.exec(), vous ne pouvez pas utiliser de caractères spéciaux tels que ">" ou "|" pour rediriger la sortie d'une exécution, "$()" pour exécuter des commandes ou même passer des arguments à une commande séparés par des espaces (vous pouvez faire echo -n "hello world"
mais vous ne pouvez pas faire python2 -c 'print "Hello world"'
). Pour encoder correctement la charge utile, vous pouvez utiliser cette page web.
N'hésitez pas à utiliser le script suivant pour créer toutes les charges utiles d'exécution de code possibles pour Windows et Linux, puis les tester sur la page web vulnérable :
serialkillerbypassgadgets
Vous pouvez utiliser https://github.com/pwntester/SerialKillerBypassGadgetCollection avec ysoserial pour créer plus d'exploits. Plus d'informations sur cet outil dans les diapositives de la présentation où l'outil a été présenté : https://es.slideshare.net/codewhitesec/java-deserialization-vulnerabilities-the-forgotten-bug-class?next_slideshow=1
marshalsec
marshalsec peut être utilisé pour générer des charges utiles pour exploiter différentes bibliothèques de sérialisation Json et Yml en Java.
Pour compiler le projet, j'ai dû ajouter ces dépendances à pom.xml
:
Installer maven, et compiler le projet:
FastJSON
En savoir plus sur cette bibliothèque JSON Java : https://www.alphabot.com/security/blog/2020/java/Fastjson-exceptional-deserialization-vulnerabilities.html
Laboratoires
Si vous souhaitez tester certains payloads ysoserial, vous pouvez exécuter cette application web : https://github.com/hvqzao/java-deserialize-webapp
Pourquoi
Java utilise beaucoup la sérialisation à des fins diverses telles que :
Requêtes HTTP : La sérialisation est largement utilisée dans la gestion des paramètres, ViewState, cookies, etc.
RMI (Invocation de Méthode à Distance) : Le protocole Java RMI, qui repose entièrement sur la sérialisation, est un pilier de la communication à distance dans les applications Java.
RMI sur HTTP : Cette méthode est couramment utilisée par les applications web clientes épaisses basées sur Java, utilisant la sérialisation pour toutes les communications d'objets.
JMX (Extensions de Gestion Java) : JMX utilise la sérialisation pour transmettre des objets sur le réseau.
Protocoles Personnalisés : En Java, la pratique standard implique la transmission d'objets Java bruts, comme le démontreront les exemples d'exploitation à venir.
Prévention
Objets Transitoires
Une classe qui implémente Serializable
peut déclarer comme transient
tout objet à l'intérieur de la classe qui ne devrait pas être sérialisé. Par exemple :
Éviter la sérialisation d'une classe qui doit implémenter Serializable
Dans les scénarios où certains objets doivent implémenter l'interface Serializable
en raison de la hiérarchie de classes, il existe un risque de désérialisation involontaire. Pour éviter cela, assurez-vous que ces objets ne sont pas désérialisables en définissant une méthode final
readObject()
qui lance systématiquement une exception, comme indiqué ci-dessous:
Améliorer la sécurité de la désérialisation en Java
Personnaliser java.io.ObjectInputStream
est une approche pratique pour sécuriser les processus de désérialisation. Cette méthode est adaptée lorsque :
Le code de désérialisation est sous votre contrôle.
Les classes attendues pour la désérialisation sont connues.
Remplacez la méthode resolveClass()
pour limiter la désérialisation aux seules classes autorisées. Cela empêche la désérialisation de toute classe sauf celles explicitement autorisées, comme dans l'exemple suivant qui restreint la désérialisation à la classe Bicycle
uniquement :
Utilisation d'un agent Java pour l'amélioration de la sécurité offre une solution de secours lorsque la modification du code n'est pas possible. Cette méthode s'applique principalement pour mettre sur liste noire les classes nuisibles, en utilisant un paramètre JVM :
Il offre un moyen de sécuriser la désérialisation de manière dynamique, idéal pour les environnements où les modifications de code immédiates sont impraticables.
Consultez un exemple dans rO0 by Contrast Security
Mise en œuvre des filtres de sérialisation: Java 9 a introduit des filtres de sérialisation via l'interface ObjectInputFilter
, offrant un mécanisme puissant pour spécifier les critères que les objets sérialisés doivent respecter avant d'être désérialisés. Ces filtres peuvent être appliqués globalement ou par flux, offrant un contrôle granulaire sur le processus de désérialisation.
Pour utiliser les filtres de sérialisation, vous pouvez définir un filtre global qui s'applique à toutes les opérations de désérialisation ou le configurer dynamiquement pour des flux spécifiques. Par exemple:
Exploiter les bibliothèques externes pour une sécurité renforcée : Des bibliothèques telles que NotSoSerial, jdeserialize et Kryo offrent des fonctionnalités avancées pour contrôler et surveiller la désérialisation Java. Ces bibliothèques peuvent fournir des couches de sécurité supplémentaires, telles que la liste blanche ou noire des classes, l'analyse des objets sérialisés avant la désérialisation et la mise en œuvre de stratégies de sérialisation personnalisées.
NotSoSerial intercepte les processus de désérialisation pour empêcher l'exécution de code non fiable.
jdeserialize permet l'analyse des objets Java sérialisés sans les désérialiser, aidant à identifier un contenu potentiellement malveillant.
Kryo est un framework de sérialisation alternatif qui met l'accent sur la vitesse et l'efficacité, offrant des stratégies de sérialisation configurables qui peuvent renforcer la sécurité.
Références
Présentation sur la désérialisation et ysoserial : http://frohoff.github.io/appseccali-marshalling-pickles/
Présentation sur gadgetinspector : https://www.youtube.com/watch?v=wPbW6zQ52w8 et diapositives : https://i.blackhat.com/us-18/Thu-August-9/us-18-Haken-Automated-Discovery-of-Deserialization-Gadget-Chains.pdf
Document Marshalsec : https://www.github.com/mbechler/marshalsec/blob/master/marshalsec.pdf?raw=true
Document sur la désérialisation CVEs : https://paper.seebug.org/123/
Injection JNDI & log4Shell
Découvrez ce qu'est l'injection JNDI, comment l'exploiter via RMI, CORBA & LDAP et comment exploiter log4shell (et un exemple de cette vulnérabilité) sur la page suivante :
JNDI - Java Naming and Directory Interface & Log4ShellJMS - Java Message Service
L'API Java Message Service (JMS) est une API middleware orientée messages en Java pour l'envoi de messages entre deux clients ou plus. Il s'agit d'une implémentation pour résoudre le problème producteur-consommateur. JMS fait partie de la plateforme Java Enterprise Edition (Java EE) et a été défini par une spécification développée chez Sun Microsystems, mais qui est depuis guidée par le Java Community Process. C'est une norme de messagerie qui permet aux composants d'application basés sur Java EE de créer, envoyer, recevoir et lire des messages. Il permet à la communication entre les différents composants d'une application distribuée d'être faiblement couplée, fiable et asynchrone. (De Wikipedia).
Produits
Plusieurs produits utilisent ce middleware pour envoyer des messages :
Exploitation
En gros, il y a une série de services utilisant JMS de manière dangereuse. Par conséquent, si vous avez suffisamment de privilèges pour envoyer des messages à ces services (généralement, vous aurez besoin d'identifiants valides), vous pourriez être en mesure d'envoyer des objets malveillants sérialisés qui seront désérialisés par le consommateur/abonné. Cela signifie que dans cette exploitation, tous les clients qui vont utiliser ce message seront infectés.
N'oubliez pas que même si un service est vulnérable (car il désérialise de manière non sécurisée les entrées utilisateur), vous devez toujours trouver des gadgets valides pour exploiter la vulnérabilité.
L'outil JMET a été créé pour se connecter et attaquer ces services en envoyant plusieurs objets malveillants sérialisés en utilisant des gadgets connus. Ces exploits fonctionneront si le service est toujours vulnérable et si l'un des gadgets utilisés est présent dans l'application vulnérable.
Références
Présentation JMET : https://www.youtube.com/watch?v=0h8DWiOWGGA
.Net
Dans le contexte de .Net, les exploits de désérialisation fonctionnent de manière similaire à ceux trouvés en Java, où des gadgets sont exploités pour exécuter un code spécifique lors de la désérialisation d'un objet.
Empreinte digitale
Boîte blanche
Le code source doit être inspecté pour rechercher les occurrences de :
TypeNameHandling
JavaScriptTypeResolver
L'accent devrait être mis sur les sérialiseurs qui permettent de déterminer le type par une variable sous contrôle utilisateur.
Boîte noire
La recherche devrait cibler la chaîne encodée en Base64 AAEAAAD///// ou tout motif similaire qui pourrait subir une désérialisation côté serveur, accordant le contrôle sur le type à désérialiser. Cela pourrait inclure, mais sans s'y limiter, des structures JSON ou XML présentant TypeObject
ou $type
.
ysoserial.net
Dans ce cas, vous pouvez utiliser l'outil ysoserial.net pour créer des exploits de désérialisation. Une fois le dépôt git téléchargé, vous devriez compiler l'outil en utilisant Visual Studio par exemple.
Si vous souhaitez en savoir plus sur comment ysoserial.net crée ses exploits, vous pouvez consulter cette page où est expliqué le gadget ObjectDataProvider + ExpandedWrapper + formateur Json.Net.
Les principales options de ysoserial.net sont : --gadget
, --formatter
, --output
et --plugin
.
--gadget
utilisé pour indiquer le gadget à exploiter (indiquer la classe/fonction qui sera exploitée lors de la désérialisation pour exécuter des commandes).