Skip to content

Commit

Permalink
Merge branch 'develop' into 202-sentry-angular
Browse files Browse the repository at this point in the history
  • Loading branch information
GODrums authored Dec 7, 2024
2 parents bf2b7c3 + 566ae85 commit 49cf48b
Show file tree
Hide file tree
Showing 11 changed files with 316 additions and 6 deletions.
12 changes: 7 additions & 5 deletions server/application-server/pom.xml
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
Expand Down Expand Up @@ -197,6 +196,11 @@
<artifactId>keycloak-admin-client</artifactId>
<version>26.0.3</version>
</dependency>
<dependency>
<groupId>io.sentry</groupId>
<artifactId>sentry-spring-boot-starter-jakarta</artifactId>
<version>7.18.1</version>
</dependency>
</dependencies>
<dependencyManagement>
<dependencies>
Expand Down Expand Up @@ -283,9 +287,7 @@
<username>root</username>
<password>root</password>
<!-- Hibernate URL for reference schema using the Hibernate 6 dialect -->
<referenceUrl>hibernate:spring:de.tum.in.www1.hephaestus?dialect=org.hibernate.dialect.PostgreSQLDialect
&amp;hibernate.physical_naming_strategy=org.hibernate.boot.model.naming.CamelCaseToUnderscoresNamingStrategy
&amp;hibernate.implicit_naming_strategy=org.springframework.boot.orm.jpa.hibernate.SpringImplicitNamingStrategy
<referenceUrl>hibernate:spring:de.tum.in.www1.hephaestus?dialect=org.hibernate.dialect.PostgreSQLDialect &amp;hibernate.physical_naming_strategy=org.hibernate.boot.model.naming.CamelCaseToUnderscoresNamingStrategy &amp;hibernate.implicit_naming_strategy=org.springframework.boot.orm.jpa.hibernate.SpringImplicitNamingStrategy
</referenceUrl>
<!-- Enable detailed logging output -->
<verbose>true</verbose>
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
package de.tum.in.www1.hephaestus.config;

import io.sentry.Sentry;
import jakarta.annotation.PostConstruct;
import java.util.Optional;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.env.Environment;

