Ionic4 JWT鉴权教程

转载:Ionic 4 JWT Authentication Tutorial: Using Angular HttpClient with Node & Express.js Server

本教程演示如何使用ionic4和angular7创建登陆注册module。

将学习如何使用HttpClient发生Post请求到使用Node+Express.js创建的鉴权后台,使用RxJS Observable跟踪授权状态;

如何使用Ionic Storage module保存Express.js 返回的JWT信息,如token、过期时间等;

开始之前,需要确定Node.js和NPM安装到开发电脑上;

设置安装 Ionic CLI 4

npm install -g ionic

全局安装ionic一般需要管理员权限,win使用管理员权限的命令行,类linux使用sudo

创建Ionic 4 项目

命令行运行下面命令

ionic start ionic-auth-demo blank --type=angular

ionic4使用type指定前端框架为angular,后面可能会有vue,react,现在反正没有。ionic4的目标是不限制框架,甚至是原生js,jquery都可以用ionic创建混合移动app或者pwa

blank指定模板类型,这个模板只带一个叫home的页面,其他还有tabs。。。

如果你想指定其他cli还能干的事情,如安装cordova,安装免费Ionic Appflow SDK这些后面教程再讲。

等cli安装完依赖后,你就可以进入创建的目录,使用ionic serve运行体验app了,ionic serve会自动打开浏览器访问localhost:8100,如果愿意,加个-l,运行起来更酷哦

创建 Angular Module

modules是用来组织你的程序代码的。创建一个module封装鉴权功能的service和pages。

ionic generate module auth

运行上面命令将自动生成src/app/auth/auth.module.ts,修改代码如下:

import { NgModule } from '@angular/core';

import { CommonModule } from '@angular/common';

@NgModule({

  declarations: [],

  imports: [

    CommonModule

  ]

})

export class AuthModule { }

auth Module只需要引入CommonModule,这个angular内建模块包含angular基本的一些 directives、pipes如:ngif,decimalPipe等

还需要再我们的root application模块(src/app/app.module.ts)引入auth模块。打开src/app/app.module.ts,再导入数组中添加AuthModule:

import { AuthModule } from './auth/auth.module';

@NgModule({

  declarations: [AppComponent],

  entryComponents: [],

  imports: [BrowserModule, IonicModule.forRoot(), AppRoutingModule,

    AuthModule

  ],

  providers: [

    StatusBar,

    SplashScreen,

    { provide: RouteReuseStrategy, useClass: IonicRouteStrategy }

  ],

  bootstrap: [AppComponent]

})

export class AppModule {}

再导入HttpClient,发送http请求,FormsModule表单,Ionic Storage module本地存储

导入HttpClient

HttpClient是angular官方http客户端,所有我们需要在基于angular的ionic里导入。打开src/app/auth/auth.module.ts导入如下:

import { NgModule } from '@angular/core';

import { CommonModule } from '@angular/common';

import { HttpClientModule } from '@angular/common/http';

@NgModule({

  declarations: [],

  imports: [

    CommonModule,

    HttpClientModule

  ]

})

export class AuthModule { }

设置表单模块

angular为表单提供了强大的api,如基于模板的表单和reactive表单,这里我们用基于模板的表单,添加src/app/auth/auth.module.ts如下:

import { NgModule } from '@angular/core';

import { CommonModule } from '@angular/common';

import { HttpClientModule } from '@angular/common/http';

import { FormsModule } from '@angular/forms';

@NgModule({

  declarations: [],

  imports: [

    CommonModule,

    HttpClientModule,

    FormsModule

  ]

})

export class AuthModule { }

设置存储模块

ionic team提供的存储模块可以在移动设备的浏览器本地存储,但是使用前必须先安装并引入

安装使用命令:

npm install --save @ionic/storage

写这教程的时候,版本是2.2.0.

import { NgModule } from '@angular/core';

import { CommonModule } from '@angular/common';

import { HttpClientModule } from '@angular/common/http';

import { FormsModule } from '@angular/forms';

import { IonicStorageModule } from '@ionic/storage';

