This shows you the differences between two versions of the page.
isi:laboratoare:07 [2024/11/17 20:03] alexandru.predescu |
isi:laboratoare:07 [2024/11/21 11:04] (current) sorin.ciolofan |
||
---|---|---|---|
Line 240: | Line 240: | ||
=== Partea 2 - Exerciții: === | === Partea 2 - Exerciții: === | ||
+ | |||
+ | <note> | ||
+ | Descărcați proiectul de pe GitHub: [[https://github.com/ACS-ISI/Messenger/tree/main|Messenger]] | ||
+ | </note> | ||
+ | |||
+ | |||
<note tip> | <note tip> | ||
Pentru a instala toate dependențele necesare pentru laboratorul de astăzi, rulați: ''pip install -r requirements.txt''. Fișierul ''requirements.txt'' poate fi descărcat de aici: [[https://github.com/ACS-ISI/Messenger/blob/main/requirements.txt|requirements.txt]]. | Pentru a instala toate dependențele necesare pentru laboratorul de astăzi, rulați: ''pip install -r requirements.txt''. Fișierul ''requirements.txt'' poate fi descărcat de aici: [[https://github.com/ACS-ISI/Messenger/blob/main/requirements.txt|requirements.txt]]. | ||
Line 245: | Line 251: | ||
0. Rulați local o instanță de RabbitMQ, fie folosind //docker-compose//-ul pus la dispoziție mai sus, fie instalând RabbitMQ direct pe mașina voastră. **Atenție**, dacă optați pentru instalarea directă pe mașina voastră, va fi necesar să instalați separat plugin-ul care oferă acces la interfața web - [[https://www.rabbitmq.com/docs/management|Plugin]]. | 0. Rulați local o instanță de RabbitMQ, fie folosind //docker-compose//-ul pus la dispoziție mai sus, fie instalând RabbitMQ direct pe mașina voastră. **Atenție**, dacă optați pentru instalarea directă pe mașina voastră, va fi necesar să instalați separat plugin-ul care oferă acces la interfața web - [[https://www.rabbitmq.com/docs/management|Plugin]]. | ||
- | 1. **Creează un script Python care să permită citirea textelor de la tastatură și publicarea acestora într-un exchange nou.** | + | 1. **Creează un script Python care să permită citirea textelor de la tastatură și publicarea acestora într-un exchange nou.** (care este creat programatic după realizarea cu succes a conexiunii la RabbitMQ). Scriptul va citi textul de la utilizator și va publica fiecare mesaj în exchange-ul RabbitMQ (in mod ''fanout''). |
- | Scriptul va citi textul de la utilizator și va publica fiecare mesaj în exchange-ul RabbitMQ, care este creat programatic după realizarea cu succes a conexiunii la RabbitMQ. Exchange-ul va fi configurat din cod imediat după inițializarea conexiunii. | + | |
+ | <code python> | ||
+ | try: | ||
+ | while True: | ||
+ | message = input("Message: ") | ||
+ | channel.basic_publish(exchange=exchange_name, routing_key='', body=message) | ||
+ | finally: | ||
+ | connection.close() | ||
+ | </code> | ||
2. **Creează un script Python care să creeze un queue, să-l lege la exchange-ul din exercițiul 1 și să afișeze fiecare mesaj primit.** | 2. **Creează un script Python care să creeze un queue, să-l lege la exchange-ul din exercițiul 1 și să afișeze fiecare mesaj primit.** | ||
- | Scriptul va crea un queue nou și îl va asocia (binding) cu exchange-ul creat în exercițiul anterior. Pentru fiecare mesaj primit în queue, scriptul va afișa conținutul acestuia pe ecran. | + | Scriptul va crea un queue nou și îl va asocia (binding) cu exchange-ul creat în exercițiul anterior. Pentru fiecare mesaj primit în queue, scriptul va afișa conținutul acestuia în consolă. |
+ | <code python> | ||
+ | channel.queue_bind(exchange=exchange_name, queue=queue_name) | ||
+ | </code> | ||
- | 3. **Chat interactiv** - Pornind de la codul pentru această aplicație de mesagerie, implementează funcționalitățile necesare, marcate cu TODO-uri in fisierul ''message_service.py'', pentru ca aceasta să funcționeze complet: | + | 3. **Chat interactiv** |
- | <note> | + | Această aplicație presupune un server Flask care interacționează cu RabbitMQ și expune o interfață web pentru utilizator. |
- | Descărcați proiectul de pe GitHub: [[https://github.com/ACS-ISI/Messenger/tree/main|Messenger]] | + | |
- | </note> | + | |
- | Implementare: | + | **Implementare:** Pornind de la codul pentru această aplicație de mesagerie, implementează funcționalitățile necesare, marcate cu TODO-uri in fisierul ''message_service.py'', pentru ca aceasta să funcționeze complet: |
* TODO a) Trimite la fiecare 30 de secunde un mesaj de tip broadcast pe exchange-ul 'chat_online_user_exchange' și fără nici un routing_key, prin care să informezi ceilalți participanți la chat că ești online și disponibil pentru conversație. Asigură-te că acest mesaj are un timp de viață de maximum 2 minute în queue. | * TODO a) Trimite la fiecare 30 de secunde un mesaj de tip broadcast pe exchange-ul 'chat_online_user_exchange' și fără nici un routing_key, prin care să informezi ceilalți participanți la chat că ești online și disponibil pentru conversație. Asigură-te că acest mesaj are un timp de viață de maximum 2 minute în queue. | ||
Line 264: | Line 279: | ||
* TODO c) Implementează trimiterea unui mesaj către un anumit utilizator pentru a putea iniția o conversație privată cu acesta. | * TODO c) Implementează trimiterea unui mesaj către un anumit utilizator pentru a putea iniția o conversație privată cu acesta. | ||
- | Testare: | + | <note tip> |
- | * Testează funcționalitatea pentru a verifica dacă poți comunica cu un alt utilizator (folosind două instanțe locale de aplicație). | + | |
- | * Schimbă conexiunea la RabbitMQ cu următoarea și testează dacă poți conversa cu alți colegi. | + | |
- | <code> | + | Folosește instanța locală și rulează scriptul ''initialize.py'' din laborator, care configurează 3 utilizatori: **stefan**, **alex** și **maria**. |
- | Host: acc-atomicsoftware.swedencentral.cloudapp.azure.com | + | |
- | User: student | + | |
- | Parola: întreabă coordonatorul de laborator | + | |
- | </code> | + | |
- | <note tip> | ||
**Testare**: Pentru a testa aplicația, puteți rula două instanțe ale aplicației pe porturi diferite, pentru a simula doi utilizatori care folosesc aplicația pe dispozitive diferite. În cadrul testării, punctul a) poate fi verificat individual, iar punctele b) și c) vor fi testate împreună. | **Testare**: Pentru a testa aplicația, puteți rula două instanțe ale aplicației pe porturi diferite, pentru a simula doi utilizatori care folosesc aplicația pe dispozitive diferite. În cadrul testării, punctul a) poate fi verificat individual, iar punctele b) și c) vor fi testate împreună. | ||
Pentru a rula aplicația, folosește comanda: ''python app.py'' sau ''flask run''. | Pentru a rula aplicația, folosește comanda: ''python app.py'' sau ''flask run''. | ||
+ | |||
Aceasta va rula implicit pe portul 5000, iar pentru a vizualiza interfața pusă la dispoziție de aceasta, deschideți [[http://localhost:5000|]]. | Aceasta va rula implicit pe portul 5000, iar pentru a vizualiza interfața pusă la dispoziție de aceasta, deschideți [[http://localhost:5000|]]. | ||
- | Pentru a rula aplicația pe un alt port decât cel implicit, rulați ''flask run --port 5001''. | + | Pentru a rula aplicația pe un alt port decât cel implicit, rulați ''flask run <nowiki>--</nowiki>port 5001''. |
</note> | </note> | ||
+ | |||
+ | **Testare:** | ||
+ | |||
+ | * Testează funcționalitatea pentru a verifica dacă poți comunica cu un alt utilizator (folosind două instanțe locale de aplicație). | ||
+ | * Schimbă conexiunea la RabbitMQ cu următoarea și testează dacă poți conversa cu alți colegi: | ||
+ | |||
+ | <code> | ||
+ | Host: acc-atomicsoftware.swedencentral.cloudapp.azure.com | ||
+ | User: student | ||
+ | Parola: întreabă coordonatorul de laborator | ||
+ | </code> | ||
<note important> | <note important> | ||
- | În cadrul acestui exercițiu, nu vei declara queue-uri sau exchange-uri, deoarece acestea sunt deja create. Până să te conectezi la instanța din cloud, folosește instanța locală și rulează scriptul ''initialize.py'' din laborator, care configurează 3 utilizatori: **stefan**, **alex** și **maria**. | + | În cadrul acestui exercițiu, nu vei declara queue-uri sau exchange-uri, deoarece acestea sunt deja create pe instanța din cloud. |
- | </note> | + | |
- | <note tip>**Important**: Când te conectezi la instanța din cloud, utilizatorii vor avea formatul ''nume_prenume_grupa''. Dacă nu ești sigur care este utilizatorul pregătit pentru tine, conectează-te la: | + | **Important**: Când te conectezi la instanța din cloud, utilizatorii vor avea formatul ''nume_prenume_grupa''. Dacă nu ești sigur care este utilizatorul pregătit pentru tine, conectează-te la: |
[[http://acc-atomicsoftware.swedencentral.cloudapp.azure.com:15672]] | [[http://acc-atomicsoftware.swedencentral.cloudapp.azure.com:15672]] | ||
cu utilizatorul si parola de mai sus și caută în tab-ul //Queues and Streams// după numele tău. Queue-ul o să se numească //chatUser.username//, iar tu trebuie să folosești doar //username// pentru a te conecta. În situația în care totuși nu te regăsești, poți folosi unul dintre utilizatorii următori: //bob//, //alice//, //stefan//, //alex//.</note> | cu utilizatorul si parola de mai sus și caută în tab-ul //Queues and Streams// după numele tău. Queue-ul o să se numească //chatUser.username//, iar tu trebuie să folosești doar //username// pentru a te conecta. În situația în care totuși nu te regăsești, poți folosi unul dintre utilizatorii următori: //bob//, //alice//, //stefan//, //alex//.</note> | ||
+ | |||
+ | <note important> | ||
+ | În situația în care întâmpinați o eroare similară cu "connection lost" sau "unable to pop from an empty queue" atunci când începeți să consumați mesaje, este indicat să adăugați o metodă nouă pentru a crea un channel și să utilizați un channel nou creat atunci când începeți să ascultați. | ||
+ | |||
+ | <code> | ||
+ | def getChannel(self): | ||
+ | connection = pika.BlockingConnection(self.connection_params) | ||
+ | return connection.channel() | ||
+ | </code> | ||
+ | </note> | ||
**Diagrame explicative ale exercițiului 3:** | **Diagrame explicative ale exercițiului 3:** | ||
+ | <spoiler> | ||
{{:isi:laboratoare:general-architecture.jpg?400|}} | {{:isi:laboratoare:general-architecture.jpg?400|}} | ||
Line 301: | Line 331: | ||
{{:isi:laboratoare:rabbitmq-send-message.jpg?500|}} | {{:isi:laboratoare:rabbitmq-send-message.jpg?500|}} | ||
+ | </spoiler> | ||