generated from Arquisoft/wiq_0
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge remote-tracking branch 'origin/develop' into develop
- Loading branch information
Showing
19 changed files
with
168 additions
and
33 deletions.
There are no files selected for viewing
2 changes: 1 addition & 1 deletion
2
api/src/main/java/lab/en2b/quizapi/questions/answer/AnswerCategory.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,6 @@ | ||
package lab.en2b.quizapi.questions.answer; | ||
|
||
public enum AnswerCategory { | ||
CAPITAL_CITY, COUNTRY, SONG, STADIUM, BALLON_DOR, GAMES_PUBLISHER, PAINTING, WTPOKEMON, GAMES_COUNTRY, GAMES_GENRE, BASKETBALL_VENUE, COUNTRY_FLAG | ||
CAPITAL_CITY, COUNTRY, SONG, STADIUM, BALLON_DOR, GAMES_PUBLISHER, PAINTING, WTPOKEMON, GAMES_COUNTRY, GAMES_GENRE, BASKETBALL_VENUE, COUNTRY_FLAG, GAMES_RELEASE | ||
} | ||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file not shown.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file not shown.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -5,33 +5,10 @@ ifndef::imagesdir[:imagesdir: ../images] | |
=== Quality Tree | ||
This quality tree is a high-level overview of the quality goals and requirements. The Quality tree uses "quality" as a root while the rest of the quality categories will be displayed as branches. | ||
|
||
[plantuml,"Quality Tree",png] | ||
[plantuml,"Quality Tree",png,align="center"] | ||
---- | ||
@startuml | ||
title Quality attributes | ||
agent Quality | ||
agent Security | ||
agent Reliability | ||
agent Transferability | ||
agent Usability | ||
agent "Performance Efficiency" | ||
agent Maintainability | ||
agent Availability | ||
agent Compatibility | ||
agent "Functional Suitability" | ||
Quality --- Security | ||
Quality --- Reliability | ||
Quality --- Transferability | ||
Quality --- Usability | ||
Quality --- "Performance Efficiency" | ||
Quality --- Maintainability | ||
Quality --- Availability | ||
Quality --- Compatibility | ||
Quality --- "Functional Suitability" | ||
@enduml | ||
include::../diagrams/10_Quality_Tree.puml[] | ||
---- | ||
|
||
=== Quality Scenarios | ||
To obtain a measurable system response to stimulus corresponding to the various quality branches outlined in the mindmap, we will use quality scenarios. Scenarios make quality requirements concrete and allow to more easily measure or decide whether they are fulfilled. | ||
|
||
|
@@ -42,17 +19,74 @@ To obtain a measurable system response to stimulus corresponding to the various | |
| Functional suitability | Users shall be able to register, log in, play the quiz, and access historical data without encountering errors or glitches. | High, Medium | ||
| Reliability | The system shall reliably generate accurate and diverse questions from Wikidata. User registrations, logins, and game data storage shall be handled without errors. | High, Medium | ||
| Availability | The system shall be available 99% of the time when a user attempts to access it. | High, High | ||
| Performance efficiency | The system shall deliver optimal performance, ensuring responsive interactions for users. It shall efficiently generate questions from Wikidata and handle real-time gameplay with up to N concurrent users. | High, High | ||
| Performance efficiency | The system shall deliver optimal performance, ensuring responsive interactions for users. It shall efficiently generate questions from Wikidata and handle real-time gameplay with up to 1000 concurrent users. | High, High | ||
| Usability | The system shall provide a user-friendly interface, allowing users to register, log in, and play the game with a learning time of less than 4 hours. | High, Medium | ||
| Security | User data shall be securely handled. Robust authentication mechanisms shall be in place for user registration and login. API access points for user information and generated questions shall be secured with proper authorization. | Medium, High | ||
| Compatibility | The system shall be compatible with various web browsers and devices, providing a seamless experience for users regardless of their choice of platform. It shall be well-optimized for different screen sizes and functionalities. | High, Medium | ||
| Transferability | The system shall allow for easy transfer of user data and game-related information through its APIs. | Medium, High | ||
| Testability | The unit tests shall have at least 80% coverage. | High, Medium | ||
| Monitoring | The system shall have monitoring in place to track the performance and availability of the system. | High, Medium | ||
|=== | ||
==== Change Scenarios | ||
[options="header",cols="1,3,1"] | ||
|=== | ||
|Quality attribute|Scenario|Priority | ||
| Maintainability | The system shall be designed and implemented in a way that allows for easy maintenance and updates. | High, Medium | ||
| Modifiability | The system shall be designed and implemented in a way that allows for easy maintenance and updates. | High, Medium | ||
| Maintainability | The code of the system shall be well-documented, and modular, allowing for efficient troubleshooting and modifications. | High, Medium | ||
|=== | ||
|=== | ||
|
||
==== Implementation | ||
|
||
===== Performance efficiency | ||
The tests were done with a 2 core and 4 GB of memory system. | ||
This system's efficiency has been measured with Gatling. For the script that we used, a user already created, logged in and played a full game. After that, the user clicked to look the statistics. | ||
The scripts were run a total of 4 times. One with 1 user, other with 100 users, another one with 1000 users and finally one with 10000 users. | ||
The results of this scripts show that response times are reasonable up until 1000 users. Having 10000 users playing a game at the same time make a lot of failures. | ||
Here are the results. | ||
|
||
**1 user:** | ||
|
||
image::Gatling_1_user.png[align="center", title="Gatling results with 1 user"] | ||
|
||
**100 users:** | ||
|
||
image::Gatling_100_users.png[align="center", title="Gatling results with 100 user"] | ||
|
||
**1000 users:** | ||
|
||
image::Gatling_1000_users.png[align="center", title="Gatling results with 1000 user"] | ||
|
||
**10000 users:** | ||
|
||
image::Gatling_10000_users.png[align="center", title="Gatling results with 10000 user"] | ||
|
||
===== Security | ||
The system is secured using Spring Security. The user data is stored in a database and the passwords are hashed using BCrypt. The API access points are secured with proper authorization. HTTPS is used to encrypt the data in transit. | ||
|
||
The system is also protected against SQL injection via using JPA repositories and prepared statements. | ||
|
||
The system is also designed in such a way that prevents cheating, by limiting the options available for the user and doing all validation in the backend, such as checking if the answer is correct, preventing request forgery. | ||
|
||
===== Testability | ||
|
||
===== Monitoring | ||
The system is monitored using Spring Boot Actuator and Prometheus. The monitoring data is visualized using Grafana. | ||
|
||
The actuator is deployed in https://kiwiq.run.place:8443/actuator/prometheus. | ||
|
||
image::actuator.png[align="center", title="Spring Boot Actuator"] | ||
|
||
The Prometheus server is deployed in http://20.199.84.5:9090. | ||
|
||
image::prometheus.png[align="center", title="Prometheus"] | ||
|
||
The Grafana is deployed in http://20.199.84.5:9091. The Grafana dashboard is available at the following link with user [email protected] and password aswgrafana. | ||
|
||
The dashboard used is: https://grafana.com/grafana/dashboards/6756-spring-boot-statistics/ | ||
Make sure to put kiwiq.run.place:8443 as the Instance and WIQ API as the application. | ||
|
||
image::grafana.png[align="center", title="Graphana Spring Boot dashboard"] | ||
|
||
===== Maintainability | ||
In our CodeScene analysis we find that our knowledge distribution is well-balanced as well as a nice code health, excepting one hotspot on a test that is not relevant. | ||
|
||
image::codescene-general.png[align="center", title="CodeScene general view"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,6 @@ | ||
package model; | ||
|
||
public enum AnswerCategory { | ||
CAPITAL_CITY, COUNTRY, SONG, STADIUM, BALLON_DOR, GAMES_PUBLISHER, PAINTING, WTPOKEMON, GAMES_COUNTRY, GAMES_GENRE, BASKETBALL_VENUE, COUNTRY_FLAG | ||
CAPITAL_CITY, COUNTRY, SONG, STADIUM, BALLON_DOR, GAMES_PUBLISHER, PAINTING, WTPOKEMON, GAMES_COUNTRY, GAMES_GENRE, BASKETBALL_VENUE, COUNTRY_FLAG, GAMES_RELEASE | ||
} | ||
|
94 changes: 94 additions & 0 deletions
94
questiongenerator/src/main/java/templates/VideogamesReleaseDate.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,94 @@ | ||
package templates; | ||
|
||
import model.*; | ||
import org.json.JSONObject; | ||
|
||
import java.util.ArrayList; | ||
import java.util.List; | ||
|
||
public class VideogamesReleaseDate extends QuestionTemplate { | ||
List<String> videoGameLabels; | ||
|
||
private static final String[] spanishStringsIni = {"¿Cuándo se publicó ", "¿En qué fecha fue lanzado ", "¿Cuándo fue lanzado ", "¿En qué fecha fue publicado "}; | ||
private static final String[] englishStringsIni= {"When was ", "On what date was ", "When was ", "On what date was "}; | ||
|
||
private static final String[] spanishStringsFin = {"?", "?", "?", "?"}; | ||
private static final String[] englishStringsFin = {" released?", " released?", " launched?", " launched?"}; | ||
|
||
public VideogamesReleaseDate(String langCode) { | ||
super(langCode); | ||
} | ||
|
||
@Override | ||
public void setQuery() { | ||
this.sparqlQuery = "SELECT ?gameLabel (MAX(?unitsSoldValue) as ?maxUnitsSold) (MIN(?publicationDate) as ?oldestPublicationDate)\n" + | ||
"WHERE { " + | ||
" ?game wdt:P31 wd:Q7889; " + | ||
" wdt:P2664 ?unitsSoldValue. " + | ||
" ?game wdt:P577 ?publicationDate. " + | ||
" SERVICE wikibase:label { bd:serviceParam wikibase:language \"" + langCode + "\". } " + | ||
"} " + | ||
"GROUP BY ?game ?gameLabel " + | ||
"ORDER BY DESC(?maxUnitsSold) " + | ||
"LIMIT 150 "; | ||
} | ||
|
||
@Override | ||
public void processResults() { | ||
videoGameLabels = new ArrayList<>(); | ||
List<Question> questions = new ArrayList<>(); | ||
List<Answer> answers = new ArrayList<>(); | ||
|
||
for (int i = 0; i < results.length()-10; i++) { | ||
|
||
JSONObject result = results.getJSONObject(i); | ||
|
||
String videoGameLabel = ""; | ||
String publishDateLabel = ""; | ||
|
||
try { | ||
JSONObject videoGameLabelObject = result.getJSONObject("gameLabel"); | ||
videoGameLabel = videoGameLabelObject.getString("value"); | ||
|
||
JSONObject publishDateLabelObject = result.getJSONObject("oldestPublicationDate"); | ||
publishDateLabel = publishDateLabelObject.getString("value"); | ||
|
||
publishDateLabel = publishDateLabel.substring(0, 4); | ||
|
||
} catch (Exception e) { | ||
continue; | ||
} | ||
|
||
if (needToSkip(videoGameLabel, publishDateLabel)) | ||
continue; | ||
|
||
Answer a = new Answer(publishDateLabel, AnswerCategory.GAMES_RELEASE, langCode); | ||
answers.add(a); | ||
|
||
String questionString = ""; | ||
|
||
if (langCode.equals("es")) | ||
questionString = spanishStringsIni[i%4] + videoGameLabel + spanishStringsFin[i%4]; | ||
else | ||
questionString = englishStringsIni[i%4] + videoGameLabel + englishStringsFin[i%4]; | ||
|
||
questions.add(new Question(a, questionString, QuestionCategory.VIDEOGAMES, QuestionType.TEXT)); | ||
} | ||
|
||
repository.saveAll(new ArrayList<>(answers)); | ||
repository.saveAll(new ArrayList<>(questions)); | ||
} | ||
|
||
private boolean needToSkip(String videoGameLabel, String countryLabel) { | ||
if (videoGameLabels.contains(videoGameLabel)) { | ||
return true; | ||
} | ||
videoGameLabels.add(videoGameLabel); | ||
|
||
if (QGHelper.isEntityName(videoGameLabel) || QGHelper.isEntityName(countryLabel)) | ||
return true; | ||
|
||
return false; | ||
} | ||
} | ||
|