@NgModule({

  declarations: [],

  imports: [

    CommonModule,

    HttpClientModule,

    FormsModule,

    IonicStorageModule.forRoot()

  ]

})

export class AuthModule { }

创建Angular鉴权服务

设置了所有需要的设置后,我们可以创建鉴权服务了,用来封装HttpClient与Express server的通讯,如下命令创建:

ionic generate interface auth/user

生成src/app/auth/user.ts,编辑如下:

export interface User {

    id: number;

    name: string;

    email: string;

    password: string;

}

注意:在命令行里的auth/前缀告诉CLI在auth module里生成。

还需要生成服务器响应的接口,

ionic generate interface auth/auth-response

export interface AuthResponse {

    user: {

        id: number,

        name: string,

        email: string

    } ,       

access_token: string,       

expires_in: number

}

具体怎么响应,在下一个部分,用Node+Express.js构建后台的时候再说。

ionic generate service auth/auth

生成auth服务,会产生两个文件:src/app/auth/auth.service.ts,src/app/auth/auth.service.spec.ts,多个spec的是用来自动化测试的,可以删除

import { Injectable } from '@angular/core';

import { HttpClient } from  '@angular/common/http';

import { tap } from  'rxjs/operators';

import { Observable, BehaviorSubject } from  'rxjs';

import { Storage } from  '@ionic/storage';

import { User } from  './user';

import { AuthResponse } from  './auth-response';

定义变量

AUTH_SERVER_ADDRESS: string = 'http://localhost:3000';authSubject = new BehaviorSubject(false);

AUTH_SERVER_ADDRESS是后台地址,authSubject是用来订阅鉴权状态的Observable类型变量。

下一步构造函数中注入HttpClient和Storage

constructor(private httpClient: HttpClient, private storage: Storage) { }

使用HttpClient发送Post请求

register(user: User): Observable<AuthResponse> {

    return this.httpClient.post<AuthResponse>(`${this.AUTH_SERVER_ADDRESS}/register`, user).pipe(

      tap(async (res:  AuthResponse ) => {

        if (res.user) {

          await this.storage.set("ACCESS_TOKEN", res.access_token);

          await this.storage.set("EXPIRES_IN", res.expires_in);

          this.authSubject.next(true);

        }

      })

    );

  }

login(user: User): Observable<AuthResponse> {

    return this.httpClient.post(`${this.AUTH_SERVER_ADDRESS}/login`, user).pipe(

      tap(async (res: AuthResponse) => {

        if (res.user) {

          await this.storage.set("ACCESS_TOKEN", res.access_token);

          await this.storage.set("EXPIRES_IN", res.expires_in);

          this.authSubject.next(true);

        }

      })

    );

  }

async logout() {

    await this.storage.remove("ACCESS_TOKEN");

    await this.storage.remove("EXPIRES_IN");

    this.authSubject.next(false);

  }

获取鉴权状态

最后添加isLoggedIn()方法用来检查用户登陆与否

isLoggedIn() {

    return this.authSubject.asObservable();

  }

创建Ionic Pages

创建页面使用如下命令:

ionic generate page auth/register

命令会自动更新src/app/app-routing.module.ts,添加如下路由:

{ path: 'register', loadChildren: './auth/register/register.module#RegisterPageModule' },

意思是可以直接通过http://localhost:8100/register访问注册页面

打开src/app/auth/register/register.page.ts文件,注入Authservice和Router

import { Component, OnInit } from '@angular/core';

import { Router } from  "@angular/router";

import { AuthService } from '../auth.service';

@Component({

  selector: 'app-register',

  templateUrl: './register.page.html',

  styleUrls: ['./register.page.scss'],

})

export class RegisterPage implements OnInit {

  constructor(private  authService:  AuthService, private  router:  Router) { }

  ngOnInit() {

  }

}

添加register()方法

register(form) {

    this.authService.register(form.value).subscribe((res) => {

      this.router.navigateByUrl('home');

    });

  }

修改src/auth/register/register.page.html页面:

