elahd / nyc311calendar

Simple fetcher for NYC 311 trash collection, school closing, and alternate side parking schedules.
MIT License
0 stars 0 forks source link

nyc311calendar

Python library for asynchronously fetching data for NYC school closures, trash collection holidays, and alternate side parking suspensions.


GitHub

Uses the NYC 311 Public API. Built to drive the Home Assistant NYC 311 Public Services Calendar add-in. ## Warning This is an alpha release. Expect breaking changes. I take no responsibility for parking tickets, overflowing trash cans, kids stranded at bus stops or missing exams, etc. 🤷🏼‍♂️ Use at your own risk. ## Usage ### First, install via pip ```bash pip install nyc311calendar ``` ### Then, get an API key An NYC API Portal developer account is required to use this library. 1. Sign up at https://api-portal.nyc.gov/signup/. 2. Log in, then subscribe to the "NYC 311 Public Developers" product at https://api-portal.nyc.gov/products?api=nyc-311-public-api. This subscription unlocks the calendar product. 3. Get your API key at https://api-portal.nyc.gov/developer. Either key (primary/secondary) will work. ### Finally, get data ```python # Import modules from nyc311calendar import CalendarType, NYC311API # Instantiate class calendar = NYC311API(session, API_KEY) # Fetch calendar resp = await calendar.get_calendar() ``` There's more to this. You'll need to provide an aiohttp ClientSession object. See the examples folder for guidance. ### Constants This library converts strings in the source API to constants wherever sensible and uses these constants everywhere (even as dictionary keys). That is, `"status": "CLOSED"` in the source API is represented as `'status_id': }` in this library, where Status is an enum in the CivCalNYC class. Constants are defined for: 1. Public services in `services.ServiceType`. 2. Service statuses in `services.StandardizedStatusTypes`. The source API is a mess of poorly named statuses. The StandardizedStatusTypes attempts to make sense of the underlying statuses. 3. Calendar types in `CalendarType`. See below for more info on calendar types. ### Calendar Day Entries Data for each calendar day is returned in a `CalendarDayEntry` dataclass. ### Calendar Types nyc311calendar can return data in several formats, each defined in the `CalendarType` enum: #### Quarter Ahead The Quarter Ahead calendar type returns all statuses for all services for 90 days starting on the day before the API request was made. The response dict has two sub-dicts keyed by calendar date and service. This is essentially a constant-ized dump from the source API. The example below is truncated to save space, showing two of 90 days. ```python async with aiohttp.ClientSession() as session: calendar = NYC311API(session, API_KEY) resp = await calendar.get_calendar( calendars=[CalendarType.QUARTER_AHEAD], scrub=True ) ``` ```python { : { : { datetime.date(2021, 12, 31): { : CalendarDayEntry(...), : CalendarDayEntry(...), : CalendarDayEntry(...) }, datetime.date(2022, 1, 1): { : CalendarDayEntry(...), : CalendarDayEntry(...), : CalendarDayEntry(...) } }, : { : { datetime.date(2021, 12, 31): CalendarDayEntry(...), datetime.date(2022, 1, 1): CalendarDayEntry(...) }, : { datetime.date(2021, 12, 31): CalendarDayEntry(...), datetime.date(2022, 1, 1): CalendarDayEntry(...) }, : { datetime.date(2021, 12, 31): CalendarDayEntry(...), datetime.date(2022, 1, 1): CalendarDayEntry(...) } } } } ``` #### Week Ahead The Week Ahead calendar type returns all statuses for all services for 8 days starting on the day before the API request was made. The response dict is keyed by number of days relative to today. This is useful for showing a calendar of the week ahead (and yesterday, just in case you forgot to move your car). The example below is truncated to save space, showing three of 90 days. ```python async with aiohttp.ClientSession() as session: calendar = NYC311API(session, API_KEY) resp = await calendar.get_calendar( calendars=[CalendarType.WEEK_AHEAD], scrub=True ) ``` ```python { : { -1: { 'date': datetime.date(2021, 12, 23), 'services': { : CalendarDayEntry(...), : CalendarDayEntry(...), : CalendarDayEntry(...) }, 0: { 'date': datetime.date(2021, 12, 24), 'services': { : CalendarDayEntry(...), : CalendarDayEntry(...), : CalendarDayEntry(...) }, 1: { 'date': datetime.date(2021, 12, 25), 'services': { : CalendarDayEntry(...), : CalendarDayEntry(...), : CalendarDayEntry(...) } } } ``` #### Next Exceptions The Next Exceptions calendar type returns the next date on which there is a service exception for either of the three covered services. The response dict is keyed by service type. This is useful when you're not interested in normal operations and only want to know, say, when the next school closure is. The example below shows the full response. Note that exceptions include things like holidays, snow days, half days, and winter break. ```python async with aiohttp.ClientSession() as session: calendar = NYC311API(session, API_KEY) resp = await calendar.get_calendar( calendars=[CalendarType.NEXT_EXCEPTIONS], scrub=True ) ``` ```python { : { : CalendarDayEntry(...), : CalendarDayEntry(...), : CalendarDayEntry(...) } } ```