mirror of
https://github.com/kremalicious/umami.git
synced 2024-11-15 17:55:08 +01:00
100 lines
2.5 KiB
JavaScript
100 lines
2.5 KiB
JavaScript
import path from 'path';
|
|
import requestIp from 'request-ip';
|
|
import { browserName, detectOS } from 'detect-browser';
|
|
import isLocalhost from 'is-localhost-ip';
|
|
import maxmind from 'maxmind';
|
|
|
|
import {
|
|
DESKTOP_OS,
|
|
MOBILE_OS,
|
|
DESKTOP_SCREEN_WIDTH,
|
|
LAPTOP_SCREEN_WIDTH,
|
|
MOBILE_SCREEN_WIDTH,
|
|
} from './constants';
|
|
|
|
let lookup;
|
|
|
|
export function getIpAddress(req) {
|
|
// Custom header
|
|
if (req.headers[process.env.CLIENT_IP_HEADER]) {
|
|
return req.headers[process.env.CLIENT_IP_HEADER];
|
|
}
|
|
// Cloudflare
|
|
else if (req.headers['cf-connecting-ip']) {
|
|
return req.headers['cf-connecting-ip'];
|
|
}
|
|
|
|
return requestIp.getClientIp(req);
|
|
}
|
|
|
|
export function getDevice(screen, os) {
|
|
if (!screen) return;
|
|
|
|
const [width] = screen.split('x');
|
|
|
|
if (DESKTOP_OS.includes(os)) {
|
|
if (os === 'Chrome OS' || width < DESKTOP_SCREEN_WIDTH) {
|
|
return 'laptop';
|
|
}
|
|
return 'desktop';
|
|
} else if (MOBILE_OS.includes(os)) {
|
|
if (os === 'Amazon OS' || width > MOBILE_SCREEN_WIDTH) {
|
|
return 'tablet';
|
|
}
|
|
return 'mobile';
|
|
}
|
|
|
|
if (width >= DESKTOP_SCREEN_WIDTH) {
|
|
return 'desktop';
|
|
} else if (width >= LAPTOP_SCREEN_WIDTH) {
|
|
return 'laptop';
|
|
} else if (width >= MOBILE_SCREEN_WIDTH) {
|
|
return 'tablet';
|
|
} else {
|
|
return 'mobile';
|
|
}
|
|
}
|
|
|
|
export async function getLocation(ip) {
|
|
// Ignore local ips
|
|
if (await isLocalhost(ip)) {
|
|
return;
|
|
}
|
|
|
|
// Database lookup
|
|
if (!lookup) {
|
|
lookup = await maxmind.open(path.resolve('node_modules/.geo/GeoLite2-City.mmdb'));
|
|
}
|
|
|
|
const result = lookup.get(ip);
|
|
const country = result?.country?.iso_code ?? result?.registered_country?.iso_code;
|
|
const subdivision1 = result?.subdivisions[0]?.iso_code;
|
|
const subdivision2 = result?.subdivisions[1]?.names?.en;
|
|
const city = result?.city?.names?.en;
|
|
|
|
return { country, subdivision1, subdivision2, city };
|
|
}
|
|
|
|
export async function getClientInfo(req, { screen }) {
|
|
const userAgent = req.headers['user-agent'];
|
|
const ip = getIpAddress(req);
|
|
const location = await getLocation(ip);
|
|
const country = location?.country;
|
|
const subdivision1 = location?.subdivision1;
|
|
const subdivision2 = location?.subdivision2;
|
|
const city = location?.city;
|
|
const browser = browserName(userAgent);
|
|
const os = detectOS(userAgent);
|
|
const device = getDevice(screen, browser, os);
|
|
|
|
return { userAgent, browser, os, ip, country, subdivision1, subdivision2, city, device };
|
|
}
|
|
|
|
export function getJsonBody(req) {
|
|
if ((req.headers['content-type'] || '').indexOf('text/plain') !== -1) {
|
|
return JSON.parse(req.body);
|
|
}
|
|
|
|
return req.body;
|
|
}
|