From 61e729c2b37045f4a19b15865b11242c464c4416 Mon Sep 17 00:00:00 2001 From: Josh Date: Wed, 3 Feb 2016 11:56:08 +0100 Subject: [PATCH] Initial Commit --- .gitignore | 2 + CHANGELOG | 3 + Dockerfile | 16 + LICENSE | 201 +++++++++++ Packerfile.json | 71 ++++ README.md | 22 ++ VERSION | 1 + Vagrantfile | 33 ++ examples/example-web-page-1.html | 152 ++++++++ resources/configs/default-iglu-resolver.json | 18 + resources/configs/iglu-resolver.json | 32 ++ .../kinesis-elasticsearch-sink-good.hocon | 77 ++++ resources/configs/scala-kinesis-enrich.hocon | 70 ++++ .../configs/scala-stream-collector.hocon | 128 +++++++ resources/elasticsearch/bad-mapping.json | 41 +++ resources/elasticsearch/good-mapping.json | 328 ++++++++++++++++++ resources/event-dictionary/README.md | 162 +++++++++ .../com.example_company/example_event_1.json | 20 ++ .../example_event/jsonschema/1-0-0 | 32 ++ .../com.example_company/example_event_1.sql | 24 ++ resources/kibana/kibana4_init | 87 +++++ scripts/1_setup_docker.sh | 61 ++++ scripts/1_setup_packer.sh | 83 +++++ scripts/2_run_docker.sh | 40 +++ scripts/2_run_packer | 30 ++ scripts/elasticsearch_upload.pl | 15 + ui/setup-page.graffle | Bin 0 -> 3307 bytes ui/setup-page.png | Bin 0 -> 52804 bytes vagrant/.gitignore | 3 + vagrant/ansible.hosts | 2 + vagrant/peru.yaml | 14 + vagrant/push.bash | 97 ++++++ vagrant/up.bash | 50 +++ vagrant/up.guidance | 3 + vagrant/up.playbooks | 2 + 35 files changed, 1920 insertions(+) create mode 100644 .gitignore create mode 100644 CHANGELOG create mode 100644 Dockerfile create mode 100644 LICENSE create mode 100644 Packerfile.json create mode 100644 README.md create mode 100644 VERSION create mode 100644 Vagrantfile create mode 100644 examples/example-web-page-1.html create mode 100644 resources/configs/default-iglu-resolver.json create mode 100644 resources/configs/iglu-resolver.json create mode 100644 resources/configs/kinesis-elasticsearch-sink-good.hocon create mode 100644 resources/configs/scala-kinesis-enrich.hocon create mode 100644 resources/configs/scala-stream-collector.hocon create mode 100644 resources/elasticsearch/bad-mapping.json create mode 100644 resources/elasticsearch/good-mapping.json create mode 100644 resources/event-dictionary/README.md create mode 100644 resources/event-dictionary/jsonpaths/com.example_company/example_event_1.json create mode 100644 resources/event-dictionary/schemas/com.example_company/example_event/jsonschema/1-0-0 create mode 100644 resources/event-dictionary/sql/com.example_company/example_event_1.sql create mode 100755 resources/kibana/kibana4_init create mode 100755 scripts/1_setup_docker.sh create mode 100755 scripts/1_setup_packer.sh create mode 100755 scripts/2_run_docker.sh create mode 100755 scripts/2_run_packer create mode 100755 scripts/elasticsearch_upload.pl create mode 100644 ui/setup-page.graffle create mode 100644 ui/setup-page.png create mode 100644 vagrant/.gitignore create mode 100644 vagrant/ansible.hosts create mode 100644 vagrant/peru.yaml create mode 100755 vagrant/push.bash create mode 100755 vagrant/up.bash create mode 100644 vagrant/up.guidance create mode 100644 vagrant/up.playbooks diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..006849d6 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +.DS_Store* +.vagrant diff --git a/CHANGELOG b/CHANGELOG new file mode 100644 index 00000000..a00e2cdf --- /dev/null +++ b/CHANGELOG @@ -0,0 +1,3 @@ +Version 0.1.0 (2016-03-01) +-------------------------- +Initial release diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 00000000..28487835 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,16 @@ +FROM java:7 +MAINTAINER Snowplow Support + +EXPOSE 8080 +EXPOSE 5601 +EXPOSE 9200 + +ADD resources/kibana/kibana4_init /etc/init.d/kibana4_init +ADD resources/configs /home/ubuntu/snowplow/configs +ADD resources/elasticsearch /home/ubuntu/snowplow/elasticsearch +ADD scripts /home/ubuntu/snowplow/scripts + +RUN /home/ubuntu/snowplow/scripts/1_setup_docker.sh +RUN rm -rf /home/ubuntu/snowplow/staging + +CMD /home/ubuntu/snowplow/scripts/2_run_docker.sh diff --git a/LICENSE b/LICENSE new file mode 100644 index 00000000..5c304d1a --- /dev/null +++ b/LICENSE @@ -0,0 +1,201 @@ +Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/Packerfile.json b/Packerfile.json new file mode 100644 index 00000000..cabff773 --- /dev/null +++ b/Packerfile.json @@ -0,0 +1,71 @@ +{ + "builders": [ + { + "type": "amazon-ebs", + "region": "eu-west-1", + "source_ami": "ami-65cb5a12", + "instance_type": "t2.small", + "ssh_username": "ubuntu", + "ami_name": "snowplow-mini-0.1.0-{{ timestamp }}", + "ami_groups": [ "all" ], + "ami_regions": [ "us-east-1", "us-west-2", "us-west-1", "eu-central-1", "ap-southeast-1", "ap-northeast-1", "ap-southeast-2", "sa-east-1" ], + "ami_description": "SnowplowMini - The Snowplow Pipeline in a box", + "tags": { + "OS_Version": "Ubuntu-12.04", + "Release": "0.1.0" + } + } + ], + + "provisioners": [ + { + "type": "shell", + "inline": [ + "mkdir -p /home/ubuntu/snowplow/configs", + "mkdir -p /home/ubuntu/snowplow/elasticsearch", + "mkdir -p /home/ubuntu/snowplow/scripts" + ] + }, + { + "type": "file", + "source": "resources/kibana/kibana4_init", + "destination": "/home/ubuntu/kibana4_init" + }, + { + "type": "shell", + "inline": [ + "sudo cp /home/ubuntu/kibana4_init /etc/init.d", + "sudo chmod 0755 /etc/init.d/kibana4_init", + "sudo update-rc.d kibana4_init defaults 95 10" + ] + }, + { + "type": "file", + "source": "resources/configs", + "destination": "/home/ubuntu/snowplow" + }, + { + "type": "file", + "source": "resources/elasticsearch", + "destination": "/home/ubuntu/snowplow" + }, + { + "type": "file", + "source": "scripts", + "destination": "/home/ubuntu/snowplow" + }, + { + "type": "shell", + "inline": [ + "sudo cp /home/ubuntu/snowplow/scripts/2_run_packer /etc/init.d", + "sudo chmod 0755 /etc/init.d/2_run_packer", + "sudo update-rc.d 2_run_packer defaults 95 10" + ] + }, + { + "type": "shell", + "script": "scripts/1_setup_packer.sh", + "execute_command": "chmod +x {{ .Path }}; sh '{{ .Path }}'" + } + ] +} diff --git a/README.md b/README.md new file mode 100644 index 00000000..d4bc0dd2 --- /dev/null +++ b/README.md @@ -0,0 +1,22 @@ +# Snowplow-Mini + +An easily-deployable, single instance version of Snowplow that serves three use cases: + +1. Gives a Snowplow consumer (e.g. an analyst / data team / marketing team) a way to quickly understand what Snowplow "does" i.e. what you put it at one end and take out of the other +2. Gives developers new to Snowplow an easy way to start with Snowplow and understand how the different pieces fit together +3. Gives people running Snowplow a quick way to debug tracker updates (because they can ) + +## v1 + +The initial version of Snowplow-mini has only a limited subset of functionality: + +1. Data can be tracked in real time and loaded into Elasticsearch, where it can be queried (either directly or via Kibana) +2. Loading data into Redshift is not supported. (So this does not yet give analysts / data teams a good idea to understand what Snowplow "does") +3. No UI is provided to indicate what is happening with each of the different subsystems (collector, enrich etc.), so this does not provide developers a very good way of understanding how the different Snowplow subsystems work with one another +4. No validation is perfomed on the data, so this is not especially useful for Snowplow users who want to debug instrumentations of e.g. new trackers prior to pushing them live on Snowplow proper + +## Documentation + +1. [Quick start guide] [get-started-guide] + +[get-started-guide]: https://github.com/snowplow/snowplow-mini/wiki/Quickstart-guide diff --git a/VERSION b/VERSION new file mode 100644 index 00000000..6c6aa7cb --- /dev/null +++ b/VERSION @@ -0,0 +1 @@ +0.1.0 \ No newline at end of file diff --git a/Vagrantfile b/Vagrantfile new file mode 100644 index 00000000..f58d9176 --- /dev/null +++ b/Vagrantfile @@ -0,0 +1,33 @@ +Vagrant.configure("2") do |config| + + config.vm.box = "ubuntu/trusty64" + config.vm.hostname = "snowplow-mini" + config.ssh.forward_agent = true + + # Use NFS for shared folders for better performance + config.vm.network :private_network, ip: '192.168.50.50' # Uncomment to use NFS + config.vm.synced_folder '.', '/vagrant', nfs: true # Uncomment to use NFS + + config.vm.network "forwarded_port", guest: 9200, host: 9200 + config.vm.network "forwarded_port", guest: 5601, host: 5601 + config.vm.network "forwarded_port", guest: 8080, host: 8080 + + config.vm.provider :virtualbox do |vb| + vb.name = Dir.pwd().split("/")[-1] + "-" + Time.now.to_f.to_i.to_s + vb.customize ["modifyvm", :id, "--natdnshostresolver1", "on"] + vb.customize [ "guestproperty", "set", :id, "--timesync-threshold", 10000 ] + # Docker is quite hungry + vb.memory = 2048 + vb.cpus = 4 + end + + config.vm.provision :shell do |sh| + sh.path = "vagrant/up.bash" + end + + # Requires Vagrant 1.7.0+ + config.push.define "publish", strategy: "local-exec" do |push| + push.script = "vagrant/push.bash" + end + +end diff --git a/examples/example-web-page-1.html b/examples/example-web-page-1.html new file mode 100644 index 00000000..a920c63d --- /dev/null +++ b/examples/example-web-page-1.html @@ -0,0 +1,152 @@ + + + + + Example events for testing Snowplow mini + + + + + + + + + + + +

