Skip to content
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
79 changes: 79 additions & 0 deletions homeassistant/components/google_pubsub/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
import datetime
import json
import logging
import os
from typing import Dict, Any

import voluptuous as vol

import homeassistant.helpers.config_validation as cv
from homeassistant.const import (
EVENT_STATE_CHANGED, STATE_UNKNOWN, STATE_UNAVAILABLE)
from homeassistant.core import HomeAssistant, Event
from homeassistant.helpers.entityfilter import FILTER_SCHEMA
Comment thread
timvancann marked this conversation as resolved.

_LOGGER = logging.getLogger(__name__)

REQUIREMENTS = ['google-cloud-pubsub==0.39.1']

DOMAIN = 'google_pubsub'

CONF_PROJECT_ID = 'project_id'
CONF_TOPIC_NAME = 'topic_name'
CONF_SERVICE_PRINCIPAL = 'credentials_json'
CONF_FILTER = 'filter'

CONFIG_SCHEMA = vol.Schema({
DOMAIN: vol.Schema({
vol.Required(CONF_PROJECT_ID): cv.string,
vol.Required(CONF_TOPIC_NAME): cv.string,
vol.Required(CONF_SERVICE_PRINCIPAL): cv.string,
vol.Required(CONF_FILTER): FILTER_SCHEMA
}),
}, extra=vol.ALLOW_EXTRA)


def setup(hass: HomeAssistant, yaml_config: Dict[str, Any]):
Comment thread
timvancann marked this conversation as resolved.
from google.cloud import pubsub_v1

config = yaml_config.get(DOMAIN, {})

Copy link
Copy Markdown
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When would the DOMAIN key be missing in the config dict? Why do we need to use dict.get?

Copy link
Copy Markdown
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It shouldn't be missing. Changed it to dict[]

project_id = config[CONF_PROJECT_ID]
topic_name = config[CONF_TOPIC_NAME]
service_principal_path = os.path.join(hass.config.config_dir,
config[CONF_SERVICE_PRINCIPAL])

os.environ['GOOGLE_APPLICATION_CREDENTIALS'] = service_principal_path
Comment thread
timvancann marked this conversation as resolved.
Outdated

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there some way to pass this without modifying the global environment?

Copy link
Copy Markdown
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Did some digging, and yes, there is 😄. Fixed it


entities_filter = config[CONF_FILTER]

publisher = pubsub_v1.PublisherClient()
topic_path = publisher.topic_path(project_id, topic_name)
Comment thread
timvancann marked this conversation as resolved.
Outdated

encoder = DateTimeJSONEncoder()

def send_to_pubsub(event: Event):
Comment thread
timvancann marked this conversation as resolved.
state = event.data.get('new_state')
if (state is None
or state.state in (STATE_UNKNOWN, '', STATE_UNAVAILABLE)
or not entities_filter(state.entity_id)):
return

as_dict = state.as_dict()
data = json.dumps(
obj=as_dict,
default=encoder.encode
).encode('utf-8')

publisher.publish(topic_path, data=data)
Comment thread
timvancann marked this conversation as resolved.

hass.bus.listen(EVENT_STATE_CHANGED, send_to_pubsub)

return True


class DateTimeJSONEncoder(json.JSONEncoder):
Comment thread
timvancann marked this conversation as resolved.
def default(self, obj):
if isinstance(obj, datetime.datetime):
return obj.isoformat()
else:
return super(DateTimeJSONEncoder, self).default(obj)
3 changes: 3 additions & 0 deletions requirements_all.txt
Original file line number Diff line number Diff line change
Expand Up @@ -456,6 +456,9 @@ gntp==1.0.3
# homeassistant.components.google
google-api-python-client==1.6.4

# homeassistant.components.google_pubsub
google-cloud-pubsub==0.39.1

# homeassistant.components.sensor.google_travel_time
googlemaps==2.5.1

Expand Down
17 changes: 17 additions & 0 deletions tests/components/google_pubsub/test_pubsub.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
from datetime import datetime

from homeassistant.components.google_pubsub import (
DateTimeJSONEncoder as victim)


class TestDateTimeJSONEncoder(object):

def test_datetime(self):
time = datetime(2019, 1, 13, 12, 30, 5)
assert victim().encode(time) == '"2019-01-13T12:30:05"'

def test_no_datetime(self):
assert victim().encode(42) == '42'

def test_nested(self):
assert victim().encode({'foo': 'bar'}) == '{"foo": "bar"}'