| LEFT | RIGHT | 
|---|
| 1 # coding: utf-8 | 1 # coding: utf-8 | 
| 2 | 2 | 
| 3 # This file is part of the Adblock Plus web scripts, | 3 # This file is part of the Adblock Plus web scripts, | 
| 4 # Copyright (C) 2006-2015 Eyeo GmbH | 4 # Copyright (C) 2006-2015 Eyeo GmbH | 
| 5 # | 5 # | 
| 6 # Adblock Plus is free software: you can redistribute it and/or modify | 6 # Adblock Plus is free software: you can redistribute it and/or modify | 
| 7 # it under the terms of the GNU General Public License version 3 as | 7 # it under the terms of the GNU General Public License version 3 as | 
| 8 # published by the Free Software Foundation. | 8 # published by the Free Software Foundation. | 
| 9 # | 9 # | 
| 10 # Adblock Plus is distributed in the hope that it will be useful, | 10 # Adblock Plus is distributed in the hope that it will be useful, | 
| 11 # but WITHOUT ANY WARRANTY; without even the implied warranty of | 11 # but WITHOUT ANY WARRANTY; without even the implied warranty of | 
| 12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
| 13 # GNU General Public License for more details. | 13 # GNU General Public License for more details. | 
| 14 # | 14 # | 
| 15 # You should have received a copy of the GNU General Public License | 15 # You should have received a copy of the GNU General Public License | 
| 16 # along with Adblock Plus.  If not, see <http://www.gnu.org/licenses/>. | 16 # along with Adblock Plus.  If not, see <http://www.gnu.org/licenses/>. | 
| 17 | 17 | 
|  | 18 import copy | 
| 18 import json | 19 import json | 
| 19 import random | 20 import random | 
| 20 import time | 21 import time | 
| 21 from urlparse import parse_qs | 22 import urlparse | 
| 22 | 23 | 
| 23 from sitescripts.notifications.parser import load_notifications | 24 from sitescripts.notifications.parser import load_notifications | 
| 24 from sitescripts.web import url_handler | 25 from sitescripts.web import url_handler | 
| 25 | 26 | 
|  | 27 def _determine_groups(version, notifications): | 
|  | 28   version_groups = dict(x.split("/") for x in version.split("-")[1:] | 
|  | 29                         if x.count("/") == 1) | 
|  | 30   groups = [] | 
|  | 31   for notification in notifications: | 
|  | 32     if "variants" not in notification: | 
|  | 33       continue | 
|  | 34     group_id = notification["id"] | 
|  | 35     if group_id in version_groups: | 
|  | 36       groups.append({"id": group_id, "variant": int(version_groups[group_id])}) | 
|  | 37   return groups | 
|  | 38 | 
| 26 def _assign_groups(notifications): | 39 def _assign_groups(notifications): | 
| 27   groups = [] | 40   groups = [] | 
| 28   selection = random.random() | 41   selection = random.random() | 
| 29   base = 0 | 42   start = 0 | 
| 30   for notification in filter(lambda notification: "variants" in notification, | 43   for notification in notifications: | 
| 31                              notifications): | 44     if "variants" not in notification: | 
|  | 45       continue | 
| 32     group = {"id": notification["id"], "variant": 0} | 46     group = {"id": notification["id"], "variant": 0} | 
| 33     groups.append(group) | 47     groups.append(group) | 
| 34     for i, variant in enumerate(notification["variants"]): | 48     for i, variant in enumerate(notification["variants"]): | 
| 35       sample_size = variant["sample"] | 49       sample_size = variant["sample"] | 
| 36       if (group["variant"] == 0 and sample_size > 0 and selection >= base and | 50       end = start + sample_size | 
| 37           selection <= sample_size + base): | 51       selected = sample_size > 0 and start <= selection <= end | 
|  | 52       start = end | 
|  | 53       if selected: | 
| 38         group["variant"] = i + 1 | 54         group["variant"] = i + 1 | 
| 39       base += sample_size | 55         break | 
| 40   return groups | 56   return groups | 
| 41 | 57 | 
| 42 def _create_response(notifications, groups): | 58 def _get_active_variant(notifications, groups): | 
| 43   response = { |  | 
| 44     "version": time.strftime("%Y%m%d%H%M", time.gmtime()) |  | 
| 45   } |  | 
| 46   for group in groups: | 59   for group in groups: | 
| 47     group_id = group["id"] | 60     group_id = group["id"] | 
| 48     variant = group["variant"] | 61     variant = group["variant"] | 
| 49     response["version"] += "-%s/%s" % (group_id, variant) | 62     if variant == 0: | 
| 50     if variant > 0: | 63       continue | 
| 51       notification = next(notification for notification in notifications | 64     notification = next(x for x in notifications if x["id"] == group_id) | 
| 52                           if notification["id"] == group_id) | 65     notification = copy.deepcopy(notification) | 
| 53       notification = notification.copy() | 66     notification.update(notification["variants"][variant - 1]) | 
| 54       notification.update(notification["variants"][variant - 1]) | 67     for key_to_remove in ("sample", "variants"): | 
| 55       for key_to_remove in ("sample", "variants"): | 68       notification.pop(key_to_remove, None) | 
| 56         notification.pop(key_to_remove, None) | 69     return notification | 
| 57       response["notifications"] = [notification] | 70 | 
| 58   if not "notifications" in response: | 71 def _generate_version(groups): | 
| 59     response["notifications"] = \ | 72   version = time.strftime("%Y%m%d%H%M", time.gmtime()) | 
| 60       filter(lambda notification: not "variants" in notification, notifications) | 73   for group in groups: | 
|  | 74     version += "-%s/%s" % (group["id"], group["variant"]) | 
|  | 75   return version | 
|  | 76 | 
|  | 77 def _create_response(notifications, groups): | 
|  | 78   active_variant = _get_active_variant(notifications, groups) | 
|  | 79   if active_variant: | 
|  | 80     notifications = [active_variant] | 
|  | 81   else: | 
|  | 82     notifications = [x for x in notifications if "variants" not in x] | 
|  | 83   response = { | 
|  | 84     "version": _generate_version(groups), | 
|  | 85     "notifications": notifications | 
|  | 86   } | 
| 61   return json.dumps(response, ensure_ascii=False, indent=2, | 87   return json.dumps(response, ensure_ascii=False, indent=2, | 
| 62                     separators=(",", ": "), sort_keys=True) | 88                     separators=(",", ": "), sort_keys=True) | 
| 63 | 89 | 
| 64 @url_handler("/notification.json") | 90 @url_handler("/notification.json") | 
| 65 def notification(environ, start_response): | 91 def notification(environ, start_response): | 
| 66   params = parse_qs(environ.get("QUERY_STRING", "")) | 92   params = urlparse.parse_qs(environ.get("QUERY_STRING", "")) | 
| 67   version = params.get("lastVersion", [""])[0] | 93   version = params.get("lastVersion", [""])[0] | 
| 68   current_groups = dict(component.split("/") |  | 
| 69                         for component in version.split("-")[1:] |  | 
| 70                         if "/" in component) |  | 
| 71   notifications = load_notifications() | 94   notifications = load_notifications() | 
| 72   groups = [] | 95   groups = _determine_groups(version, notifications) | 
| 73   for notification in notifications: |  | 
| 74     if "variants" in notification: |  | 
| 75       group_id = notification["id"] |  | 
| 76       if group_id in current_groups: |  | 
| 77         groups.append({"id": group_id, |  | 
| 78                        "variant": int(current_groups[group_id])}) |  | 
| 79   if not groups: | 96   if not groups: | 
| 80     groups = _assign_groups(notifications) | 97     groups = _assign_groups(notifications) | 
| 81   response = _create_response(notifications, groups) | 98   response = _create_response(notifications, groups) | 
| 82   start_response("200 OK", [("Content-Type", "application/json")]) | 99   start_response("200 OK", | 
|  | 100                  [("Content-Type", "application/json; charset=utf-8")]) | 
| 83   return response.encode("utf-8") | 101   return response.encode("utf-8") | 
| LEFT | RIGHT | 
|---|