Send example events into Snowplow-mini

+ +

Note: before loading this page in your browser and firing events from it into Snowplow-mini, please make sure to replace all references to 'http://ec2-54-208-64-111.compute-1.amazonaws.com:8080' with your snowplow-mini public DNS.

+ +

Press the buttons below to trigger individual tracking events:
+
+
+
+ +

+ Link +
+ Ignored link + + + + \ No newline at end of file diff --git a/resources/configs/default-iglu-resolver.json b/resources/configs/default-iglu-resolver.json new file mode 100644 index 00000000..8ce1f6ac --- /dev/null +++ b/resources/configs/default-iglu-resolver.json @@ -0,0 +1,18 @@ +{ + "schema": "iglu:com.snowplowanalytics.iglu/resolver-config/jsonschema/1-0-0", + "data": { + "cacheSize": 500, + "repositories": [ + { + "name": "Iglu Central", + "priority": 0, + "vendorPrefixes": ["com.snowplowanalytics"], + "connection": { + "http": { + "uri": "http://iglucentral.com" + } + } + } + ] + } +} diff --git a/resources/configs/iglu-resolver.json b/resources/configs/iglu-resolver.json new file mode 100644 index 00000000..66132bad --- /dev/null +++ b/resources/configs/iglu-resolver.json @@ -0,0 +1,32 @@ +{ + "schema": "iglu:com.snowplowanalytics.iglu/resolver-config/jsonschema/1-0-0", + "data": { + "cacheSize": 500, + "repositories": [ + { + "name": "Iglu Central", + "priority": 1, + "vendorPrefixes": [ + "com.snowplowanalytics" + ], + "connection": { + "http": { + "uri": "http://iglucentral.com" + } + } + }, + { + "name": "Iglu snowplow mini usre", + "priority": 0, + "vendorPrefixes": [ + "com.snowplow-mini-user" + ], + "connection": { + "http": { + "uri": "https://s3.amazonaws.com/bucket-name-here" + } + } + } + ] + } +} diff --git a/resources/configs/kinesis-elasticsearch-sink-good.hocon b/resources/configs/kinesis-elasticsearch-sink-good.hocon new file mode 100644 index 00000000..a67d4e8b --- /dev/null +++ b/resources/configs/kinesis-elasticsearch-sink-good.hocon @@ -0,0 +1,77 @@ +# Default configuration for kinesis-elasticsearch-sink + +sink { + + # Sources currently supported are: + # 'kinesis' for reading records from a Kinesis stream + # 'stdin' for reading unencoded tab-separated events from stdin + # If set to "stdin", JSON documents will not be sent to Elasticsearch + # but will be written to stdout. + source = "stdin" + + # Sinks currently supported are: + # 'elasticsearch-kinesis' for writing good records to Elasticsearch and bad records to Kinesis + # 'stdouterr' for writing good records to stdout and bad records to stderr + sink = "elasticsearch-stderr" + + # The following are used to authenticate for the Amazon Kinesis sink. + # + # If both are set to 'default', the default provider chain is used + # (see http://docs.aws.amazon.com/AWSJavaSDK/latest/javadoc/com/amazonaws/auth/DefaultAWSCredentialsProviderChain.html) + # + # If both are set to 'iam', use AWS IAM Roles to provision credentials. + # + # If both are set to 'env', use environment variables AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY + aws { + access-key: "" + secret-key: "" + } + + kinesis { + + in { + stream-name: "" # Kinesis stream name + + # "good" for a stream of successfully enriched events + # "bad" for a stream of bad events + stream-type: "good" + + # LATEST: most recent data. + # TRIM_HORIZON: oldest available data. + # Note: This only affects the first run of this application + # on a stream. + initial-position: "TRIM_HORIZON" + } + + out { + # Stream for enriched events which are rejected by Elasticsearch + stream-name: "" + shards: 1 + } + + region: "" + + # "app-name" is used for a DynamoDB table to maintain stream state. + # You can set it automatically using: "SnowplowElasticsearchSink-$\\{connector.kinesis.in.stream-name\\}" + app-name: "" + } + + elasticsearch { + cluster-name: "elasticsearch" + endpoint: "localhost" + max-timeout: "10000" + index: "good" # Elasticsearch index name + type: "good" # Elasticsearch type name + } + + # Events are accumulated in a buffer before being sent to Elasticsearch. + # The buffer is emptied whenever: + # - the combined size of the stored records exceeds byte-limit or + # - the number of stored records exceeds record-limit or + # - the time in milliseconds since it was last emptied exceeds time-limit + buffer { + byte-limit: 5242880 + record-limit: 10000 + time-limit: 60000 + } +} diff --git a/resources/configs/scala-kinesis-enrich.hocon b/resources/configs/scala-kinesis-enrich.hocon new file mode 100644 index 00000000..616ddb42 --- /dev/null +++ b/resources/configs/scala-kinesis-enrich.hocon @@ -0,0 +1,70 @@ +# Default Configuration for Scala Kinesis Enrich. + +enrich { + # Sources currently supported are: + # 'kinesis' for reading Thrift-serialized records from a Kinesis stream + # 'stdin' for reading Base64-encoded Thrift-serialized records from stdin + source = "stdin" + + # Sinks currently supported are: + # 'kinesis' for writing enriched events to one Kinesis stream and invalid events to another. + # 'stdouterr' for writing enriched events to stdout and invalid events to stderr. + # Using "sbt assembly" and "java -jar" is recommended to disable sbt + # logging. + sink = "stdouterr" + + # AWS credentials + # + # If both are set to 'cpf', a properties file on the classpath is used. + # http://docs.aws.amazon.com/AWSJavaSDK/latest/javadoc/com/amazonaws/auth/ClasspathPropertiesFileCredentialsProvider.html + # + # If both are set to 'iam', use AWS IAM Roles to provision credentials. + # + # If both are set to 'env', use environment variables AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY + aws { + access-key: "" + secret-key: "" + } + + streams { + in: { + raw: "" + + # After enrichment, are accumulated in a buffer before being sent to Kinesis. + # The buffer is emptied whenever: + # - the number of stored records reaches record-limit or + # - the combined size of the stored records reaches byte-limit or + # - the time in milliseconds since it was last emptied exceeds time-limit when + # a new event enters the buffer + buffer: { + byte-limit: 4500000 + record-limit: 500 + time-limit: 5000 + } + } + + out: { + enriched: "" + bad: "" + + # Minimum and maximum backoff periods + # - Units: Milliseconds + backoffPolicy: { + minBackoff: 50 + maxBackoff: 500 + } + } + + # "app-name" is used for a DynamoDB table to maintain stream state. + # You can set it automatically using: "SnowplowKinesisEnrich-$\\{enrich.streams.in.raw\\}" + app-name: "" + + # LATEST: most recent data. + # TRIM_HORIZON: oldest available data. + # Note: This only effects the first run of this application + # on a stream. + initial-position = "TRIM_HORIZON" + + region: "" + } +} diff --git a/resources/configs/scala-stream-collector.hocon b/resources/configs/scala-stream-collector.hocon new file mode 100644 index 00000000..483ce88f --- /dev/null +++ b/resources/configs/scala-stream-collector.hocon @@ -0,0 +1,128 @@ +# Copyright (c) 2013-2014 Snowplow Analytics Ltd. All rights reserved. +# +# This program is licensed to you under the Apache License Version 2.0, and +# you may not use this file except in compliance with the Apache License +# Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +# http://www.apache.org/licenses/LICENSE-2.0. +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the Apache License Version 2.0 is distributed on an "AS +# IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or +# implied. See the Apache License Version 2.0 for the specific language +# governing permissions and limitations there under. + +# This file (application.conf.example) contains a template with +# configuration options for the Scala Stream Collector. +# +# To use, copy this to 'application.conf' and modify the configuration options. + +# 'collector' contains configuration options for the main Scala collector. +collector { + # The collector runs as a web service specified on the following + # interface and port. + interface = "0.0.0.0" + port = 8080 + + # Production mode disables additional services helpful for configuring and + # initializing the collector, such as a path '/dump' to view all + # records stored in the current stream. + production = true + + # Configure the P3P policy header. + p3p { + policyref = "/w3c/p3p.xml" + CP = "NOI DSP COR NID PSA OUR IND COM NAV STA" + } + + # The collector returns a cookie to clients for user identification + # with the following domain and expiration. + cookie { + # Set to 0 to disable the cookie + expiration = 365 days + # The domain is optional and will make the cookie accessible to other + # applications on the domain. Comment out this line to tie cookies to + # the collector's full domain + domain = "" + } + + # The collector has a configurable sink for storing data in + # different formats for the enrichment process. + sink { + # Sinks currently supported are: + # 'kinesis' for writing Thrift-serialized records to a Kinesis stream + # 'stdout' for writing Base64-encoded Thrift-serialized records to stdout + # Recommended settings for 'stdout' so each line printed to stdout + # is a serialized record are: + # 1. Setting 'akka.loglevel = OFF' and 'akka.loggers = []' + # to disable all logging. + # 2. Using 'sbt assembly' and 'java -jar ...' to disable + # sbt logging. + enabled = "stdout" + + kinesis { + thread-pool-size: 10 # Thread pool size for Kinesis API requests + + # The following are used to authenticate for the Amazon Kinesis sink. + # + # If both are set to 'cpf', a properties file on the classpath is used. + # http://docs.aws.amazon.com/AWSJavaSDK/latest/javadoc/com/amazonaws/auth/ClasspathPropertiesFileCredentialsProvider.html + # + # If both are set to 'iam', use AWS IAM Roles to provision credentials. + # + # If both are set to 'env', use environment variables AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY + aws { + access-key: "" + secret-key: "" + } + + # Data will be stored in the following stream. + stream { + region: "" + good: "" + bad: "" + } + + # Minimum and maximum backoff periods + backoffPolicy: { + minBackoff: 3000 # 3 seconds + maxBackoff: 600000 # 5 minutes + } + + # Incoming events are stored in a buffer before being sent to Kinesis. + # The buffer is emptied whenever: + # - the number of stored records reaches record-limit or + # - the combined size of the stored records reaches byte-limit or + # - the time in milliseconds since the buffer was last emptied reaches time-limit + buffer { + byte-limit: 4000000 + record-limit: 500 + time-limit: 5000 + } + } + } +} + +# Akka has a variety of possible configuration options defined at +# http://doc.akka.io/docs/akka/2.2.3/general/configuration.html. +akka { + loglevel = OFF # 'OFF' for no logging, 'DEBUG' for all logging. + loggers = ["akka.event.slf4j.Slf4jLogger"] +} + +# spray-can is the server the Stream collector uses and has configurable +# options defined at +# https://github.com/spray/spray/blob/master/spray-can/src/main/resources/reference.conf +spray.can.server { + # To obtain the hostname in the collector, the 'remote-address' header + # should be set. By default, this is disabled, and enabling it + # adds the 'Remote-Address' header to every request automatically. + remote-address-header = on + + uri-parsing-mode = relaxed + raw-request-uri-header = on + + # Define the maximum request length (the default is 2048) + parsing { + max-uri-length = 32768 + } +} diff --git a/resources/elasticsearch/bad-mapping.json b/resources/elasticsearch/bad-mapping.json new file mode 100644 index 00000000..b83c8454 --- /dev/null +++ b/resources/elasticsearch/bad-mapping.json @@ -0,0 +1,41 @@ +{ + "settings": { + "analysis": { + "analyzer": { + "default": { + "type": "keyword" + } + } + }, + "index" : { + "number_of_replicas" : "0", + "number_of_shards" : "2" + } + }, + "mappings": { + "bad": { + "_timestamp" : { + "enabled" : "yes", + "path" : "failure_tstamp" + }, + "_ttl": { + "enabled": true, + "default": "604800000" + }, + "properties": { + "errors": { + "type": "string", + "analyzer": "standard" + }, + "failure_tstamp": { + "type": "date", + "format": "dateOptionalTime" + }, + "line": { + "type": "string", + "analyzer": "standard" + } + } + } + } +} diff --git a/resources/elasticsearch/good-mapping.json b/resources/elasticsearch/good-mapping.json new file mode 100644 index 00000000..1102d531 --- /dev/null +++ b/resources/elasticsearch/good-mapping.json @@ -0,0 +1,328 @@ +{ + "settings": { + "analysis": { + "analyzer": { + "default": { + "type": "keyword" + } + } + }, + "index" : { + "number_of_replicas" : "0", + "number_of_shards" : "2" + } + }, + "mappings": { + "good": { + "_timestamp" : { + "enabled" : "yes", + "path" : "collector_tstamp" + }, + "_ttl": { + "enabled": true, + "default": "604800000" + }, + "properties": { + "app_id": { + "type": "string", + "index": "not_analyzed" + }, + "br_colordepth": { + "type": "string", + "index": "not_analyzed" + }, + "br_cookies": { + "type": "boolean" + }, + "br_family": { + "type": "string", + "index": "not_analyzed" + }, + "br_features_director": { + "type": "boolean" + }, + "br_features_flash": { + "type": "boolean" + }, + "br_features_gears": { + "type": "boolean" + }, + "br_features_java": { + "type": "boolean" + }, + "br_features_pdf": { + "type": "boolean" + }, + "br_features_quicktime": { + "type": "boolean" + }, + "br_features_realplayer": { + "type": "boolean" + }, + "br_features_silverlight": { + "type": "boolean" + }, + "br_features_windowsmedia": { + "type": "boolean" + }, + "br_lang": { + "type": "string", + "index": "not_analyzed" + }, + "br_name": { + "type": "string", + "index": "not_analyzed" + }, + "br_renderengine": { + "type": "string", + "index": "not_analyzed" + }, + "br_type": { + "type": "string", + "index": "not_analyzed" + }, + "br_version": { + "type": "string", + "index": "not_analyzed" + }, + "br_viewheight": { + "type": "long" + }, + "br_viewwidth": { + "type": "long" + }, + "collector_tstamp": { + "type": "date", + "format": "dateOptionalTime" + }, + "doc_charset": { + "type": "string", + "index": "not_analyzed" + }, + "doc_height": { + "type": "long" + }, + "doc_width": { + "type": "long" + }, + "domain_sessionid": { + "type": "string", + "index": "not_analyzed" + }, + "domain_sessionidx": { + "type": "long" + }, + "domain_userid": { + "type": "string", + "index": "not_analyzed" + }, + "dvce_ismobile": { + "type": "boolean" + }, + "dvce_screenheight": { + "type": "long" + }, + "dvce_screenwidth": { + "type": "long" + }, + "dvce_sent_tstamp": { + "type": "date", + "format": "dateOptionalTime" + }, + "dvce_tstamp": { + "type": "date", + "format": "dateOptionalTime" + }, + "dvce_type": { + "type": "string", + "index": "not_analyzed" + }, + "etl_tstamp": { + "type": "date", + "format": "dateOptionalTime" + }, + "event": { + "type": "string", + "index": "not_analyzed" + }, + "event_id": { + "type": "string", + "index": "not_analyzed" + }, + "geo_location": { + "type": "geo_point" + }, + "mkt_campaign": { + "type": "string", + "index": "not_analyzed" + }, + "mkt_content": { + "type": "string", + "index": "not_analyzed" + }, + "mkt_medium": { + "type": "string", + "index": "not_analyzed" + }, + "mkt_source": { + "type": "string", + "index": "not_analyzed" + }, + "mkt_term": { + "type": "string", + "index": "not_analyzed" + }, + "name_tracker": { + "type": "string", + "index": "not_analyzed" + }, + "network_userid": { + "type": "string", + "index": "not_analyzed" + }, + "os_family": { + "type": "string", + "index": "not_analyzed" + }, + "os_manufacturer": { + "type": "string", + "index": "not_analyzed" + }, + "os_name": { + "type": "string", + "index": "not_analyzed" + }, + "os_timezone": { + "type": "string", + "index": "not_analyzed" + }, + "page_referrer": { + "type": "string", + "index": "not_analyzed" + }, + "page_title": { + "type": "string", + "index": "not_analyzed" + }, + "page_url": { + "type": "string", + "index": "not_analyzed" + }, + "page_urlfragment": { + "type": "string", + "index": "not_analyzed" + }, + "page_urlhost": { + "type": "string", + "index": "not_analyzed" + }, + "page_urlpath": { + "type": "string", + "index": "not_analyzed" + }, + "page_urlport": { + "type": "long" + }, + "page_urlquery": { + "type": "string", + "index": "not_analyzed" + }, + "page_urlscheme": { + "type": "string", + "index": "not_analyzed" + }, + "platform": { + "type": "string", + "index": "not_analyzed" + }, + "pp_xoffset_max": { + "type": "long" + }, + "pp_xoffset_min": { + "type": "long" + }, + "pp_yoffset_max": { + "type": "long" + }, + "pp_yoffset_min": { + "type": "long" + }, + "refr_medium": { + "type": "string", + "index": "not_analyzed" + }, + "refr_source": { + "type": "string", + "index": "not_analyzed" + }, + "refr_term": { + "type": "string", + "index": "not_analyzed" + }, + "refr_urlfragment": { + "type": "string", + "index": "not_analyzed" + }, + "refr_urlhost": { + "type": "string", + "index": "not_analyzed" + }, + "refr_urlpath": { + "type": "string", + "index": "not_analyzed" + }, + "refr_urlport": { + "type": "long" + }, + "refr_urlquery": { + "type": "string", + "index": "not_analyzed" + }, + "refr_urlscheme": { + "type": "string", + "index": "not_analyzed" + }, + "se_action": { + "type": "string", + "index": "not_analyzed" + }, + "se_category": { + "type": "string", + "index": "not_analyzed" + }, + "se_label": { + "type": "string", + "index": "not_analyzed" + }, + "user_fingerprint": { + "type": "string", + "index": "not_analyzed" + }, + "user_id": { + "type": "string", + "index": "not_analyzed" + }, + "user_ipaddress": { + "type": "string", + "index": "not_analyzed" + }, + "useragent": { + "type": "string", + "index": "not_analyzed" + }, + "v_collector": { + "type": "string", + "index": "not_analyzed" + }, + "v_etl": { + "type": "string", + "index": "not_analyzed" + }, + "v_tracker": { + "type": "string", + "index": "not_analyzed" + } + } + } + } +} diff --git a/resources/event-dictionary/README.md b/resources/event-dictionary/README.md new file mode 100644 index 00000000..0a226b6a --- /dev/null +++ b/resources/event-dictionary/README.md @@ -0,0 +1,162 @@ +# example event dictionary + +## Overview + +Before you can send your own event and context types into Snowplow (using the track unstructured events and custom contexts features of Snowplow), you need to: + +1. Define a JSON schema for each of the events and context types +2. Upload those schemas to your Iglu schema repository +3. Define a corresponding jsonpath file, and make sure this is uploaded your jsonpaths directory in Amazon S3 +4. Create a corresponding Redshfit table definition, and create this table in your Redshift cluster + +Once you have completed the above, you can send in data that conforms to the schemas as custom unstructured events or custom contexts. + +## Prerequisites + +We recommend setting up the following 3 tools before staring: + +1. Git so you can easily clone the repo and make updates to it +2. [Schema Guru] [schema-guru-github]. This will auto-generate your jsonpath and sql table definitions +3. The [AWS CLI] [aws-cli]. This will make it easy to push updates to Iglu at the command line. + + +## 1. Creating the schemas + +In order to start sending a new event or context type into Snowplow, you first need to define a new schema for that event. + +1. Create a file in the repo for the new schema e.g. `/schemas/com.mycompany/new_event_or_context_name/jsonschema/1-0-0` +2. Create the schema in that file. Follow the `/schemas/com.example_company/example_event/jsonschema/1-0-0` +3. Save the file schema + +Note that if you have JSON data already and you want to create a corresponding schema, you can do so using [Schema Guru][schema-guru-online], both the [web UI][schema-guru-online] and the [CLI][schema-guru-github]. + +## 2. Uploading the schemas to Iglu + +Once you've created your schemas, you need to upload them to Iglu. In practice, this means copying them into S3. + +This can be done directly via the [AWS CLI](http://aws.amazon.com/cli/). In the project root, first commit the schema to Git: + +``` +git add . +git commit -m "Committed finalized schema" +git push +``` + +Then push it to Iglu. Note that as a trial user you will have to ask the Snowplow team to do this for you. As a Managed Services +customer you would be able to do it yourself as follows: + +``` +aws s3 cp schemas s3://{{ s3 bucket for schemas }}/schemas --include "*" --recursive +``` + +Useful resources + +* [Iglu schema repository 0.1.0 release blog post](http://snowplowanalytics.com/blog/2014/07/01/iglu-schema-repository-released/) +* [Iglu central](https://github.com/snowplow/iglu-central) - centralized repository for all the schemas hosted by the Snowplow team +* [Iglu](https://github.com/snowplow/iglu) - respository with both Iglu server and client libraries + + +## 3. Creating the jsonpath files and SQL table definitions + +Once you've defined the jsonschema for your new event or context type you need to create a correpsonding jsonpath file and sql table definition. This can be done programmatically using [Schema Guru] [schema-guru-github]. From the root of the repo: + +``` +/path/to/schema-guru-0.4.0 ddl --with-json-paths schemas/com.mycompany/new_event_or_context_name +``` + +A corresponding jsonpath file and sql table definition file will be generated in the appropriate folder in the repo. + +Note that you can create SQL table definition and jsonpath files for all the events / contexts schema'd as follows: + +``` +/path/to/schema-guru-0.4.0 ddl --with-json-paths schemas/com.mycompany +``` + + +## 4. Uploading the jsonpath files to Iglu + +One you've finalized the new jsonpath file, commit it to Git. From the project root: + +``` +git add . +git commit -m "Committed finalized jsonpath" +git push +``` + +Then push to Iglu. Again, you can only do this yourself as a Managed Services customers. As a trial user you will need to +ask a member of the Snowplow Analytics team to do this for you. + +``` +aws s3 cp jsonpaths s3://{{ s3 bucket for jsonpath files }}/jsonpaths --include "*" --recursive +``` + +## 5. Creating or updating the table definition in Redshift + +Once you've committed your updated table definition into Github, you need to either create or modify the table in Redshift, either by executing the `CREATE TABLE...` statement directly, or `ALTER TABLE` (if you're e.g. adding a column to an existing table). + +## 6. Sending data into Snowplow using the schema reference as custom unstructured events or contexts + +Once you have gone through the above process, you can start sending data that conforms to the schema(s) you've created into Snowplow as unstructured events and custom contexts. + +In both cases (custom unstructured events and contexts), the data is sent in as a JSON with two fields, a schema field with a reference to the location of the schema in Iglu, and a data field, with the actual data being sent, e.g. + +```json +{ + "schema": "iglu: com.acme_company/viewed_product/jsonschema/2-0-0", + "data": { + "productId": "ASO01043", + "category": "Dresses", + "brand": "ACME", + "price": 49.95, + "sizes": [ + "xs", + "s", + "l", + "xl", + "xxl" + ], + "availableSince": "2013-03-07" + } +} +``` + +For more detail, please see the technical documentation for the specific tracker you're implementing. + +Note: we recommend testing that the data you're sending into Snowplow conforms to the schemas you've defined and uploaded into Iglu, before pushing updates into production. This [online JSON schema validator](http://jsonschemalint.com/draft4/) is a very useful resource for doing so. + +## 7. Managing schema migrations + +When you use Snowplow, the schema for each event and context lives with the data. That means you have the flexibility to evolve your schema definition over time. + +If you want to change your schema over time, you will need to: + +1. Create a new jsonschema file. Depending on how different this is to your current version, you will need to give it the appropriate version number. The [SchemaVer][schema-ver] specification we use when versioning data schemas can be found [here][schema-ver] +2. Update the corresponding jsonpath files. If you've created a new major schema version, you'll need to create a new jsonpath file e.g. `exmaple_event_2.json`, that exists alongside your existing `example_event_1.json` +3. For minor schema updates, you should be able to update your existing Redshift table definition e.g. to add add additional columns. For major schema updates, you'll need to create a new Redshift table definition e.g. `com_mycompany_exmaple_event_2.sql` +4. Start sending data into Snowplow using the new schema version (i.e. update the Iglu reference to point at the new version e.g. `2-0-0` or `1-0-1` rather than `1-0-0`). Note that you will continue to be able to send in data that conforms to the old schema at the same time. In the event that you have an event with two different major schema definitions, each event version will be loaded into a different Redshift table + +## Additional resources + +Documentation on jsonschemas: + +* Other example jsonschemas can be found in [Iglu Central](https://github.com/snowplow/iglu-central/tree/master/schemas). Note how schemas are namespaced in different folders +* [Schema Guru] [schema-guru-online] is an [online] [schema-guru-online] and [command line tool] [schema-guru-github] for programmatically generating schemas from existing JSON data +* [Snowplow 0.9.5 release blog post](http://snowplowanalytics.com/blog/2014/07/09/snowplow-0.9.5-released-with-json-validation-shredding/), which gives an overview of the way that Snowplow uses jsonschemas to process, validate and shred unstructured event and custom context JSONs +* It can be useful to test jsonschemas using online validators e.g. [this one](http://jsonschemalint.com/draft4/) +* [json-schema.org](http://json-schema.org/) contains links to the actual jsonschema specification, examples and guide for schema authors +* The original specification for self-describing JSONs, produced by the Snowplow team, can be found [here](http://snowplowanalytics.com/blog/2014/05/15/introducing-self-describing-jsons/) + +Documentation on jsonpaths: + +* Example jsonpath files can be found on the [Snowplow repo](https://github.com/snowplow/snowplow/tree/master/4-storage/redshift-storage/jsonpaths). Note that the corresponding jsonschema definitions are stored in [Iglu central](https://github.com/snowplow/iglu-central/tree/master/schemas) +* Amazon documentation on jsonpath files can be found [here](http://docs.aws.amazon.com/redshift/latest/dg/copy-usage_notes-copy-from-json.html) + +Documentaiton on creating tablels in Redshift: + +* Example Redshift table definitions can be found on the [Snowplow repo](https://github.com/snowplow/snowplow/tree/master/4-storage/redshift-storage/sql). Note that corresponding jsonschema definitions are stored in [Iglu central](https://github.com/snowplow/iglu-central/tree/master/schemas) +* Amazon documentation on Redshift create table statements can be found [here](http://docs.aws.amazon.com/redshift/latest/dg/r_CREATE_TABLE_NEW.html). A list of Redshift data types can be found [here](http://docs.aws.amazon.com/redshift/latest/dg/c_Supported_data_types.html) + +[schema-guru-online]: http://schemaguru.snowplowanalytics.com/ +[schema-guru-github]: https://github.com/snowplow/schema-guru?_sp=44dbe9a530cc476d.1436355830779 +[aws-cli]: https://aws.amazon.com/cli/ +[schema-ver]: http://snowplowanalytics.com/blog/2014/05/13/introducing-schemaver-for-semantic-versioning-of-schemas/ diff --git a/resources/event-dictionary/jsonpaths/com.example_company/example_event_1.json b/resources/event-dictionary/jsonpaths/com.example_company/example_event_1.json new file mode 100644 index 00000000..efa95da7 --- /dev/null +++ b/resources/event-dictionary/jsonpaths/com.example_company/example_event_1.json @@ -0,0 +1,20 @@ +{ + "jsonpaths": [ + + "$.schema.vendor", + "$.schema.name", + "$.schema.format", + "$.schema.version", + + "$.hierarchy.rootId", + "$.hierarchy.rootTstamp", + "$.hierarchy.refRoot", + "$.hierarchy.refTree", + "$.hierarchy.refParent", + + "$.data.exampleStringField", + "$.data.exampleIntegerField", + "$.data.exampleNumericField", + "$.data.exampleTimestampField" + ] +} \ No newline at end of file diff --git a/resources/event-dictionary/schemas/com.example_company/example_event/jsonschema/1-0-0 b/resources/event-dictionary/schemas/com.example_company/example_event/jsonschema/1-0-0 new file mode 100644 index 00000000..53f04516 --- /dev/null +++ b/resources/event-dictionary/schemas/com.example_company/example_event/jsonschema/1-0-0 @@ -0,0 +1,32 @@ +{ + "$schema": "http://iglucentral.com/schemas/com.snowplowanalytics.self-desc/schema/jsonschema/1-0-0#", + "description": "Schema for an example event", + "self": { + "vendor": "com.example_company", + "name": "example_event", + "format": "jsonschema", + "version": "1-0-0" + }, + + "type": "object", + "properties": { + "exampleStringField": { + "type": "string", + "maxLength": 255 + }, + "exampleIntegerField": { + "type": "integer" + }, + "exampleNumericField": { + "type": "number", + "maxDecimal": 3 + }, + "exampleTimestampField": { + "type": "string", + "format": "date-time" + } + }, + "minProperties":1, + "required": ["exampleStringField", "exampleIntegerField"], + "additionalProperties": false +} diff --git a/resources/event-dictionary/sql/com.example_company/example_event_1.sql b/resources/event-dictionary/sql/com.example_company/example_event_1.sql new file mode 100644 index 00000000..37649252 --- /dev/null +++ b/resources/event-dictionary/sql/com.example_company/example_event_1.sql @@ -0,0 +1,24 @@ +-- Compatibility: iglu:com.example_company/example_event/jsonschema/1-0-0 + +CREATE TABLE atomic.com_example_company_example_event_1 ( + -- Schema of this type + schema_vendor varchar(128) encode runlength not null, + schema_name varchar(128) encode runlength not null, + schema_format varchar(128) encode runlength not null, + schema_version varchar(128) encode runlength not null, + -- Parentage of this type + root_id char(36) encode raw not null, + root_tstamp timestamp encode raw not null, + ref_root varchar(255) encode runlength not null, + ref_tree varchar(1500) encode runlength not null, + ref_parent varchar(255) encode runlength not null, + -- Properties of this type + example_string_field varchar(255) not null, + example_integer_field integer not null, + example_numeric_field decimal(8,2), + example_timestamp_field timestamp +) +DISTSTYLE KEY +-- Optimized join to atomic.events +DISTKEY (root_id) +SORTKEY (root_tstamp); \ No newline at end of file diff --git a/resources/kibana/kibana4_init b/resources/kibana/kibana4_init new file mode 100755 index 00000000..da5b12da --- /dev/null +++ b/resources/kibana/kibana4_init @@ -0,0 +1,87 @@ +#!/bin/sh +# +# /etc/init.d/kibana4_init -- startup script for kibana4 +# bsmith@the408.com 2015-02-20; used elasticsearch init script as template +# https://github.com/akabdog/scripts/edit/master/kibana4_init +# +### BEGIN INIT INFO +# Provides: kibana4_init +# Required-Start: $network $remote_fs $named +# Required-Stop: $network $remote_fs $named +# Default-Start: 2 3 4 5 +# Default-Stop: 0 1 6 +# Short-Description: Starts kibana4_init +# Description: Starts kibana4_init using start-stop-daemon +### END INIT INFO + +#configure this with wherever you unpacked kibana: +KIBANA_BIN=/opt/kibana/bin + +NAME=kibana4 +PID_FILE=/var/run/$NAME.pid +PATH=/bin:/usr/bin:/sbin:/usr/sbin:$KIBANA_BIN +DAEMON=$KIBANA_BIN/kibana +DESC="Kibana4" + +if [ `id -u` -ne 0 ]; then + echo "You need root privileges to run this script" + exit 1 +fi + +. /lib/lsb/init-functions + +if [ -r /etc/default/rcS ]; then + . /etc/default/rcS +fi + +case "$1" in + start) + log_daemon_msg "Starting $DESC" + + pid=`pidofproc -p $PID_FILE kibana` + if [ -n "$pid" ] ; then + log_begin_msg "Already running." + log_end_msg 0 + exit 0 + fi + + # Start Daemon + start-stop-daemon --start --pidfile "$PID_FILE" --make-pidfile --background --exec $DAEMON + log_end_msg $? + ;; + stop) + log_daemon_msg "Stopping $DESC" + + if [ -f "$PID_FILE" ]; then + start-stop-daemon --stop --pidfile "$PID_FILE" \ + --retry=TERM/20/KILL/5 >/dev/null + if [ $? -eq 1 ]; then + log_progress_msg "$DESC is not running but pid file exists, cleaning up" + elif [ $? -eq 3 ]; then + PID="`cat $PID_FILE`" + log_failure_msg "Failed to stop $DESC (pid $PID)" + exit 1 + fi + rm -f "$PID_FILE" + else + log_progress_msg "(not running)" + fi + log_end_msg 0 + ;; + status) + status_of_proc -p $PID_FILE kibana kibana && exit 0 || exit $? + ;; + restart|force-reload) + if [ -f "$PID_FILE" ]; then + $0 stop + sleep 1 + fi + $0 start + ;; + *) + log_success_msg "Usage: $0 {start|stop|restart|force-reload|status}" + exit 1 + ;; +esac + +exit 0 diff --git a/scripts/1_setup_docker.sh b/scripts/1_setup_docker.sh new file mode 100755 index 00000000..5025b05f --- /dev/null +++ b/scripts/1_setup_docker.sh @@ -0,0 +1,61 @@ +#!/bin/bash -e + +apt-get update +apt-get install -y unzip + +############# +# Constants # +############# + +main_dir=/home/ubuntu/snowplow + +configs_dir=$main_dir/configs +staging_dir=$main_dir/staging +executables_dir=$main_dir/bin +unix_pipes_dir=$main_dir/pipes +es_dir=$main_dir/elasticsearch +scripts_dir=$main_dir/scripts + +raw_events_pipe=$unix_pipes_dir/raw-events-pipe +enriched_pipe=$unix_pipes_dir/enriched-events-pipe + +kinesis_package=snowplow_kinesis_r67_bohemian_waxwing.zip +kibana_v=4.0.1 + +########################### +# Setup Directories/Files # +########################### + +mkdir -p $configs_dir +mkdir -p $staging_dir +mkdir -p $executables_dir +mkdir -p $unix_pipes_dir +mkdir -p $es_dir +mkdir -p $scripts_dir + +mkfifo $raw_events_pipe +mkfifo $enriched_pipe + +################################ +# Install Kinesis Applications # +################################ + +wget http://dl.bintray.com/snowplow/snowplow-generic/${kinesis_package} -P $staging_dir +unzip $staging_dir/${kinesis_package} -d $executables_dir + +######################### +# Install Elasticsearch # +######################### + +wget -qO - https://packages.elastic.co/GPG-KEY-elasticsearch | apt-key add - +echo "deb http://packages.elastic.co/elasticsearch/1.4/debian stable main" | tee -a /etc/apt/sources.list +apt-get update -y && apt-get install elasticsearch -y +/usr/share/elasticsearch/bin/plugin --install mobz/elasticsearch-head + +################## +# Install Kibana # +################## + +wget "https://download.elasticsearch.org/kibana/kibana/kibana-${kibana_v}-linux-x64.zip" -P $staging_dir +unzip $staging_dir/kibana-${kibana_v}-linux-x64.zip -d /opt/ +ln -s /opt/kibana-${kibana_v}-linux-x64 /opt/kibana diff --git a/scripts/1_setup_packer.sh b/scripts/1_setup_packer.sh new file mode 100755 index 00000000..60fb844d --- /dev/null +++ b/scripts/1_setup_packer.sh @@ -0,0 +1,83 @@ +#!/bin/bash -e + +sudo apt-get update +sudo apt-get install -y unzip + +############# +# Constants # +############# + +main_dir=/home/ubuntu/snowplow + +configs_dir=$main_dir/configs +staging_dir=$main_dir/staging +executables_dir=$main_dir/bin +unix_pipes_dir=$main_dir/pipes +es_dir=$main_dir/elasticsearch +scripts_dir=$main_dir/scripts + +raw_events_pipe=$unix_pipes_dir/raw-events-pipe +enriched_pipe=$unix_pipes_dir/enriched-events-pipe + +kinesis_package=snowplow_kinesis_r67_bohemian_waxwing.zip +kibana_v=4.0.1 + +########################### +# Setup Directories/Files # +########################### + +mkdir -p $configs_dir +mkdir -p $staging_dir +mkdir -p $executables_dir +mkdir -p $unix_pipes_dir +mkdir -p $es_dir +mkdir -p $scripts_dir + +mkfifo $raw_events_pipe +mkfifo $enriched_pipe + +################## +# Install Java 7 # +################## + +sudo add-apt-repository ppa:webupd8team/java -y +sudo apt-get update +echo oracle-java7-installer shared/accepted-oracle-license-v1-1 select true | sudo /usr/bin/debconf-set-selections +sudo apt-get install oracle-java7-installer -y + +################################ +# Install Kinesis Applications # +################################ + +wget http://dl.bintray.com/snowplow/snowplow-generic/${kinesis_package} -P $staging_dir +unzip $staging_dir/${kinesis_package} -d $executables_dir + +######################### +# Install Elasticsearch # +######################### + +wget -qO - https://packages.elastic.co/GPG-KEY-elasticsearch | sudo apt-key add - +echo "deb http://packages.elastic.co/elasticsearch/1.4/debian stable main" | sudo tee -a /etc/apt/sources.list +sudo apt-get update -y && sudo apt-get install elasticsearch -y +sudo update-rc.d elasticsearch defaults 95 10 +sudo /usr/share/elasticsearch/bin/plugin --install mobz/elasticsearch-head + +################## +# Install Kibana # +################## + +wget "https://download.elasticsearch.org/kibana/kibana/kibana-${kibana_v}-linux-x64.zip" -P $staging_dir +sudo unzip $staging_dir/kibana-${kibana_v}-linux-x64.zip -d /opt/ +sudo ln -s /opt/kibana-${kibana_v}-linux-x64 /opt/kibana + +sudo chown -R ubuntu:ubuntu $main_dir + +################ +# Add Mappings # +################ + +sudo service elasticsearch start +sleep 15 + +curl -XPUT 'http://localhost:9200/good' -d @${es_dir}/good-mapping.json +curl -XPUT 'http://localhost:9200/bad' -d @${es_dir}/bad-mapping.json diff --git a/scripts/2_run_docker.sh b/scripts/2_run_docker.sh new file mode 100755 index 00000000..bdf02eed --- /dev/null +++ b/scripts/2_run_docker.sh @@ -0,0 +1,40 @@ +#!/bin/bash -e + +############# +# Constants # +############# + +main_dir=/home/ubuntu/snowplow + +configs_dir=$main_dir/configs +executables_dir=$main_dir/bin +unix_pipes_dir=$main_dir/pipes +es_dir=$main_dir/elasticsearch +scripts_dir=$main_dir/scripts + +raw_events_pipe=$unix_pipes_dir/raw-events-pipe +enriched_pipe=$unix_pipes_dir/enriched-events-pipe + +##################### +# Start ES + Kibana # +##################### + +service elasticsearch start +service kibana4_init start + +sleep 15 + +################ +# Add Mappings # +################ + +curl -XPUT 'http://localhost:9200/good' -d @${es_dir}/good-mapping.json +curl -XPUT 'http://localhost:9200/bad' -d @${es_dir}/bad-mapping.json + +################################################# +# Start Collector/Enrichment/Elasticsearch Sink # +################################################# + +${executables_dir}/snowplow-stream-collector-0.5.0 --config ${configs_dir}/scala-stream-collector.hocon > $raw_events_pipe & +cat $raw_events_pipe | ${executables_dir}/snowplow-kinesis-enrich-0.6.0 --config ${configs_dir}/scala-kinesis-enrich.hocon --resolver file:${configs_dir}/default-iglu-resolver.json > $enriched_pipe & +cat $enriched_pipe | ${executables_dir}/snowplow-elasticsearch-sink-0.4.0 --config ${configs_dir}/kinesis-elasticsearch-sink-good.hocon | ${scripts_dir}/elasticsearch_upload.pl diff --git a/scripts/2_run_packer b/scripts/2_run_packer new file mode 100755 index 00000000..be7d5da1 --- /dev/null +++ b/scripts/2_run_packer @@ -0,0 +1,30 @@ +#!/bin/bash -e + +############# +# Constants # +############# + +main_dir=/home/ubuntu/snowplow + +configs_dir=$main_dir/configs +executables_dir=$main_dir/bin +unix_pipes_dir=$main_dir/pipes +scripts_dir=$main_dir/scripts + +raw_events_pipe=$unix_pipes_dir/raw-events-pipe +enriched_pipe=$unix_pipes_dir/enriched-events-pipe + +##################### +# Start ES + Kibana # +##################### + +sudo service elasticsearch start +sudo service kibana4_init start + +################################################# +# Start Collector/Enrichment/Elasticsearch Sink # +################################################# + +${executables_dir}/snowplow-stream-collector-0.5.0 --config ${configs_dir}/scala-stream-collector.hocon > $raw_events_pipe & +cat $raw_events_pipe | ${executables_dir}/snowplow-kinesis-enrich-0.6.0 --config ${configs_dir}/scala-kinesis-enrich.hocon --resolver file:${configs_dir}/default-iglu-resolver.json > $enriched_pipe & +cat $enriched_pipe | ${executables_dir}/snowplow-elasticsearch-sink-0.4.0 --config ${configs_dir}/kinesis-elasticsearch-sink-good.hocon | ${scripts_dir}/elasticsearch_upload.pl & diff --git a/scripts/elasticsearch_upload.pl b/scripts/elasticsearch_upload.pl new file mode 100755 index 00000000..48a30cb0 --- /dev/null +++ b/scripts/elasticsearch_upload.pl @@ -0,0 +1,15 @@ +#!/usr/bin/perl -w + +use strict; + +sub send_to_elastic { + my $to_send = $_[0]; + `curl -XPOST http://localhost:9200/good/good -d '$to_send'` +} + +while () +{ + chomp; + my $ln = $_; + send_to_elastic($ln); +} diff --git a/ui/setup-page.graffle b/ui/setup-page.graffle new file mode 100644 index 0000000000000000000000000000000000000000..a8e751e624f4c8bcde2a2ba99cf9f6c7797ce00a GIT binary patch literal 3307 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 literal 0 HcmV?d00001 diff --git a/ui/setup-page.png b/ui/setup-page.png new file mode 100644 index 0000000000000000000000000000000000000000..08ab149f850d61572f18d5bf5d1a3c130cb1f859 GIT binary patch literal 52804 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 Zm`1M@OH?nbpO3)5_aZXFg@Srs{|kGNKU)9* literal 0 HcmV?d00001 diff --git a/vagrant/.gitignore b/vagrant/.gitignore new file mode 100644 index 00000000..1b4b29ff --- /dev/null +++ b/vagrant/.gitignore @@ -0,0 +1,3 @@ +.peru +oss-playbooks +ansible diff --git a/vagrant/ansible.hosts b/vagrant/ansible.hosts new file mode 100644 index 00000000..588fa08c --- /dev/null +++ b/vagrant/ansible.hosts @@ -0,0 +1,2 @@ +[vagrant] +127.0.0.1:2222 diff --git a/vagrant/peru.yaml b/vagrant/peru.yaml new file mode 100644 index 00000000..e7fdf41c --- /dev/null +++ b/vagrant/peru.yaml @@ -0,0 +1,14 @@ +imports: + ansible: ansible + ansible_playbooks: oss-playbooks + +curl module ansible: + # Equivalent of git cloning tags/v1.6.6 but much, much faster + url: https://codeload.github.com/ansible/ansible/zip/69d85c22c7475ccf8169b6ec9dee3ee28c92a314 + unpack: zip + export: ansible-69d85c22c7475ccf8169b6ec9dee3ee28c92a314 + +git module ansible_playbooks: + url: https://github.com/snowplow/ansible-playbooks.git + # Comment out to fetch a specific rev instead of master: + # rev: xxx diff --git a/vagrant/push.bash b/vagrant/push.bash new file mode 100755 index 00000000..4c7a1384 --- /dev/null +++ b/vagrant/push.bash @@ -0,0 +1,97 @@ +#!/bin/bash +set -e + +bintray_user=snowplowbot +bintray_repository=snowplow-docker-snowplow-docker.bintray.io +bintray_email=systems@snowplowanalytics.com +img_name=generic/snowplow-mini +tar_name=generic-snowplow-mini + +# Similar to Perl die +function die() { + echo "$@" 1>&2 ; exit 1; +} + +# Check if our Vagrant box is running. Expects `vagrant status` to look like: +# +# > Current machine states: +# > +# > default poweroff (virtualbox) +# > +# > The VM is powered off. To restart the VM, simply run `vagrant up` +# +# Parameters: +# 1. out_running (out parameter) +function is_running { + [ "$#" -eq 1 ] || die "1 argument required, $# provided" + local __out_running=$1 + + set +e + vagrant status | sed -n 3p | grep -q "^default\s*running (virtualbox)$" + local retval=${?} + set -e + if [ ${retval} -eq "0" ] ; then + eval ${__out_running}=1 + else + eval ${__out_running}=0 + fi +} + +# Get version, checking we are on the latest +# +# Parameters: +# 1. out_version (out parameter) +# 2. out_error (out parameter) +function get_version { + [ "$#" -eq 2 ] || die "2 arguments required, $# provided" + local __out_version=$1 + local __out_error=$2 + + file_version=`cat VERSION` + expected_tag="$file_version" + tag_version=`git describe --abbrev=0 --tags` + if [ ${expected_tag} != ${tag_version} ] ; then + eval ${__out_error}="'File version ${expected_tag} != tag version ${tag_version}'" + else + eval ${__out_version}=${expected_tag} + fi +} + +# Go to parent-parent dir of this script +function cd_root() { + source="${BASH_SOURCE[0]}" + while [ -h "${source}" ] ; do source="$(readlink "${source}")"; done + dir="$( cd -P "$( dirname "${source}" )/.." && pwd )" + cd ${dir} +} + +cd_root + +# Precondition for running +running=0 && is_running "running" +[ ${running} -eq 1 ] || die "Vagrant guest must be running to push" + +# Git tag must match version in package.json +version=`cat VERSION` +#version="" && error="" && get_version "version" "error" +#[ "${error}" ] && die "Versions don't match: ${error}. Are you trying to publish an old version, or maybe on the wrong branch?" + +# Can't pass args thru vagrant push so have to prompt +read -e -p "Please enter API key for Bintray user ${bintray_user}: " bintray_api_key + +# Build Docker Image +cmd="cd /vagrant && sudo docker build -t ${img_name}:${version} ." +vagrant ssh -c "${cmd}" + +# Get Image ID +cmd="sudo docker images | grep \"^${img_name}\" -m 1 | awk '{print \$3}'" +img_id=`vagrant ssh -c "${cmd}"` +[ "${img_id}" != "" ] || die "Image ID not found cannot push to Bintray." + +# Upload to Bintray +cmd="sudo docker login -u ${bintray_user} -p ${bintray_api_key} -e ${bintray_email} ${bintray_repository} && \ + sudo docker tag ${img_id:0:12} ${bintray_repository}/${img_name}:${version} && \ + sudo docker push ${bintray_repository}/${img_name}:${version}" +vagrant ssh -c "${cmd}" + +exit 0 diff --git a/vagrant/up.bash b/vagrant/up.bash new file mode 100755 index 00000000..7450ae89 --- /dev/null +++ b/vagrant/up.bash @@ -0,0 +1,50 @@ +#!/bin/bash +set -e + +vagrant_dir=/vagrant/vagrant +bashrc=/home/vagrant/.bashrc + +echo "========================================" +echo "INSTALLING PERU AND ANSIBLE DEPENDENCIES" +echo "----------------------------------------" +apt-get update +apt-get install -y language-pack-en git unzip libyaml-dev python3-pip python-yaml python-paramiko python-jinja2 + +echo "===============" +echo "INSTALLING PERU" +echo "---------------" +sudo pip3 install peru + +echo "=======================================" +echo "CLONING ANSIBLE AND PLAYBOOKS WITH PERU" +echo "---------------------------------------" +cd ${vagrant_dir} && peru sync -v +echo "... done" + +env_setup=${vagrant_dir}/ansible/hacking/env-setup +hosts=${vagrant_dir}/ansible.hosts + +echo "===================" +echo "CONFIGURING ANSIBLE" +echo "-------------------" +touch ${bashrc} +echo "source ${env_setup}" >> ${bashrc} +echo "export ANSIBLE_HOSTS=${hosts}" >> ${bashrc} +echo "... done" + +echo "==========================================" +echo "RUNNING PLAYBOOKS WITH ANSIBLE*" +echo "* no output while each playbook is running" +echo "------------------------------------------" +while read pb; do + su - -c "source ${env_setup} && ${vagrant_dir}/ansible/bin/ansible-playbook ${vagrant_dir}/${pb} --connection=local --inventory-file=${hosts}" vagrant +done <${vagrant_dir}/up.playbooks + +guidance=${vagrant_dir}/up.guidance + +if [ -f ${guidance} ]; then + echo "===========" + echo "PLEASE READ" + echo "-----------" + cat $guidance +fi diff --git a/vagrant/up.guidance b/vagrant/up.guidance new file mode 100644 index 00000000..0575dbc6 --- /dev/null +++ b/vagrant/up.guidance @@ -0,0 +1,3 @@ +To get started: +vagrant ssh +cd /vagrant diff --git a/vagrant/up.playbooks b/vagrant/up.playbooks new file mode 100644 index 00000000..c59a8819 --- /dev/null +++ b/vagrant/up.playbooks @@ -0,0 +1,2 @@ +oss-playbooks/packer.yml +oss-playbooks/docker.yml