<ion-contentcolor="primary"><form#form="ngForm"(ngSubmit)="register(form)"><ion-grid><ion-rowcolor="primary"justify-content-center><ion-colalign-self-centersize-md="6"size-lg="5"size-xs="12"><divtext-center><h3>Create your account!</h3></div><divpadding><ion-item><ion-inputname="name"type="text"placeholder="Name"ngModelrequired></ion-input></ion-item><ion-item><ion-inputname="email"type="email"placeholder="your@email.com"ngModelrequired></ion-input></ion-item><ion-item><ion-inputname="password"type="password"placeholder="Password"ngModelrequired></ion-input></ion-item><ion-item><ion-inputname="confirm"type="password"placeholder="Password again"ngModelrequired></ion-input></ion-item></div><divpadding><ion-buttonsize="large"type="submit"[disabled]="form.invalid"expand="block">Register</ion-button></div></ion-col></ion-row></ion-grid></form></ion-content>

编辑src/auth/register/register.page.scss,自定义UI:

ion-item{--background:#3880ff;--color:#fff;}ion-button{--background:#062f77;}

登陆页面略,详见:Ionic 4 Forms Tutorial: Login & Register UI Example with Theming

创建并运行后台服务

创建express-auth-demo文件夹,命令行进入当前文件夹后,运行

npm init -y

初始化package.json文件,然后运行下面命令添加依赖:

npm install --save express body-parser sqlite3 bcryptjs jsonwebtoken cors

然后创建index.js文件,添加如下代码:

"use strict";

const express=require('express');

const bodyParser=require('body-parser');

const cors=require('cors');

const sqlite3=require('sqlite3').verbose();

const jwt=require('jsonwebtoken');

const bcrypt=require('bcryptjs');

const SECRET_KEY="secretkey23456";

const app=express();

constrouter=express.Router();app.use(cors())router.use(bodyParser.urlencoded({extended:false}));router.use(bodyParser.json());constdatabase=newsqlite3.Database("./my.db");constcreateUsersTable=()=>{constsqlQuery=`

        CREATE TABLE IF NOT EXISTS users (

        id integer PRIMARY KEY,

        name text,

        email text UNIQUE,

        password text)`;returndatabase.run(sqlQuery);}constfindUserByEmail=(email,cb)=>{returndatabase.get(`SELECT * FROM users WHERE email = ?`,[email],(err,row)=>{cb(err,row)});}constcreateUser=(user,cb)=>{returndatabase.run('INSERT INTO users (name, email, password) VALUES (?,?,?)',user,(err)=>{cb(err)});}createUsersTable();router.get('/',(req,res)=>{res.status(200).send('This is an authentication server');});router.post('/register',(req,res)=>{constname=req.body.name;constemail=req.body.email;console.log(req.body);constpassword=bcrypt.hashSync(req.body.password);createUser([name,email,password],(err)=>{if(err)returnres.status(500).send("Server error!");findUserByEmail(email,(err,user)=>{if(err)returnres.status(500).send('Server error!');constexpiresIn=24*60*60;constaccessToken=jwt.sign({id:user.id},SECRET_KEY,{expiresIn:expiresIn});res.status(200).send({"user":user,"access_token":accessToken,"expires_in":expiresIn});});});});router.post('/login',(req,res)=>{constemail=req.body.email;constpassword=req.body.password;findUserByEmail(email,(err,user)=>{if(err)returnres.status(500).send('Server error!');if(!user)returnres.status(404).send('User not found!');constresult=bcrypt.compareSync(password,user.password);if(!result)returnres.status(401).send('Password not valid!');constexpiresIn=24*60*60;constaccessToken=jwt.sign({id:user.id},SECRET_KEY,{expiresIn:expiresIn});res.status(200).send({"user":user,"access_token":accessToken,"expires_in":expiresIn});});});app.use(router);constport=process.env.PORT||3000;constserver=app.listen(port,()=>{console.log('Server listening at http://localhost:'+port);});


具体代码意思,查看Node Express JWT Authentication — jsonwebtoken and bcryptjs

编辑package.json,添加如下:

"scripts": {

    "start": "node index.js"

},

运行 npm start启动后台服务

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

推荐阅读更多精彩内容