# GeoClue 2 daemon.
{
config,
lib,
pkgs,
...
}:
let
cfg = config.services.geoclue2;
defaultWhitelist = [
"gnome-shell"
"io.elementary.desktop.agent-geoclue2"
];
appConfigModule = lib.types.submodule (
{ name, ... }:
options = {
desktopID = lib.mkOption {
type = lib.types.str;
description = "Desktop ID of the application.";
};
isAllowed = lib.mkOption {
type = lib.types.bool;
description = ''
Whether the application will be allowed access to location information.
'';
isSystem = lib.mkOption {
Whether the application is a system component or not.
users = lib.mkOption {
type = lib.types.listOf lib.types.str;
default = [ ];
List of UIDs of all users for which this application is allowed location
info access, Defaults to an empty string to allow it for all users.
config.desktopID = lib.mkDefault name;
}
);
appConfigToINICompatible =
_:
desktopID,
isAllowed,
isSystem,
users,
name = desktopID;
value = {
allowed = isAllowed;
system = isSystem;
users = lib.concatStringsSep ";" users;
in
###### interface
services.geoclue2 = {
enable = lib.mkOption {
default = false;
Whether to enable GeoClue 2 daemon, a DBus service
that provides location information for accessing.
enableDemoAgent = lib.mkOption {
default = true;
Whether to use the GeoClue demo agent. This should be
overridden by desktop environments that provide their own
agent.
enableNmea = lib.mkOption {
Whether to fetch location from NMEA sources on local network.
enable3G = lib.mkOption {
Whether to enable 3G source.
enableCDMA = lib.mkOption {
Whether to enable CDMA source.
enableModemGPS = lib.mkOption {
Whether to enable Modem-GPS source.
enableWifi = lib.mkOption {
Whether to enable WiFi source.
geoProviderUrl = lib.mkOption {
default = "https://location.services.mozilla.com/v1/geolocate?key=geoclue";
example = "https://www.googleapis.com/geolocation/v1/geolocate?key=YOUR_KEY";
The url to the wifi GeoLocation Service.
package = lib.mkOption {
type = lib.types.package;
default = pkgs.geoclue2;
defaultText = lib.literalExpression "pkgs.geoclue2";
apply =
pkg:
pkg.override {
# the demo agent isn't built by default, but we need it here
withDemoAgent = cfg.enableDemoAgent;
description = "The geoclue2 package to use";
submitData = lib.mkOption {
Whether to submit data to a GeoLocation Service.
submissionUrl = lib.mkOption {
default = "https://location.services.mozilla.com/v1/submit?key=geoclue";
The url to submit data to a GeoLocation Service.
submissionNick = lib.mkOption {
default = "geoclue";
A nickname to submit network data with.
Must be 2-32 characters long.
appConfig = lib.mkOption {
type = lib.types.attrsOf appConfigModule;
default = { };
example = lib.literalExpression ''
"com.github.app" = {
isAllowed = true;
isSystem = true;
users = [ "300" ];
Specify extra settings per application.
###### implementation
config = lib.mkIf cfg.enable {
environment.systemPackages = [ cfg.package ];
services.dbus.packages = [ cfg.package ];
systemd.packages = [ cfg.package ];
# we cannot use DynamicUser as we need the the geoclue user to exist for the
# dbus policy to work
users = {
users.geoclue = {
isSystemUser = true;
home = "/var/lib/geoclue";
group = "geoclue";
description = "Geoinformation service";
groups.geoclue = { };
systemd.services.geoclue = {
wants = lib.optionals cfg.enableWifi [ "network-online.target" ];
after = lib.optionals cfg.enableWifi [ "network-online.target" ];
# restart geoclue service when the configuration changes
restartTriggers = [
config.environment.etc."geoclue/geoclue.conf".source
serviceConfig.StateDirectory = "geoclue";
# this needs to run as a user service, since it's associated with the
# user who is making the requests
systemd.user.services = lib.mkIf cfg.enableDemoAgent {
geoclue-agent = {
description = "Geoclue agent";
# this should really be `partOf = [ "geoclue.service" ]`, but
# we can't be part of a system service, and the agent should
# be okay with the main service coming and going
wantedBy = [ "default.target" ];
unitConfig.ConditionUser = "!@system";
serviceConfig = {
Type = "exec";
ExecStart = "${cfg.package}/libexec/geoclue-2.0/demos/agent";
Restart = "on-failure";
PrivateTmp = true;
services.geoclue2.appConfig.epiphany = {
isSystem = false;
services.geoclue2.appConfig.firefox = {
environment.etc."geoclue/geoclue.conf".text = lib.generators.toINI { } (
agent = {
whitelist = lib.concatStringsSep ";" (
lib.optional cfg.enableDemoAgent "geoclue-demo-agent" ++ defaultWhitelist
network-nmea = {
enable = cfg.enableNmea;
"3g" = {
enable = cfg.enable3G;
cdma = {
enable = cfg.enableCDMA;
modem-gps = {
enable = cfg.enableModemGPS;
wifi = {
enable = cfg.enableWifi;
url = cfg.geoProviderUrl;
submit-data = lib.boolToString cfg.submitData;
submission-url = cfg.submissionUrl;
submission-nick = cfg.submissionNick;
// lib.mapAttrs' appConfigToINICompatible cfg.appConfig
meta = with lib; {
maintainers = with maintainers; [ ] ++ teams.pantheon.members;