@Configuration
public class SentryConfiguration {

private static final Logger logger = LoggerFactory.getLogger(SentryConfiguration.class);

@Autowired
private Environment environment;

@Value("${spring.application.version}")
private String hephaestusVersion;

@Value("${sentry.dsn}")
private Optional<String> sentryDsn;

/**
* Init sentry with the correct environment and version
*/
@PostConstruct
public void init() {
if (environment.matchesProfiles("specs")) {
logger.info("Sentry is disabled in specs profile");
return;
}

if (sentryDsn.isEmpty() || sentryDsn.get().isEmpty()) {
logger.info("Sentry is disabled: Provide a DSN to enable Sentry.");
return;
}

try {
final String dsn = sentryDsn.get() + "?stacktrace.app.packages=de.tum.in.www1.hephaestus";

Sentry.init(options -> {
options.setDsn(dsn);
options.setSendDefaultPii(true);
options.setEnvironment(getEnvironment());
options.setRelease(hephaestusVersion);
options.setTracesSampleRate(getTracesSampleRate());
});

logger.info("Sentry configuration was successful");
} catch (Exception ex) {
logger.error("Sentry configuration was not successful due to exception!", ex);
}
}

private String getEnvironment() {
if (environment.matchesProfiles("test")) {
return "test";
} else if (environment.matchesProfiles("prod")) {
return "prod";
} else {
return "local";
}
}

/**
* Get the traces sample rate based on the environment.
*
* @return 0% for local, 100% for test, 20% for production environments
*/
private double getTracesSampleRate() {
return switch (getEnvironment()) {
case "test" -> 1.0;
case "prod" -> 0.2;
default -> 0.0;
};
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,9 @@ public class User extends BaseGitServiceEntity {
@ToString.Exclude
private Set<PullRequestReviewComment> reviewComments = new HashSet<>();

// Current ranking points for the leaderboard leagues
private int leaguePoints;

public enum Type {
USER, ORGANIZATION, BOT
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,14 @@ public interface UserRepository extends JpaRepository<User, Long> {
""")
Optional<User> findByLogin(@Param("login") String login);

@Query("""
SELECT u
FROM User u
LEFT JOIN FETCH u.mergedPullRequests
WHERE u.login = :login
""")
Optional<User> findByLoginWithEagerMergedPullRequests(@Param("login") String login);

@Query("""
SELECT u
FROM User u
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
package de.tum.in.www1.hephaestus.leaderboard;

import de.tum.in.www1.hephaestus.leaderboard.tasks.LeaguePointsUpdateTask;
import de.tum.in.www1.hephaestus.leaderboard.tasks.SlackWeeklyLeaderboardTask;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
Expand Down Expand Up @@ -41,6 +42,9 @@ public class LeaderboardTaskScheduler {
@Autowired
private SlackWeeklyLeaderboardTask slackWeeklyLeaderboardTask;

@Autowired
private LeaguePointsUpdateTask leaguePointsUpdateTask;


@EventListener(ApplicationReadyEvent.class)
public void activateTaskScheduler() {
Expand All @@ -61,7 +65,7 @@ public void activateTaskScheduler() {
}

scheduleSlackMessage(cron);

scheduleLeaguePointsUpdate(cron);
}

/**
Expand All @@ -78,4 +82,9 @@ private void scheduleSlackMessage(String cron) {
logger.info("Scheduling Slack message to run with {}", cron);
taskScheduler.schedule(slackWeeklyLeaderboardTask, new CronTrigger(cron));
}

private void scheduleLeaguePointsUpdate(String cron) {
logger.info("Scheduling league points update to run with {}", cron);
taskScheduler.schedule(leaguePointsUpdateTask, new CronTrigger(cron));
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
package de.tum.in.www1.hephaestus.leaderboard;

import de.tum.in.www1.hephaestus.gitprovider.pullrequest.PullRequest;
import de.tum.in.www1.hephaestus.gitprovider.user.User;
import java.time.OffsetDateTime;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Service;

@Service
public class LeaguePointsCalculationService {
private final Logger logger = LoggerFactory.getLogger(LeaguePointsCalculationService.class);

// Starting points for new players
public static int POINTS_DEFAULT = 1000;
// Upper bound for first reducting in the k-factor
public static int POINTS_THRESHOLD_HIGH = 1750;
// Lower bound for first reducting in the k-factor
public static int POINTS_THRESHOLD_LOW = 1250;
// Minimum amount of points to decay each cycle
public static int DECAY_MINIMUM = 10;
// Factor to determine how much of the current points are decayed each cycle
public static double DECAY_FACTOR = 0.05;
// K-factors depending on the player's league points
public static double K_FACTOR_NEW_PLAYER = 2.0;
public static double K_FACTOR_LOW_POINTS = 1.5;
public static double K_FACTOR_MEDIUM_POINTS = 1.2;
public static double K_FACTOR_HIGH_POINTS = 1.1;



public int calculateNewPoints(User user, LeaderboardEntryDTO entry) {
// Initialize points for new players
if (user.getLeaguePoints() == 0) {
user.setLeaguePoints(POINTS_DEFAULT);
}

int oldPoints = user.getLeaguePoints();

double kFactor = getKFactor(user);
// Base decay
int decay = calculateDecay(user.getLeaguePoints());
// Bonus based on leaderboard score
int performanceBonus = calculatePerformanceBonus(entry.score());
// Additional bonus for placements
int placementBonus = calculatePlacementBonus(entry.rank());
// Calculate final point change
int pointChange = (int) (kFactor * (performanceBonus + placementBonus - decay));
// Apply minimum change to prevent extreme swings
int newPoints = Math.max(1, oldPoints + pointChange);

logger.info("Points calculation: old={}, k={}, decay={}, performanceBonus={}, placement={}, pointchange={}, new={}",
oldPoints, kFactor, decay, performanceBonus, placementBonus, pointChange, newPoints);

return newPoints;
}

/**
* Calculate the K factor for the user based on their current points and if they are a new player.
* The K-factor is used to control the sensitivity of the rating system to changes in the leaderboard.
* New players have a higher K-factor to allow them to quickly reach their true skill level.
* @param user
* @return K factor
* @see <a href="https://en.wikipedia.org/wiki/Elo_rating_system#Most_accurate_K-factor">Wikipedia: Most accurate K-factor</a>
*/
private double getKFactor(User user) {
if (isNewPlayer(user)) {
return K_FACTOR_NEW_PLAYER;
} else if (user.getLeaguePoints() < POINTS_THRESHOLD_LOW) {
return K_FACTOR_LOW_POINTS;
} else if (user.getLeaguePoints() < POINTS_THRESHOLD_HIGH) {
return K_FACTOR_MEDIUM_POINTS;
} else {
return K_FACTOR_HIGH_POINTS;
}
}

/**
* Check if the user's earliest merged pull request is within the last 30 days.
* @param user
* @return true if the pull request is within the last 30 days
*/
private boolean isNewPlayer(User user) {
return user.getMergedPullRequests().stream()
.filter(PullRequest::isMerged)
.map(PullRequest::getMergedAt)
.noneMatch(date -> date.isAfter(OffsetDateTime.now().minusDays(30)));
}

/**
* Calculate the base decay in points based on the current points.
* @param currentPoints Current amount of league points
* @return Amount of decay points
*/
private int calculateDecay(int currentPoints) {
// decay a part of the current points, at least DECAY_MINIMUM points
return currentPoints > 0 ? Math.max(DECAY_MINIMUM, (int)(currentPoints * DECAY_FACTOR)) : 0;
}

/**
* Calculate the bonus points based on the leaderboard score.
* @param score Leaderboard score
* @return Bonus points
*/
private int calculatePerformanceBonus(int score) {
// Convert leaderboard score directly to points with diminishing returns
return (int)(Math.sqrt(score) * 10);
}

/**
* Calculate the bonus points based on the placement in the leaderboard.
* @param placement Placement in the leaderboard
* @return Bonus points
*/
private int calculatePlacementBonus(int placement) {
// Bonus for top 3 placements
return placement <= 3 ? 20 * (4 - placement) : 0;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
package de.tum.in.www1.hephaestus.leaderboard.tasks;

import de.tum.in.www1.hephaestus.gitprovider.user.UserRepository;
import de.tum.in.www1.hephaestus.leaderboard.LeaderboardEntryDTO;
import de.tum.in.www1.hephaestus.leaderboard.LeaderboardService;
import de.tum.in.www1.hephaestus.leaderboard.LeaguePointsCalculationService;
import jakarta.transaction.Transactional;
import java.time.DayOfWeek;
import java.time.OffsetDateTime;
import java.time.temporal.TemporalAdjusters;
import java.util.List;
import java.util.Optional;
import java.util.function.Consumer;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

@Component
public class LeaguePointsUpdateTask implements Runnable {

@Value("${hephaestus.leaderboard.schedule.day}")
private String scheduledDay;

@Value("${hephaestus.leaderboard.schedule.time}")
private String scheduledTime;

@Autowired
private UserRepository userRepository;
@Autowired
private LeaderboardService leaderboardService;
@Autowired
private LeaguePointsCalculationService leaguePointsCalculationService;

@Override
@Transactional
public void run() {
List<LeaderboardEntryDTO> leaderboard = getLatestLeaderboard();
leaderboard.forEach(updateLeaderboardEntry());
}

/**
* Update ranking points of a user based on its leaderboard entry.
* @return {@code Consumer} that updates {@code leaguePoints} based on its leaderboard entry.
*/
private Consumer<? super LeaderboardEntryDTO> updateLeaderboardEntry() {
return entry -> {
var user = userRepository.findByLoginWithEagerMergedPullRequests(entry.user().login()).orElseThrow();
int newPoints = leaguePointsCalculationService.calculateNewPoints(user, entry);
user.setLeaguePoints(newPoints);
userRepository.save(user);
};
}

/**
* Retrieves the latest leaderboard based on the scheduled time of the environment.
* @return List of {@code LeaderboardEntryDTO} representing the latest leaderboard
*/
private List<LeaderboardEntryDTO> getLatestLeaderboard() {
String[] timeParts = scheduledTime.split(":");
OffsetDateTime before = OffsetDateTime.now()
.with(TemporalAdjusters.previousOrSame(DayOfWeek.of(Integer.parseInt(scheduledDay))))
.withHour(Integer.parseInt(timeParts[0]))
.withMinute(timeParts.length > 0 ? Integer.parseInt(timeParts[1]) : 0)
.withSecond(0)
.withNano(0);
OffsetDateTime after = before.minusWeeks(1);
return leaderboardService.createLeaderboard(after, before, Optional.empty());
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -51,3 +51,6 @@ github:
slack:
token: ${SLACK_BOT_TOKEN}
signing-secret: ${SLACK_SIGNING_SECRET}

sentry:
dsn: ${SENTRY_DSN}
4 changes: 4 additions & 0 deletions server/application-server/src/main/resources/application.yml
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
spring:
application:
name: Hephaestus
version: "0.0.1"

datasource:
url: jdbc:postgresql://localhost:5432/hephaestus
Expand Down Expand Up @@ -74,3 +75,6 @@ github:
slack:
token: ""
signing-secret: ""

sentry:
dsn: ""
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
<?xml version="1.1" encoding="UTF-8" standalone="no"?>
<databaseChangeLog xmlns="http://www.liquibase.org/xml/ns/dbchangelog" xmlns:ext="http://www.liquibase.org/xml/ns/dbchangelog-ext" xmlns:pro="http://www.liquibase.org/xml/ns/pro" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog-ext http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-ext.xsd http://www.liquibase.org/xml/ns/pro http://www.liquibase.org/xml/ns/pro/liquibase-pro-latest.xsd http://www.liquibase.org/xml/ns/dbchangelog http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-latest.xsd">
<changeSet author="godrums" id="1732966971482-1">
<addColumn tableName="user">
<column name="league_points" type="integer" defaultValue="0">
<constraints nullable="false"/>
</column>
</addColumn>
</changeSet>
</databaseChangeLog>
Loading

0 comments on commit 49cf48b

Please sign in to comment.