La patate chaude
Dépôt GitHub : https://github.com/Nouuu/la-patate-chaude
Flash news: la FAQ
Vous connaissez peut-être l'expression ou le jeu de la patate chaude.
En quelques mots : comment se débarrasser d'un objet gênant avant qu'il ne soit trop tard !
Le projet est ici d'élaborer une version numérique de ce jeu où l'apothéose sera quand tous vos projets joueront ensemble une grande partie sur le réseau pour déterminer qui est le plus fiable, le plus rapide ou le plus malin.
Le jeu est composé de:
-
un ou plusieurs agents "joueurs" (que nous nommerons aussi "client" pour coller au vocabulaire usuel au niveau technique)
-
un agent "arbitre" (que nous nommerons aussi "serveur")
La partie se déroule ainsi :
-
Le serveur démarre ; après initialisation, il écoute sur un port (7878 par défaut).
-
Les différents clients démarrent alors et entrent en relation avec le serveur suivant un protocole décrit plus loin.
-
Une fois que les clients se sont connectés, une commande sur le serveur scelle le groupe de participants et le serveur n'accepte plus de nouveaux entrants. Le jeu peut démarrer.
-
La partie se joue en plusieurs manches (aussi nommées round) dont le nombre est défini par la configuration du serveur.
Pour chaque manche :
-
Le serveur envoie à tous les joueurs actifs, un tableau général des scores de tous les joueurs.
-
Le serveur détermine (secrètement) le temps alloué pour le prochain round. Ce timer décompte uniquement le temps entre l'envoi (par le serveur) du challenge à un joueur et sa réponse.
-
Jusqu'à ce qu'un joueur perde :
(i.e. ne résout pas son challenge dans le délai imparti ou bien la durée du round est expirée)
-
Le serveur sélectionne le prochain joueur :
- tel que désigné par le précédent joueur
- ou bien choisi au hasard parmi les joueurs encore dans la partie (si par exemple le joueur désigné n'est plus actif ou bien si c'est le premier tour).
- Le serveur se réserve le droit d'y ajouter des règles d'exclusion pour éviter des comportements extrêmes (tout en restant équitable entre les joueurs dans la partie)
-
Le serveur envoie au joueur un challenge à résoudre.
-
Le serveur attend la réponse du joueur jusqu'au timeout (défini dans la configuration du server)
- Si le timer du round expire pendant la résolution du challenge, le joueur perd 1 point.
- Si le joueur répond correctement au challenge, alors la patate est passée au joueur qu'il désigne
- Si le joueur ne répond pas correctement, mais avant le timeout, il perd un point et reste dans la partie. Le round s'arrête.
- Si le joueur ne répond pas avant le timeout ou bien qu'il ne répond à la sollicitation de challenge, il est exclu des joueurs actifs jusqu'à la fin de la partie. Le round s'arrête.
-
-
Une fois le round terminé (Quelle que soit la raison), le serveur envoi un résumé du round avec la liste des joueurs ayant participé (dans l'ordre des joueurs ayant participé au round) avec chacun
- son nom
- son résultat
- s'il a réussi (avec sa durée d'exécution et sa prochaine cible)
- ou bien s'il n'a pas réussi (avec sa durée d'exécution et sa prochaine cible)
- ou bien s'il a été timeout
- ou bien s'il n'est plus accessible (et donc exclu des joueurs actifs)
-
En fin de partie, le ou les vainqueurs sont identifiés. Il existe deux manières de scorer:
- avoir le plus de points
- avoir résolu le plus de challenges
Les challenges
C'est ici que l'ingéniosité algorithmique et l'efficacité dans la mise en œuvre vont être décisifs.
Tout challenge doit respecter l'interface imposée par le trait
suivant:
trait Challenge {
/// Données en entrée du challenge
type Input;
/// Données en sortie du challenge
type Output;
/// Nom du challenge
fn name() -> String;
/// Create a challenge from the specific input
fn new(input: Self::Input) -> Self;
/// Résout le challenge
fn solve(&self) -> Self::Output;
/// Vérifie qu'une sortie est valide pour le challenge
fn verify(&self, answer: &Self::Output) -> bool;
}
Différents challenges pourront être requis pour le jeu.
Le premier challenge est le HashCash; au moins un autre challenge sera à définir collectivement.
-
Nonogram solver (à qualifier plus tard)
-
Bloxorz Solver (à qualifier plus tard)
Votre objectif
-
Réaliser un client écrit en Rust sans bibliothèque extérieure autres que celles autorisées.
C'est la partie principale du projet.
-
Réaliser un serveur minimal qui permette de tester un client (pas besoin d'interface TUI ou GUI, même si ma démo peut le suggérer ; ce peut être fait en "bonus").
-
Il ne doit pas y avoir de duplication de code entre le client et le serveur. Vous définirez un "crate" pour:
- Le client
- Le serveur
- Les éléments communs au client et au serveur
Les modalités de réalisation
-
Le projet doit être traité par groupe de 2 ou 3 personnes
-
Le code doit remettre sous Git (github ou gitlab).
Le projet Git devra être créé à partir d'un fork du projet portant le sujet (et n'oubliez pas de m'en donner l'accès en lecture).
-
Le code doit être fonctionnel sous Linux, macOS et Windows
-
Le code devra être raisonnablement testé (par des tests unitaires et des tests d'intégration)
-
Le code devra suivre les règles de codage défini par
rustfmt
-
Le code devra être documenté avec
rustdoc
-
La documentation devra être intégrée au dépôt du code et écrite au format Markdown.
-
Les seuls modules (aka crates) autorisés ici sont:
rand
clap
serde
Le jour de la soutenance orale, vous serez évalués sur :
-
Le respect des consignes
-
La fiabilité et le respect du protocole entre client et serveur
-
Le respect des idiomes Rust (dont la gestion des erreurs)
-
L'organisation et la lisibilité du code
-
Il y aura une note collective et une note individuelle.
-
La doc Markdown doit mettre en évidence
-
Votre organisation du travail en tant qu'équipe
-
Votre démarche d'élaboration des différents composants du projet
-
Les spécificités de votre projet (i.e. ce qui n'est pas déjà dans le sujet)
-
Vos éventuels bonus (parmi la liste présentée ou bien d'autres si validés au préalable par l'enseignant)
Les bonus ne sont pris en compte uniquement si le client "joueur" est fonctionnel (fonctionnement raisonnablement sans planter dans des situations "normales" de jeu). Le niveau minimal fonctionnel du client et du serveur (en mode test de votre client) défini la note de 10/20.
-
Bonus possibles :
-
Réaliser une interface pour le client et/ou le serveur
-
Ajouter une intégration continue qui permette de tester votre code client et serveur (sous GitHub ou GitLab)
-
Employer une stratégie de jeu pour maximiser les chances de victoires d'un joueur
Cela peut être une stratégie locale au joueur ou bien en interaction avec les autres joueurs
-
Déployer des techniques avancées pour optimiser la performance de résolution du challenge
-
Réussir à faire crasher le serveur de référence
NB : Pour les Bonus, vous avez le droit d'employer des modules (aka crates) additionnels.
Aide à la mise au point
- Un serveur de référence sera distribué sous forme compilée pour tester vos clients.
Le protocole d'échange
Tous les messages se passent sur un flux TCP qui doit rester ouvert pendant toute la durée de la partie (et fermer _ proprement_ en fin de partie).
Toute rupture de connexion entraîne le retrait du joueur de la liste des joueurs actifs (et donc fini les chances de victoire).
Tous les messages sont de la forme :
JSON message size | JSON message |
---|---|
(u32 encodé en Big Endian) | (encodé en utf8) |
Les messages possibles :
Nom du message | Champs du message | Exemple |
---|---|---|
Hello |
"Hello" |
|
Welcome |
version: u8 |
{"Welcome":{"version":1}} |
Subscribe |
name: String |
{"Subscribe":{"name":"free_patato"}} |
SubscribeResult |
enum { Ok, Err(SubscribeError) } |
{"SubscribeResult":{"Err":"InvalidName"}} |
PublicLeaderBoard |
Vec<PublicPlayer> |
{"PublicLeaderBoard":[{"name":"free_patato","stream_id":"127.0.0.1","score":10,"steps":20,"is_active":true,"total_used_time":1.234},{"name":"dark_salad","stream_id":"127.0.0.1","score":6,"steps":200,"is_active":true,"total_used_time":0.1234}]} |
Challenge |
enum { ChallengeName(ChallengeInput) } |
{"Challenge":{"MD5HashCash":{"complexity":5,"message":"Hello"}}} |
ChallengeResult |
result: ChallengeAnswer next_target: String |
{"ChallengeResult":{"answer":{"MD5HashCash":{"seed":12345678,"hashcode":"68B329DA9893E34099C7D8AD5CB9C940"}},"next_target":"dark_salad"}} |
RoundSummary |
challenge: String chain: Vec<ReportedChallengeResult> |
{"RoundSummary":{"challenge":"MD5HashCash","chain":[{"name":"free_patato","value":{"Ok":{"used_time":0.1,"next_target":"dark_salad"}}},{"name":"dark_salad","value":"Unreachable"}]}} |
EndOfGame |
leader_board: PublicLeaderBoard |
{"EndOfGame":{"leader_board":[{"name":"free_patato","stream_id":"127.0.0.1","score":10,"steps":20,"is_active":true,"total_used_time":1.234},{"name":"dark_salad","stream_id":"127.0.0.1","score":6,"steps":200,"is_active":true,"total_used_time":0.1234}]}} |
Séquencement des messages
Les types additionnels :
Nom du type | Description du type |
---|---|
SubscribeError |
enum { AlreadyRegistered, InvalidName } |
PublicPlayer |
name: String stream_id: String score: i32 steps: u32 is_active: bool total_used_time: f64 |
ChallengeAnswer |
enum { ChallengeName(ChallengeOutput) } |
ChallengeResult |
name: ChallengeAnswer next_target: String |
ChallengeValue |
enum { Unreachable, Timeout, BadResult { used_time: f64, next_target: String }, Ok { used_time: f64, next_target: String } } |
ReportedChallengeResult |
name: String, value: JobValue |
PublicLeaderBoard |
.0: Vec<PublicPlayer> |