Skip to content

Instantly share code, notes, and snippets.

@simonw
Forked from RhetTbull/tzname.py
Last active January 26, 2025 16:57
Show Gist options
  • Save simonw/6daf3d8c8274b00df4f51f53da644581 to your computer and use it in GitHub Desktop.
Save simonw/6daf3d8c8274b00df4f51f53da644581 to your computer and use it in GitHub Desktop.
# /// script
# dependencies = [
# "pyobjc-core",
# "pyobjc-framework-CoreLocation"
# ]
# ///
"""Get named timezone for a location on macOS using CoreLocation
To use this script, you need to have pyobjc-core and pyobjc-framework-CoreLocation installed:
pip install pyobjc-core pyobjc-framework-CoreLocation
This script uses CoreLocation to get the timezone for a given latitude and longitude
as well as the timezone offset for a given date.
It effectively does the same thing as python packages like [tzfpy](https://pypi.org/project/tzfpy/)
or [timezonefinder](https://pypi.org/project/timezonefinder/) but uses macOS native APIs.
"""
import datetime
import time
import objc
from CoreLocation import CLGeocoder, CLLocation
from Foundation import NSDate, NSRunLoop, NSTimeZone
WAIT_FOR_COMPLETION = 0.01 # wait time for async completion in seconds
COMPLETION_TIMEOUT = 5.0 # timeout for async completion in seconds
def timezone_for_location(latitude: float, longitude: float) -> NSTimeZone:
with objc.autorelease_pool():
location = CLLocation.alloc().initWithLatitude_longitude_(latitude, longitude)
geocoder = CLGeocoder.alloc().init()
result = {"timezone": None, "error": None}
completed = False
def completion(placemarks, error):
nonlocal completed
if error:
result["error"] = error.localizedDescription()
else:
placemark = placemarks[0] if placemarks else None
if placemark and placemark.timeZone():
result["timezone"] = placemark.timeZone()
else:
result["error"] = "Unable to determine timezone"
completed = True
geocoder.reverseGeocodeLocation_completionHandler_(location, completion)
# reverseGeocodeLocation_completionHandler_ is async so run the event loop until completion
# I usuall use threading.Event for this type of thing in pyobjc but the the thread blocked forever
waiting = 0
while not completed:
NSRunLoop.currentRunLoop().runMode_beforeDate_(
"NSDefaultRunLoopMode",
NSDate.dateWithTimeIntervalSinceNow_(WAIT_FOR_COMPLETION),
)
waiting += WAIT_FOR_COMPLETION
if waiting >= COMPLETION_TIMEOUT:
raise TimeoutError(
f"Timeout waiting for completion of reverseGeocodeLocation_completionHandler_: {waiting} seconds"
)
time.sleep(WAIT_FOR_COMPLETION)
if result["error"]:
raise Exception(f"Error: {result['error']}")
return result["timezone"]
if __name__ == "__main__":
latitude, longitude = 34, -118 # Los Angeles
date = datetime.datetime.now()
try:
start_t = time.time_ns()
timezone = timezone_for_location(latitude, longitude)
offset = timezone.secondsFromGMTForDate_(
date
) # takes an NSDate but pybojc will convert
end_t = time.time_ns()
print(
f"Timezone: {timezone.name()}, offset: {offset}, took: {(end_t - start_t) / 1e6:.2f} ms"
)
except Exception as e:
print(f"Error: {e}")
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment