Cookie与登录注册

预备知识


请预习 Cookie、Session、Cache-Control 等 HTTP 知识

要点:

Cookie 的特点

  1. 服务器通过 Set-Cookie 响应头设置 Cookie
  2. 浏览器得到 Cookie 之后,每次请求都要带上 Cookie
  3. 服务器读取 Cookie 就知道登录用户的信息(email)

问题

  1. 我在 Chrome 登录了得到 Cookie,用 Safari 访问,Safari 会带上 Cookie 吗
    no
  2. Cookie 存在哪
    Windows 存在 C 盘的一个文件里
  3. Cookie会被用户篡改吗?
    可以,下节课会讲 Session 来解决这个问题,防止用户篡改
  4. Cookie 有效期吗?
    默认有效期20分钟左右,不同浏览器策略不同
    后端可以强制设置有效期,具体语法看 MDN
  5. Cookie 遵守同源策略吗?
    也有,不过跟 AJAX 的同源策略稍微有些不同。
    当请求 qq.com 下的资源时,浏览器会默认带上 qq.com 对应的 Cookie,不会带上 baidu.com 对应的 Cookie
    当请求 v.qq.com 下的资源时,浏览器不仅会带上 v.qq.com 的Cookie,还会带上 qq.com 的 Cookie
    另外 Cookie 还可以根据路径做限制,请自行了解,这个功能用得比较少。
  6. 今天讲的Cookie、Session、Cache-Control都是响应头里面的内容,请求头没有相关内容
  7. Cache-Control用于控制缓存的,Cookie的意思是曲奇饼
  8. 请求和响应


    image.png
  9. JS发送请求


    image.png

注册


  1. 我们给原先创造的server添加一个路由,路径是sign_up,并创建相应的文件,以便访问


    image.png

    image.png
  2. 我们做一个登录界面


    image.png
  3. 我们希望当点击提交的时候,会连带这些输入信息提交到服务器,服务器看看这些是不是正确的。
  4. 我们打开这个https://www.bootcdn.cn来找jQuery的链接
  5. 收集用户输入的东西,并存入哈希中
    let hash = {}
    $('#signUpForm').on('submit', (e)=>{
        e.preventDefault()
        let need = ['email', 'password', 'password_confirmation']
        need.forEach((name)=>{
            let value = $('#signUpForm').find(`[name=${name}]`).val()
            hash[name] = value
        })
        console.log(hash)
    })
    
  6. 接下来,我们就可以使用jquery的ajax发送ajax,问,下面这种ajax返回的会是什么?是404,还是一串html?
    let hash = {}
    $('#signUpForm').on('submit', (e)=>{
        // 拿到用户提交的信息并传入hash中
        e.preventDefault()
        let need = ['email', 'password', 'password_confirmation']
        need.forEach((name)=>{
            let value = $('#signUpForm').find(`[name=${name}]`).val()
            hash[name] = value
        })
        // 发送ajax
        $.post('/sign_up', hash)
            .then(()=>{console.log('success')}, ()=>{console.log('error')})
    })
    
  7. 返回的是一串字符串,因为我们在路由那边写了,只要是这个路径就是200,其他什么都没管,没有限定请求方式等。


    image.png

    image.png
  8. 我们可以区分一下这个路径及方式


    image.png
  9. 我们点击提交的时候,还是200,返回的response是空的。接下来我们需要读取用户提交的数据


    image.png
  10. email=1&password=2&password_confirmation=3,这个形式的代码,node不好读取,因为上传是一段一段上传的,我们需要借助工具。这个跟TCP有关,第四部分不一块上传
  11. google搜索node http get post data,
    image.png
  12. 点击之后,后台会打印字符串,前端发送请求


    image.png

    image.png
  13. 解析这个拿到的字符串


    image.png

    image.png
  14. 我们对拿到的字符串解析后进行验证


    image.png

    image.png
  15. 当我们前后端分离后,我们需要定前后端协议,我们返回的信息可以使用JSON,JS就可以读取JSON语法数据


    image.png
  16. 这样打印出来的是字符串还是对象呢?


    image.png

    image.png
  17. 答案是符合JSON对象语法的字符串,我们传入JSON.parse中去,就是一个对象了


    image.png

    image.png
  18. 我们每次都需要进行JSON.parse一下,比较麻烦,其实如果在server那边声明是JSON,jQuery就会自动识别的;response.setHeader('Content-Type', 'application/json;charset=utf-8'),然后使用request.responseJSON就可以拿到的,这是基于协议的,因此可以自动识别
  19. 这样我们就可以针对错误,发起不同提示


    image.png

    image.png
  20. 前端验证,在POST之前展示错误并return


    image.png
  21. 后端验证是一定需要的,前端验证不一定要,因为有的直接使用curl发送请求了,并不通过JS

    [
    image.png

登录以及Cookie


  1. 如果服务器发现密码和邮箱等正确,一般会存下来,但是密码一般不存。

  2. 我们需要有文件存储数据,创建一个文件,专门用于存储数据

  3. 之前拿到的数据需要使用decodeURIComponent()进行转译


    image.png
  4. 我们可以通过后台将这个新的账户存入到数据文件中

  5. 数据存储必须使用字符串,不可以使用对象,所以需要使用JSON.stringify

  6. 我们存入的时候,需要遍历,看看有没有存进去,如果存进去,就不要再存了

  7. 理论上,我们不能存用户的密码,应该存加密后的东西,只要后面能对上就代表输对了

  8. 我们再添加一个sing_in路由

  9. 假设登陆成功,我们跳到首页。

  10. 问题1:登录成功与否都可以到首页,这样有问题

  11. 问题2:能不能登录成功有名字显示呢

  12. 这就是引入今天的主题,需要一个Cookie,即告诉服务器,或者浏览器,我是谁;在登录成功的一瞬间,我们需要Cookie

  13. 可以查询http set cookie

  14. 设置好之后,点击preserver log阻止页面刷新,可以看到response有特殊的响应头,下面所有的请求,只要是相同的源,都会携带这个Cookie


    image.png

    image.png
  15. 有了这个标识信息,网页就可以认得是谁访问了

  16. 总结:

    1. 服务器通过 Set-Cookie 响应头设置 Cookie
    2. 浏览器得到 Cookie 之后,每次请求都要带上 Cookie
    3. 服务器读取 Cookie 就知道登录用户的信息(email)
  17. 问题

    1. 我在 Chrome 登录了得到的 Cookie,用 Safari 访问,Safari会带上Cookie吗?
      答案: 不会
    2. Cookie存在哪?
      Window 存在 C 盘的一个文件里
    3. 票能作假吗?
      可以
    4. Cookie有效期吗?
      默认有效期20分钟左右
      后端可以强制设置有效期
      HttpOnly是可以防止用户使用JS修改的,但是还是可以使用手动


      image.png
  18. 拿到Cookie使用线程现成的框架就好

  19. 退出登录,点击X就好了


    image.png
  20. 总结:


    image.png
  21. server代码

    var http = require('http')
    var fs = require('fs')
    var url = require('url')
    var port = process.argv[2]
    if(!port){
        console.log('请指定端口奥好不好?\nnode server.js 8888 这样不会吗?')
        process.exit(1)
    }
    var server = http.createServer(function(request, response){
        var parsedUrl = url.parse(request.url, true)
        var pathWithQuery = request.url
        var queryString = ''
        if(pathWithQuery.indexOf('?') >= 0){queryString=pathWithQuery.substring(pathWithQuery.indexOf('?'))}
        var path = parsedUrl.pathname
        var query = parsedUrl.query
        var method = request.method
        /*********从这里开始看,上面不要看*********/
        console.log('方方说:含查询字符串的路径为\n' + pathWithQuery)
    
        if(path === '/'){
            let string = fs.readFileSync('./index.html', 'utf-8')  //同步读取网页文件
            let cookies = request.headers.cookie.split('; ')
            let hash = {}
            for(let i=0; i<cookies.length; i++){
                let parts = cookies[i].split('=')
                let key = parts[0]
                let value = parts[1]
                hash[key] = value
            }
            let email = hash.sign_in_email
            let users = fs.readFileSync('./db/users', 'utf8')
            users = JSON.parse(users)
            let foundUser
            for(let i=0; i< users.length; i++){
                if(users[i].email === email){
                    foundUser = users[i]
                    break
                }
            }
            if(foundUser){
                string = string.replace('__password__', foundUser.password)
            }else{
                string = string.replace('__password__', '不知道')
            }
            response.statusCode = 200
            response.setHeader('Content-Type', 'text/html;charset=utf-8')
            response.write(string)    // 将网页文件写入到response中返回
            response.end()
        }else if(path==='/sign_up' && method === 'GET'){
            let string = fs.readFileSync('./sign_up.html', 'utf-8')
            response.statusCode = 200
            response.setHeader('Content-Type', 'text/html;charset=utf-8')
            response.write(string)
            response.end()
        }else if(path==='/sign_up' && method === 'POST'){
            readBody(request).then((body)=>{
                let strings = body.split('&')    // ['email=1', 'password=2', 'password_confirmation=3']
                let hash = {}
                strings.forEach((string)=>{
                    let parts = string.split('=')    // ['email', '1']
                    let key = parts[0]
                    let value = parts[1]
                    // 进行转译,否则会拿不到@,使用%40代替
                    hash[key] = decodeURIComponent(value)    // hash['email'] = '1'
                })
                // let email = hash['email']
                // let password = hash['password']
                // let password_confirmation = hash['password_confirmation']
                let {email, password, password_confirmation} = hash
                if(email.indexOf('@') === -1){
                    response.statusCode = 400
                    response.setHeader('Content-Type', 'application/json;charset=utf-8')
                    // 我们返回JSON数据,这样就可以实现选择性展示出错误信息,最外端的''不属于JSON,只表示是一个字符串
                    // JS可以理解JSON语法
                    response.write(`{
                        "errors": {
                            "email": "invalid"
                        }
                    }`)
                }else if(password !== password_confirmation){
                    response.statusCode = 400
                    response.write('password not match')
                }else{
                    // 这个users拿到之后是一个字符串
                    var users = fs.readFileSync('./db/users', 'utf8')
                    try{
                        users = JSON.parse(users)    // []
                    }catch{
                        users = []
                    }
                    let inUse = false
                    for(let i=0; i<users.length; i++){
                        let user = users[i]
                        if(user.email === email){
                            inUse = true
                            break;
                        }
                    }
                    if(inUse){
                        response.statusCode = 400
                        response.write('email in use')
                    }else{
                        users.push({email: email, password: password})
                        // 对象不能直接存,我们需要使用字符串化
                        var usersString = JSON.stringify(users)
                        fs.writeFileSync('./db/users', usersString)
                        response.statusCode = 200
                    }
    
                }
                response.end()
            })
        }else if(path === '/sign_in' && method === 'GET'){
            let string = fs.readFileSync('./sign_in.html', 'utf-8')
            response.statusCode = 200
            response.setHeader('Content-Type', 'text/html;charset=utf-8')
            response.write(string)
            response.end()
        }else if(path === '/sign_in' && method === 'POST'){
            readBody(request).then((body)=>{
                let strings = body.split('&')    // ['email=1', 'password=2', 'password_confirmation=3']
                let hash = {}
                strings.forEach((string)=>{
                    let parts = string.split('=')    // ['email', '1']
                    let key = parts[0]
                    let value = parts[1]
                    // 进行转译,否则会拿不到@,使用%40代替
                    hash[key] = decodeURIComponent(value)    // hash['email'] = '1'
                })
                // let email = hash['email']
                // let password = hash['password']
                // let password_confirmation = hash['password_confirmation']
                let {email, password} = hash
                var users = fs.readFileSync('./db/users', 'utf8')
                try{
                    users = JSON.parse(users)    // []
                }catch{
                    users = []
                }
                let found
                for(let i=0;i<users.length;i++){
                    if(users[i].email === email && users[i].password === password){
                        found = true
                        break
                    }
                }
                console.log(found)
                if(found){
                    // 设置cookie
                    response.setHeader('Set-Cookie', `sign_in_email=${email}; HttpOnly`)
                    response.statusCode = 200
                }else{
                    response.statusCode = 401
                }
                response.end()
            })
        }else if(path==='/xxxx'){
            response.statusCode = 200
            response.setHeader('Content-Type', 'text/json;charset=utf-8')
            // 加一个后台的响应头,告诉浏览器,这个域名可以访问我的后台
            response.setHeader('Access-Control-Allow-Origin', 'http://frank.com:8001')
            // 这边其实返回的是一个字符串,这个字符串刚好符合JSON语法,此处绝对不是一个对象
            response.write(`
                {
                    "note": {
                        "to": "小谷",
                        "from": "芳芳",
                        "heading": "打招呼",
                        "content": "hi"
                    }
                }
            `)
            response.end()
        }else if(path='/main.js'){
            let string = fs.readFileSync('./main.js')  //同步读取网页文件
            response.statusCode = 200
            response.setHeader('Content-Type', 'text/javascript;charset=utf-8')
            response.write(string)    // 将网页文件写入到response中返回
            response.end()
        }else{
            response.statusCode = 404
            response.setHeader('Content-Type', 'text/html;charset=utf-8')
            response.write('呜呜呜')
            response.end()
        }
        /*********代码结束,下面不要看*********/
    })
    function readBody(request){
        return new Promise((resolve, reject)=>{
            let body = [];    // 请求体
            // request监听data事件,每次data事件会给一小块数据
            request.on('data', (chunk) => {
                // 将一小块数据放入body中
                body.push(chunk);
            }).on('end', () => {
                // end就是上传结束,将body里面的数据全部连成字符串
                body = Buffer.concat(body).toString();
                resolve(body)
            })
        })
    }
    server.listen(port)
    console.log('监听 ' + port + ' 成功\n请在空中转体720度然后用电饭煲打开 http://localhost:' + port)
    
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容