| @@ -134,6 +134,7 @@ urlpatterns += [ | ||
| 134 | 134 | # 系统相关 | 
| 135 | 135 | urlpatterns += [ | 
| 136 | 136 | url(r'^op/upgrade$', op_views.upgrade_api, name='upgrade_api'), # APP 升级 | 
| 137 | + url(r'^op/patch$', op_views.patch_api, name='patch_api'), # APP 补丁 | |
| 137 | 138 | url(r'^op/online$', op_views.online_api, name='online_api'), # 是否上线 | 
| 138 | 139 | url(r'^op/splash$', op_views.splash_api, name='splash_api'), # 启动页面 | 
| 139 | 140 | url(r'^op/feedback$', op_views.feedback_api, name='feedback_api'), # 用户反馈 | 
| @@ -5,10 +5,12 @@ from django.contrib import admin | ||
| 5 | 5 | from django.template.loader import render_to_string | 
| 6 | 6 | from pysnippets.strsnippets import strip | 
| 7 | 7 |  | 
| 8 | -from operation.models import APPSettingsInfo, FeedbackInfo, GuestEntranceControlInfo, LatestAppInfo, SplashInfo | |
| 8 | +from operation.models import (APPSettingsInfo, FeedbackInfo, GuestEntranceControlInfo, LatestAppInfo, PatchInfo, | |
| 9 | + SplashInfo) | |
| 9 | 10 | from utils.disk_utils import write_to_disk | 
| 10 | 11 | from utils.redis.rapp import set_latest_app | 
| 11 | 12 | from utils.redis.rguest import delete_guest_entrance_control, set_guest_entrance_control | 
| 13 | +from utils.redis.rpatch import del_app_patch_info, set_app_patch_info | |
| 12 | 14 | from utils.redis.rsettings import del_app_settings_info, set_app_settings_info | 
| 13 | 15 |  | 
| 14 | 16 |  | 
| @@ -68,6 +70,24 @@ class LatestAppInfoAdmin(admin.ModelAdmin): | ||
| 68 | 70 | set_latest_app(obj.src) | 
| 69 | 71 |  | 
| 70 | 72 |  | 
| 73 | +class PatchInfoAdmin(admin.ModelAdmin): | |
| 74 | +    list_display = ('platform', 'version', 'patch', 'src', 'status', 'created_at', 'updated_at') | |
| 75 | +    list_filter = ('platform', 'src', 'status') | |
| 76 | + | |
| 77 | + def save_model(self, request, obj, form, change): | |
| 78 | + obj.version = strip(obj.version) | |
| 79 | + obj.save() | |
| 80 | + | |
| 81 | + # 设置 APP 补丁信息 | |
| 82 | + set_app_patch_info(obj) | |
| 83 | + | |
| 84 | + def delete_model(self, request, obj): | |
| 85 | + obj.delete() | |
| 86 | + | |
| 87 | + # 删除 APP 补丁信息 | |
| 88 | + del_app_patch_info(obj) | |
| 89 | + | |
| 90 | + | |
| 71 | 91 | class APPSettingsInfoAdmin(admin.ModelAdmin): | 
| 72 | 92 |      list_display = ('platform', 'channel', 'version', 'online', 'status', 'created_at', 'updated_at') | 
| 73 | 93 |      list_filter = ('platform', 'online', 'status') | 
| @@ -118,6 +138,7 @@ class GuestEntranceControlInfoAdmin(admin.ModelAdmin): | ||
| 118 | 138 |  | 
| 119 | 139 |  | 
| 120 | 140 | admin.site.register(LatestAppInfo, LatestAppInfoAdmin) | 
| 141 | +admin.site.register(PatchInfo, PatchInfoAdmin) | |
| 121 | 142 | admin.site.register(APPSettingsInfo, APPSettingsInfoAdmin) | 
| 122 | 143 | admin.site.register(SplashInfo, SplashInfoAdmin) | 
| 123 | 144 | admin.site.register(FeedbackInfo, FeedbackInfoAdmin) | 
| @@ -0,0 +1,37 @@ | ||
| 1 | +# -*- coding: utf-8 -*- | |
| 2 | +from __future__ import unicode_literals | |
| 3 | + | |
| 4 | +from django.db import models, migrations | |
| 5 | +import operation.models | |
| 6 | + | |
| 7 | + | |
| 8 | +class Migration(migrations.Migration): | |
| 9 | + | |
| 10 | + dependencies = [ | |
| 11 | +        ('operation', '0009_auto_20161220_1354'), | |
| 12 | + ] | |
| 13 | + | |
| 14 | + operations = [ | |
| 15 | + migrations.CreateModel( | |
| 16 | + name='PatchInfo', | |
| 17 | + fields=[ | |
| 18 | +                ('id', models.AutoField(verbose_name='ID', serialize=False, auto_created=True, primary_key=True)), | |
| 19 | +                ('status', models.BooleanField(default=True, help_text='\u72b6\u6001', db_index=True, verbose_name='status')), | |
| 20 | +                ('created_at', models.DateTimeField(help_text='\u521b\u5efa\u65f6\u95f4', verbose_name='created_at', auto_now_add=True)), | |
| 21 | +                ('updated_at', models.DateTimeField(help_text='\u66f4\u65b0\u65f6\u95f4', verbose_name='updated_at', auto_now=True)), | |
| 22 | +                ('platform', models.IntegerField(default=0, help_text='\u652f\u6301\u5e73\u53f0', db_index=True, verbose_name='plat', choices=[(0, '\u5168\u5e73\u53f0'), (1, 'Android'), (2, 'iOS')])), | |
| 23 | +                ('version', models.CharField(help_text='\u7248\u672c\uff081.0.0\uff09', max_length=255, null=True, verbose_name='version', blank=True)), | |
| 24 | +                ('patch', models.FileField(help_text='\u8865\u4e01', upload_to=operation.models.upload_path, null=True, verbose_name='patch', blank=True)), | |
| 25 | +                ('src', models.IntegerField(default=0, help_text='\u6700\u65b0\u7248\u6765\u6e90', db_index=True, verbose_name='src', choices=[(0, '\u62cd\u7231\u7528\u6237\u7aef'), (1, '\u62cd\u7231\u6444\u5f71\u5e08\u7aef'), (2, '\u62cd\u7231\u5bfc\u6e38\u7aef')])), | |
| 26 | + ], | |
| 27 | +            options={ | |
| 28 | + 'verbose_name': 'patchinfo', | |
| 29 | + 'verbose_name_plural': 'patchinfo', | |
| 30 | + }, | |
| 31 | + ), | |
| 32 | + migrations.AlterField( | |
| 33 | + model_name='appsettingsinfo', | |
| 34 | + name='version', | |
| 35 | + field=models.CharField(help_text='\u7248\u672c\uff081.0.0\uff09', max_length=255, null=True, verbose_name='version', blank=True), | |
| 36 | + ), | |
| 37 | + ] | 
| @@ -63,9 +63,43 @@ class LatestAppInfo(CreateUpdateMixin): | ||
| 63 | 63 | } | 
| 64 | 64 |  | 
| 65 | 65 |  | 
| 66 | +class PatchInfo(CreateUpdateMixin, PlatformMixin): | |
| 67 | + PAIAI_USER = 0 | |
| 68 | + PAIAI_LENSMAN = 1 | |
| 69 | + PAIAI_TOURGUIDE = 2 | |
| 70 | + | |
| 71 | + SRC = ( | |
| 72 | + (PAIAI_USER, u'拍爱用户端'), | |
| 73 | + (PAIAI_LENSMAN, u'拍爱摄影师端'), | |
| 74 | + (PAIAI_TOURGUIDE, u'拍爱导游端'), | |
| 75 | + ) | |
| 76 | + | |
| 77 | + version = models.CharField(_(u'version'), max_length=255, blank=True, null=True, help_text=u'版本(1.0.0)') | |
| 78 | + patch = models.FileField(_(u'patch'), upload_to=upload_path, blank=True, null=True, help_text=u'补丁') | |
| 79 | + | |
| 80 | + src = models.IntegerField(_(u'src'), choices=SRC, default=PAIAI_USER, help_text=u'最新版来源', db_index=True) | |
| 81 | + | |
| 82 | + class Meta: | |
| 83 | +        verbose_name = _('patchinfo') | |
| 84 | +        verbose_name_plural = _('patchinfo') | |
| 85 | + | |
| 86 | + def __unicode__(self): | |
| 87 | +        return u'{0.pk}'.format(self) | |
| 88 | + | |
| 89 | + @property | |
| 90 | + def patch_url(self): | |
| 91 | +        return u'{}{}'.format(settings.DOMAIN, self.patch and self.patch.url) | |
| 92 | + | |
| 93 | + @property | |
| 94 | + def data(self): | |
| 95 | +        return { | |
| 96 | + 'patch_url': self.patch_url, | |
| 97 | + } | |
| 98 | + | |
| 99 | + | |
| 66 | 100 | class APPSettingsInfo(CreateUpdateMixin, PlatformMixin): | 
| 67 | 101 | channel = models.CharField(_(u'channel'), max_length=255, blank=True, null=True, help_text=u'渠道') | 
| 68 | - version = models.CharField(_(u'version'), max_length=255, blank=True, null=True, help_text=u'版本') | |
| 102 | + version = models.CharField(_(u'version'), max_length=255, blank=True, null=True, help_text=u'版本(1.0.0)') | |
| 69 | 103 |  | 
| 70 | 104 | online = models.BooleanField(_(u'online'), default=True, help_text=u'是否上线') | 
| 71 | 105 |  | 
| @@ -9,6 +9,7 @@ from operation.models import FeedbackInfo, LatestAppInfo, SplashInfo | ||
| 9 | 9 | from utils.error.errno_utils import UserStatusCode | 
| 10 | 10 | from utils.error.response_utils import response | 
| 11 | 11 | from utils.redis.rapp import get_latest_app | 
| 12 | +from utils.redis.rpatch import get_app_patch_info | |
| 12 | 13 | from utils.redis.rsettings import get_app_settings_info | 
| 13 | 14 |  | 
| 14 | 15 |  | 
| @@ -37,6 +38,21 @@ def upgrade_api(request): | ||
| 37 | 38 | }) | 
| 38 | 39 |  | 
| 39 | 40 |  | 
| 41 | +@logit | |
| 42 | +def patch_api(request): | |
| 43 | + """ APP 补丁 """ | |
| 44 | +    platform = request.REQUEST.get('platform', '') | |
| 45 | +    version = request.REQUEST.get('version', '') | |
| 46 | +    src = int(request.POST.get('src', 0)) | |
| 47 | + | |
| 48 | + patch_info = get_app_patch_info(platform, version, src) | |
| 49 | + | |
| 50 | +    return response(200, 'Get Patch Info Success', u'获取补丁信息成功', { | |
| 51 | +        'patch_url': patch_info.get('patch_url', ''), | |
| 52 | + }) | |
| 53 | + | |
| 54 | + | |
| 55 | +@logit | |
| 40 | 56 | def online_api(request): | 
| 41 | 57 | """ 是否上线 """ | 
| 42 | 58 |      platform = request.REQUEST.get('platform', '') | 
| @@ -32,7 +32,9 @@ class PlatformMixin(models.Model): | ||
| 32 | 32 | abstract = True | 
| 33 | 33 |  | 
| 34 | 34 |      Platforms = { | 
| 35 | + 'ios': IOS, | |
| 35 | 36 | 'iOS': IOS, | 
| 37 | + 'adr': ADR, | |
| 36 | 38 | 'android': ADR, | 
| 37 | 39 | 'Android': ADR, | 
| 38 | 40 | } | 
| @@ -64,3 +64,4 @@ GUEST_ENTRANCE_CONTROL_INFO = 'guest:entrance:control:info' # STRING,游客 | ||
| 64 | 64 | # APP 相关 | 
| 65 | 65 | LATEST_APP_INFO = 'latest:app:info:%s' # STRING,最新 APP 信息,src | 
| 66 | 66 | APP_SETTINGS_INFO = 'app:settings:info:%s:%s:%s' # STRING,APP 设置信息,platform、channel、version | 
| 67 | +APP_PATCH_INFO = 'app:patch:info:%s:%s:%s' # STRING,APP 补丁信息,platform、version、src | 
| @@ -0,0 +1,23 @@ | ||
| 1 | +# -*- coding: utf-8 -*- | |
| 2 | + | |
| 3 | +import json | |
| 4 | + | |
| 5 | +from pai2.basemodels import PlatformMixin | |
| 6 | +from utils.redis.connect import r | |
| 7 | +from utils.redis.rkeys import APP_PATCH_INFO | |
| 8 | + | |
| 9 | + | |
| 10 | +def set_app_patch_info(apppatch): | |
| 11 | + """ 设置 APP 补丁信息 """ | |
| 12 | + r.set(APP_PATCH_INFO % (apppatch.platform, apppatch.version, apppatch.src), json.dumps(apppatch.data)) | |
| 13 | + | |
| 14 | + | |
| 15 | +def del_app_patch_info(apppatch): | |
| 16 | + """ 删除 APP 补丁信息 """ | |
| 17 | + r.delete(APP_PATCH_INFO % (apppatch.platform, apppatch.version, apppatch.src)) | |
| 18 | + | |
| 19 | + | |
| 20 | +def get_app_patch_info(platform, version, src): | |
| 21 | + """ 获取 APP 补丁信息 """ | |
| 22 | + platform = platform if isinstance(platform, int) else PlatformMixin.Platforms.get(platform) | |
| 23 | +    return json.loads(r.get(APP_PATCH_INFO % (platform, version, src)) or '{}') |