mirror of
https://github.com/AuthMe/AuthMeReloaded.git
synced 2024-12-19 07:07:55 +01:00
Download database only if there is a newer one uploaded (Fixes #1581)
This commit is contained in:
parent
14d3d1ad91
commit
5058747b10
@ -21,8 +21,9 @@ import fr.xephi.authme.util.InternetProtocolUtils;
|
|||||||
|
|
||||||
import java.io.BufferedInputStream;
|
import java.io.BufferedInputStream;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.OutputStream;
|
import java.net.HttpURLConnection;
|
||||||
import java.net.InetAddress;
|
import java.net.InetAddress;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.net.UnknownHostException;
|
import java.net.UnknownHostException;
|
||||||
@ -33,6 +34,9 @@ import java.nio.file.StandardCopyOption;
|
|||||||
import java.nio.file.attribute.FileTime;
|
import java.nio.file.attribute.FileTime;
|
||||||
import java.time.Duration;
|
import java.time.Duration;
|
||||||
import java.time.Instant;
|
import java.time.Instant;
|
||||||
|
import java.time.ZoneId;
|
||||||
|
import java.time.ZonedDateTime;
|
||||||
|
import java.time.format.DateTimeFormatter;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.zip.GZIPInputStream;
|
import java.util.zip.GZIPInputStream;
|
||||||
@ -55,6 +59,10 @@ public class GeoIpService {
|
|||||||
|
|
||||||
private static final int UPDATE_INTERVAL_DAYS = 30;
|
private static final int UPDATE_INTERVAL_DAYS = 30;
|
||||||
|
|
||||||
|
// The server for MaxMind doesn't seem to understand RFC1123,
|
||||||
|
// but every HTTP implementation have to support RFC 1023
|
||||||
|
private static final String TIME_RFC_1023 = "EEE, dd-MMM-yy HH:mm:ss zzz";
|
||||||
|
|
||||||
private final Path dataFile;
|
private final Path dataFile;
|
||||||
private final BukkitService bukkitService;
|
private final BukkitService bukkitService;
|
||||||
|
|
||||||
@ -104,7 +112,7 @@ public class GeoIpService {
|
|||||||
// don't fire the update task - we are up to date
|
// don't fire the update task - we are up to date
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
ConsoleLogger.debug("GEO Ip database is older than " + UPDATE_INTERVAL_DAYS + " Days");
|
ConsoleLogger.debug("GEO IP database is older than " + UPDATE_INTERVAL_DAYS + " Days");
|
||||||
}
|
}
|
||||||
} catch (IOException ioEx) {
|
} catch (IOException ioEx) {
|
||||||
ConsoleLogger.logException("Failed to load GeoLiteAPI database", ioEx);
|
ConsoleLogger.logException("Failed to load GeoLiteAPI database", ioEx);
|
||||||
@ -113,43 +121,38 @@ public class GeoIpService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// File is outdated or doesn't exist - let's try to download the data file!
|
// File is outdated or doesn't exist - let's try to download the data file!
|
||||||
startDownloadTask();
|
// use bukkit's cached threads
|
||||||
|
bukkitService.runTaskAsynchronously(this::updateDatabase);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a thread which will attempt to download new data from the GeoLite website.
|
* Tries to update the database by downloading a new version from the website.
|
||||||
*/
|
*/
|
||||||
private void startDownloadTask() {
|
private void updateDatabase() {
|
||||||
downloading = true;
|
downloading = true;
|
||||||
|
|
||||||
// use bukkit's cached threads
|
ConsoleLogger.info("Downloading GEO IP database, because the old database is older than "
|
||||||
bukkitService.runTaskAsynchronously(() -> {
|
+ UPDATE_INTERVAL_DAYS + " days or doesn't exist");
|
||||||
ConsoleLogger.info("Downloading GEO IP database, because the old database is outdated or doesn't exist");
|
|
||||||
|
|
||||||
Path tempFile = null;
|
Path tempFile = null;
|
||||||
try {
|
try {
|
||||||
// download database to temporarily location
|
// download database to temporarily location
|
||||||
tempFile = Files.createTempFile(ARCHIVE_FILE, null);
|
tempFile = Files.createTempFile(ARCHIVE_FILE, null);
|
||||||
try (OutputStream out = Files.newOutputStream(tempFile)) {
|
if (!downloadDatabaseArchive(tempFile)) {
|
||||||
Resources.copy(new URL(ARCHIVE_URL), out);
|
ConsoleLogger.info("There is no newer GEO IP database uploaded. Using the old one for now.");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// MD5 checksum verification
|
// MD5 checksum verification
|
||||||
String targetChecksum = Resources.toString(new URL(CHECKSUM_URL), StandardCharsets.UTF_8);
|
String expectedChecksum = Resources.toString(new URL(CHECKSUM_URL), StandardCharsets.UTF_8);
|
||||||
if (!verifyChecksum(Hashing.md5(), tempFile, targetChecksum)) {
|
verifyChecksum(Hashing.md5(), tempFile, expectedChecksum);
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// tar extract database and copy to target destination
|
// tar extract database and copy to target destination
|
||||||
if (!extractDatabase(tempFile, dataFile)) {
|
extractDatabase(tempFile, dataFile);
|
||||||
ConsoleLogger.warning("Cannot find database inside downloaded GEO IP file at " + tempFile);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ConsoleLogger.info("Successfully downloaded new GEO IP database to " + dataFile);
|
|
||||||
|
|
||||||
//only set this value to false on success otherwise errors could lead to endless download triggers
|
//only set this value to false on success otherwise errors could lead to endless download triggers
|
||||||
|
ConsoleLogger.info("Successfully downloaded new GEO IP database to " + dataFile);
|
||||||
downloading = false;
|
downloading = false;
|
||||||
} catch (IOException ioEx) {
|
} catch (IOException ioEx) {
|
||||||
ConsoleLogger.logException("Could not download GeoLiteAPI database", ioEx);
|
ConsoleLogger.logException("Could not download GeoLiteAPI database", ioEx);
|
||||||
@ -159,7 +162,49 @@ public class GeoIpService {
|
|||||||
FileUtils.delete(tempFile.toFile());
|
FileUtils.delete(tempFile.toFile());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Downloads the archive to the destination file if it's newer than the locally version.
|
||||||
|
*
|
||||||
|
* @param lastModified modification timestamp of the already present file
|
||||||
|
* @param destination save file
|
||||||
|
* @return false if we already have the newest version, true if successful
|
||||||
|
* @throws IOException if failed during downloading and writing to destination file
|
||||||
|
*/
|
||||||
|
private boolean downloadDatabaseArchive(Instant lastModified, Path destination) throws IOException {
|
||||||
|
HttpURLConnection connection = (HttpURLConnection) new URL(ARCHIVE_URL).openConnection();
|
||||||
|
if (lastModified != null) {
|
||||||
|
// Only download if we actually need a newer version - this field is specified in GMT zone
|
||||||
|
ZonedDateTime zonedTime = lastModified.atZone(ZoneId.of("GMT"));
|
||||||
|
String timeFormat = DateTimeFormatter.ofPattern(TIME_RFC_1023).format(zonedTime);
|
||||||
|
connection.addRequestProperty("If-Modified-Since", timeFormat);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (connection.getResponseCode() == HttpURLConnection.HTTP_NOT_MODIFIED) {
|
||||||
|
//we already have the newest version
|
||||||
|
connection.getInputStream().close();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Files.copy(connection.getInputStream(), destination, StandardCopyOption.REPLACE_EXISTING);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Downloads the archive to the destination file if it's newer than the locally version.
|
||||||
|
*
|
||||||
|
* @param destination save file
|
||||||
|
* @return false if we already have the newest version, true if successful
|
||||||
|
* @throws IOException if failed during downloading and writing to destination file
|
||||||
|
*/
|
||||||
|
private boolean downloadDatabaseArchive(Path destination) throws IOException {
|
||||||
|
Instant lastModified = null;
|
||||||
|
if (Files.exists(dataFile)) {
|
||||||
|
lastModified = Files.getLastModifiedTime(dataFile).toInstant();
|
||||||
|
}
|
||||||
|
|
||||||
|
return downloadDatabaseArchive(lastModified, destination);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -168,19 +213,15 @@ public class GeoIpService {
|
|||||||
* @param function the checksum function like MD5, SHA256 used to generate the checksum from the file
|
* @param function the checksum function like MD5, SHA256 used to generate the checksum from the file
|
||||||
* @param file the file we want to calculate the checksum from
|
* @param file the file we want to calculate the checksum from
|
||||||
* @param expectedChecksum the expected checksum
|
* @param expectedChecksum the expected checksum
|
||||||
* @return true if equal, false otherwise
|
* @throws IOException on I/O error reading the file or the checksum verification failed
|
||||||
* @throws IOException on I/O error reading the file
|
|
||||||
*/
|
*/
|
||||||
private boolean verifyChecksum(HashFunction function, Path file, String expectedChecksum) throws IOException {
|
private void verifyChecksum(HashFunction function, Path file, String expectedChecksum) throws IOException {
|
||||||
HashCode actualHash = function.hashBytes(Files.readAllBytes(file));
|
HashCode actualHash = function.hashBytes(Files.readAllBytes(file));
|
||||||
HashCode expectedHash = HashCode.fromString(expectedChecksum);
|
HashCode expectedHash = HashCode.fromString(expectedChecksum);
|
||||||
if (Objects.equals(actualHash, expectedHash)) {
|
if (!Objects.equals(actualHash, expectedHash)) {
|
||||||
return true;
|
throw new IOException("GEO IP Checksum verification failed. " +
|
||||||
|
"Expected: " + expectedChecksum + "Actual:" + actualHash);
|
||||||
}
|
}
|
||||||
|
|
||||||
ConsoleLogger.warning("GEO IP checksum verification failed");
|
|
||||||
ConsoleLogger.warning("Expected: " + expectedHash + " Actual: " + actualHash);
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -188,38 +229,37 @@ public class GeoIpService {
|
|||||||
*
|
*
|
||||||
* @param tarInputFile gzipped tar input file where the database is
|
* @param tarInputFile gzipped tar input file where the database is
|
||||||
* @param outputFile destination file for the database
|
* @param outputFile destination file for the database
|
||||||
* @return true if the database was found, false otherwise
|
* @throws IOException on I/O error reading the tar archive, or writing the output
|
||||||
* @throws IOException on I/O error reading the tar archive or writing the output
|
* @throws FileNotFoundException if the database cannot be found inside the archive
|
||||||
*/
|
*/
|
||||||
private boolean extractDatabase(Path tarInputFile, Path outputFile) throws IOException {
|
private void extractDatabase(Path tarInputFile, Path outputFile) throws FileNotFoundException, IOException {
|
||||||
// .gz -> gzipped file
|
// .gz -> gzipped file
|
||||||
try (BufferedInputStream in = new BufferedInputStream(Files.newInputStream(tarInputFile));
|
try (BufferedInputStream in = new BufferedInputStream(Files.newInputStream(tarInputFile));
|
||||||
TarInputStream tarIn = new TarInputStream(new GZIPInputStream(in))) {
|
TarInputStream tarIn = new TarInputStream(new GZIPInputStream(in))) {
|
||||||
TarEntry entry;
|
for (TarEntry entry = tarIn.getNextEntry(); entry != null; entry = tarIn.getNextEntry()) {
|
||||||
while ((entry = tarIn.getNextEntry()) != null) {
|
|
||||||
if (!entry.isDirectory()) {
|
|
||||||
// filename including folders (absolute path inside the archive)
|
// filename including folders (absolute path inside the archive)
|
||||||
String filename = entry.getName();
|
String filename = entry.getName();
|
||||||
if (filename.endsWith(DATABASE_EXT)) {
|
if (entry.isDirectory() || !filename.endsWith(DATABASE_EXT)) {
|
||||||
// found the database file
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// found the database file and copy file
|
||||||
Files.copy(tarIn, outputFile, StandardCopyOption.REPLACE_EXISTING);
|
Files.copy(tarIn, outputFile, StandardCopyOption.REPLACE_EXISTING);
|
||||||
|
|
||||||
// update the last modification date to be same as in the archive
|
// update the last modification date to be same as in the archive
|
||||||
Files.setLastModifiedTime(outputFile, FileTime.from(entry.getModTime().toInstant()));
|
Files.setLastModifiedTime(outputFile, FileTime.from(entry.getModTime().toInstant()));
|
||||||
return true;
|
return;
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
throw new FileNotFoundException("Cannot find database inside downloaded GEO IP file at " + tarInputFile);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the country code of the given IP address.
|
* Get the country code of the given IP address.
|
||||||
*
|
*
|
||||||
* @param ip textual IP address to lookup.
|
* @param ip textual IP address to lookup.
|
||||||
* @return two-character ISO 3166-1 alpha code for the country.
|
* @return two-character ISO 3166-1 alpha code for the country or "--" if it cannot be fetched.
|
||||||
*/
|
*/
|
||||||
public String getCountryCode(String ip) {
|
public String getCountryCode(String ip) {
|
||||||
return getCountry(ip).map(Country::getIsoCode).orElse("--");
|
return getCountry(ip).map(Country::getIsoCode).orElse("--");
|
||||||
@ -229,7 +269,7 @@ public class GeoIpService {
|
|||||||
* Get the country name of the given IP address.
|
* Get the country name of the given IP address.
|
||||||
*
|
*
|
||||||
* @param ip textual IP address to lookup.
|
* @param ip textual IP address to lookup.
|
||||||
* @return The name of the country.
|
* @return The name of the country or "N/A" if it cannot be fetched.
|
||||||
*/
|
*/
|
||||||
public String getCountryName(String ip) {
|
public String getCountryName(String ip) {
|
||||||
return getCountry(ip).map(Country::getName).orElse("N/A");
|
return getCountry(ip).map(Country::getName).orElse("N/A");
|
||||||
@ -255,7 +295,7 @@ public class GeoIpService {
|
|||||||
try {
|
try {
|
||||||
InetAddress address = InetAddress.getByName(ip);
|
InetAddress address = InetAddress.getByName(ip);
|
||||||
|
|
||||||
//Reader.getCountry() can be null for unknown addresses
|
// Reader.getCountry() can be null for unknown addresses
|
||||||
return Optional.ofNullable(databaseReader.getCountry(address)).map(CountryResponse::getCountry);
|
return Optional.ofNullable(databaseReader.getCountry(address)).map(CountryResponse::getCountry);
|
||||||
} catch (UnknownHostException e) {
|
} catch (UnknownHostException e) {
|
||||||
// Ignore invalid ip addresses
|
// Ignore invalid ip addresses
|
||||||
|
Loading…
Reference in New Issue
Block a user