Vue 2.0 起步(5) 订阅列表上传和下载 - 微信公众号RSS

上一篇:Vue 2.0 起步(4)第二版 轻量级后端Flask用户认证 - 微信公众号RSS

用户在本地搜索、订阅了公众号以后,可以方便地上传订阅列表到服务器上。也可以在另一地方下载订阅列表到本地

完成图:点击上传、下载图标就行

第5篇完成图

步骤:

  1. 后端Flask models更新
  2. api支持订阅列表上传、下载
  3. 前端vue.js ajax上传、下载订阅列表

1. 后端Flask models更新

Mp模型,添加to_json、from_json方法,分别对应ajax下载、上传来返回数据库查询。
注意from_json,用户上传是一个订阅列表,json里面包含多个mp,所以返回的是一个Mp模型的数组

/app/models.py

# 公众号
class Mp(db.Model):
    __tablename__ = 'mps'
    id = db.Column(db.Integer, primary_key=True)
    weixinhao = db.Column(db.Text)
    image = db.Column(db.Text)
    summary = db.Column(db.Text)
    sync_time = db.Column(db.DateTime, index=True, default=datetime.utcnow) # Search.vue: date
    mpName = db.Column(db.Text)
    encGzhUrl = db.Column(db.Text)  # 临时主页
    subscribeDate = db.Column(db.DateTime())
    # 如果加了dynamic, Flask-Admin里会显示raw SQL
    articles = db.relationship('Article', backref='mp', lazy='dynamic')
    subscribers = db.relationship('Subscription',
                               foreign_keys=[Subscription.mp_id],
                               backref=db.backref('mp', lazy='joined'),
                               lazy='dynamic',
                               cascade='all, delete-orphan')
    def to_json(self):
        json_mp = {
            'weixinhao': self.weixinhao,
            'mpName': self.mpName,
            'image': self.image,
            'summary': self.summary,
            'subscribeDate': self.subscribeDate,
            'articles_count': self.articles.count()
        }
        return json_mp

    @staticmethod
    def from_json(json_post):
        mps = json_post.get('mps')
        print mps
        if mps is None or mps == '':
            raise ('POST does not have mps')
        Mps = []
        for mp in mps:
#           print mp
            Mps.append( Mp(mpName=mp['mpName'], image=mp['image'], weixinhao=mp['weixinhao'] ) )
        return Mps

2. api支持订阅列表上传、下载

用户使用POST上传列表,带email、mps参数

  1. api取得email参数,来查询User已经订阅的Mps。
  2. Mp.from_json()把json.mps,转换成Mp模型列表
  3. 如果不存在这个订阅号,则添加到Mp,并订阅
  4. 如果用户没有订阅,则订阅

/app/api_1_0/mps.py

from flask_security import auth_token_required

@api.route('/mps', methods=['POST'])
@auth_token_required
def new_mps():
    email = request.get_json()['email']
    user = User.query.filter_by(email=email).first()
    Mps = Mp.from_json(request.json)
    subscribed_mps_weixinhao = [i.weixinhao for i in user.subscribed_mps]
    rsp = []
    for mp in Mps:
        mp_sql = Mp.query.filter_by(weixinhao=mp.weixinhao).first()
    # 如果不存在这个订阅号,则添加到Mp,并订阅
        if mp_sql is None:
                db.session.add(mp)
                user.subscribe(mp)
                rsp.append(mp.to_json())
                db.session.commit()
    # 如果用户没有订阅,则订阅
        elif not mp.weixinhao in subscribed_mps_weixinhao:
                user.subscribe(mp_sql)
                rsp.append(mp.to_json())
                db.session.commit()
            
    return jsonify(rsp), 201, \
        {'Location': url_for('api.get_mps', id=mp.id, _external=True)}

用户下载订阅列表,使用GET。服务器拿到email后,通过SQLAlchemy过滤器和联结查询,返回subscribed_mps

# 带 /mps/ 斜杠的,必须放在 /mps POST后面,不然默认会选择以下GET
@api.route('/mps/')
@auth_token_required
def get_mps():
    # request.args.items().__str__()
    email = request.args.get('email')
    print email
    mps = User.query.filter_by(email=email).first().subscribed_mps
    mps_list = [ mp.to_json() for mp in mps ]
    print mps_list
    return jsonify(mps_list)

3. 前端vue.js ajax上传、下载

/src/components/Siderbar.vue

上传函数uploadSubscription(),ajax POST请求头:注意Authentication-Token,Content-Type参数

POST header.png

json参数带上email, mps


POST body.png

服务器响应201,以及新添加的mps


new added Mp.png
            uploadSubscription() {
                if (this.subscribeList.length === 0) return false;
                this.$http.post('/api/v1.0/mps',
                    //body
                    {
                        email: this.username,
                        mps: this.subscribeList
                    },
                    //options
                    {
                        headers: {
                            'Content-Type': 'application/json; charset=UTF-8',
                            'Authentication-Token': this.token
                        }
                    }).then((response) => {
                    // 响应成功回调
                    var data = response.body,
                        mp;
                    alert('成功上传订阅号:\n' + JSON.stringify(data))

                }, (response) => {
                    // 响应错误回调
                    alert('同步出错了! ' + JSON.stringify(response))
                    if (response.status == 401) {
                        alert('登录超时,请重新登录');
                        this.is_login = false;
                        this.password = '';
                        window.localStorage.removeItem("user")
                    }
                });
            },

下载函数getSubscription(),GET url编码为:http://localhost:5000/api/v1.0/mps/?email=admin
对于服务器返回的json,是一个数组:

server response.png

Vue先要清空当前本地订阅列表,然后逐个订阅从服务器取回的Mp

            getSubscription() {
                this.$http.get('/api/v1.0/mps', {
                    params: {
                        email: this.username
                    },
                    headers: {
                        'Content-Type': 'application/json; charset=UTF-8',
                        'Authentication-Token': this.token
                    }
                }).then((response) => {
                    // 响应成功回调
                    var data = response.body,
                        mp, found_tag = false;
                    alert('订阅号 from server:\n' + JSON.stringify(data));
                    for (let item of this.subscribeList) {
                        this.$store.dispatch('unsubscribeMp', item.weixinhao);
                    }
                    this.$store.dispatch('clearSubscription', 'get sublist from Server');
                    for (let mp of data) {
                        mp['showRemoveBtn'] = false;
                        this.$store.dispatch('subscribeMp', mp);
                    }
                }, (response) => {
                    // 响应错误回调
                    alert('同步出错了! ' + JSON.stringify(response))
                    if (response.status == 401) {
                        alert('登录超时,请重新登录');
                        this.is_login = false;
                        this.password = '';
                        window.localStorage.removeItem("user")
                    }
                });
            }

OK,试试吧:

Demo:http://vue2.heroku.com
源码:https://github.com/kevinqqnj/vue-tutorial
请使用新的template: https://github.com/kevinqqnj/flask-template-advanced

下一篇:Vue 2.0 起步(6) 后台管理Flask-Admin - 微信公众号RSS

http://www.jianshu.com/p/ab778fde3b99

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351

推荐阅读更多精彩内容