草庐IT

html - 通过 session 变量在 Golang layout.tpl 中有条件地呈现 HTML

coder 2024-07-11 原文

我使用 Gorilla session (通过 negroni-sessions)将我的用户 session 存储在 cookie 中。我还使用 github.com/unrolled/render 进行 HTML 模板渲染:

ma​​in.go:

    package main

    import (
        ...
        "github.com/codegangsta/negroni"
        "github.com/goincremental/negroni-sessions"
        "github.com/goincremental/negroni-sessions/cookiestore"
        "github.com/julienschmidt/httprouter"
        "github.com/unrolled/render"
        ...
    )

    func init() {
        ...
        ren = render.New(render.Options{
            Directory:     "templates",
            Layout:        "layout",
            Extensions:    []string{".html"},
            Funcs:         []template.FuncMap{TemplateHelpers}, 
            IsDevelopment: false,
        })
        ...
    }

    func main() {
        ...
        router := httprouter.New()
        router.GET("/", HomeHandler)

        // Add session store
        store := cookiestore.New([]byte("my password"))
        store.Options(sessions.Options{
            //MaxAge: 1200,
            Domain: "",
            Path:   "/",
        })

        n := negroni.New(
            negroni.NewRecovery(),
            sessions.Sessions("cssession", store),
            negroni.NewStatic(http.Dir("../static")), 
        )

        n.UseHandler(router)
        n.Run(":9000")

    }

正如您在上面看到的,我使用了一个 layout.html 主 HTML 模板,它在任何页面呈现时都包含在内,比如我的主页:

    package main

    import (
        "html/template"
        "github.com/julienschmidt/httprouter"
    )

    func HomeHandler(w http.ResponseWriter, r *http.Request, p httprouter.Params) {

        var model = struct {
            CatalogPicks   []PromotionalModelList
            ClearanceItems []Model
        }{
            CatalogPicks:   GetCatalogPicks(),
            ClearanceItems: GetClearanceItems(),
        }

        ren.HTML(w, http.StatusOK, "home", model)
    }

在我的 layout.html 主 HTML 模板中,我想呈现一个管理菜单,但前提是当前用户是管理员:

layout.html:

<!doctype html>
<html>
  <head>
    <meta charset="utf-8">
      <title>{{ template "title" . }}</title>
        ...
  </head>

  <body>
    ...
    <!--Main Menu-->
      <nav class="menu">
         <ul class="catalog">
             <li class="has-submenu">
                {{ RenderMenuCategories }}
             </li>
             <li><a href="javascript:void(0)">Blog</a></li>
             <li><a href="javascript:void(0)">Company</a></li>

             {{ RenderAdminMenu }}

         </ul>
      </nav>

...

我的问题是上面的模板辅助函数 RenderAdminMenu() 无法访问 HTTP 请求对象,因此无法访问 User session 对象以确定是否用户是管理员。

我可以通过主页处理程序将用户对象传递到模板上下文中,并使用 if 语句 RenderAdminMenu() 函数,如下所示

{{ if .User.IsAdmin }}
   {{ RenderAdminMenu }}
{{ end }}

...但是因为我使用的是主模板,所以我必须从网站上的每个网页都这样做。有没有更有效的方法?

我在想也许有一种方法可以从 RenderAdminMenu()(或 layout.html)中访问某种全局 Context 对象,其中包含 请求详细信息(就像在 ASP.NET 中一样)

最佳答案

您需要做一些事情才能将其联系在一起。我不打算展示一个完整的示例,因为它会相当冗长并且可能与您的代码(您尚未发布)不匹配。不过,它将包含基本的构建 block :如果您遇到困难,请返回一个直接的问题和一个代码片段,您将获得更直接的答案:)

  1. 在 [登录] 处理程序中编写一些中间件或逻辑,用于在用户登录时将用户数据保存在 session 中。用户 ID、电子邮件和管理员 bool 值就足够了。例如

    // In your login handler, once you've retrieved the user &
    // matched their password hash (scrypt, of course!) from the DB.
    session.Values["user"] = &youruserobject
    err := session.Save(r, w)
    if err != nil {
        // Throw a HTTP 500
    }
    

注意:请记住,您需要根据 gorilla/sessions docs gob.Register(&youruserobject{})如果您想存储自己的类型。

  1. 编写一个帮助程序,在您将其从 session 中拉出时对您的类型进行类型断言,例如

    var ErrInvalidUser= errors.New("invalid user stored in session")
    
    func GetUser(session *sessions.Session) (*User, error) {
        // You can make the map key a constant to avoid typos/errors
        user, ok := session.Values["user"].(*User)
        if !ok || user == nil {
            return nil, ErrInvalidUser
        }
        return user, nil
    }
    
    
    // Use it like this in a handler that serves user content
    session, err := store.Get("yoursessionname", r)
    if err != nil {
        // Throw a HTTP 500
    }
    
    user, err := GetUser(session)
       if err != nil {
        // Re-direct back to the login page or
        // show a HTTP 403 Forbidden, etc.
    }
    
  2. 写一些东西来检查返回的用户是否是管理员:

    func IsAdmin(user *User) bool {
            if user.Admin == true && user.ID != "" && user.Email != "" {
                    return true
            }
    
            return false
    }
    
  3. 将其传递给模板:

     err := template.Execute(w, "sometemplate.html", map[string]interface{}{
            "admin": IsAdmin(user),
            "someotherdata": someStructWithData,
        }
    
     // In your template...
     {{ if .admin }}{{ template "admin_menu" }}{{ end }}
    

还要确保您正在为您的 session cookie 设置一个身份验证 key (阅读 gorilla 文档),最好是一个加密 key ,并且您正在通过 HTTPS 为您的站点提供服务,同时设置了 Secure: true 标志。

请记住,上述方法也得到了简化:如果您在数据库中将用户取消标记为管理员,只要他们的 session 持续,应用程序就会继续将他们检测为管理员。默认情况下,这可能是 7 天,因此如果您处于管理员流失是一个真正问题的危险环境中,则可能需要非常短的 session 点击 IsAdmin 中的数据库 功能只是为了安全。如果这是一个个人博客,而且只有您一个人,那就没那么多了。


添加:如果您想将用户对象直接传递给模板,您也可以这样做。请注意,与在模板逻辑中相比,在您的处理程序/中间件中执行此操作的性能更高。您还可以获得更多错误处理的灵 active ,以及​​更早“退出”的选项 - 即,如果 session 不包含任何内容,您可以引发 HTTP 500 错误,而不是呈现半个模板或必须在您的代码中放入大量逻辑处理 nil 数据的模板。

您仍然需要在 session 中存储您的 User 对象(或等效对象),并在将其传递给模板之前从 session.Values 中检索它。

 func GetUser(r *http.Request) *User {
        session, err := store.Get("yoursessionname", r)
        if err != nil {
            // Throw a HTTP 500
        }
        if user, ok := session.Values["user"].(*User); ok {
            return user
        }

        return nil
 }

 // In the handler itself
 err := template.Execute(w, "sometemplate.html", map[string]interface{}{
        "user": GetUser(r),
        "someotherdata": someStructWithData,
        }


 // In your template...
 {{ if .User.admin }}{{ template "admin_menu" }}{{ end }}

关于html - 通过 session 变量在 Golang layout.tpl 中有条件地呈现 HTML,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29689426/

有关html - 通过 session 变量在 Golang layout.tpl 中有条件地呈现 HTML的更多相关文章

  1. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

    我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

  2. ruby - 通过 rvm 升级 ruby​​gems 的问题 - 2

    尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub

  3. ruby - 通过 erb 模板输出 ruby​​ 数组 - 2

    我正在使用puppet为ruby​​程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby​​不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这

  4. ruby-on-rails - 如何使用 instance_variable_set 正确设置实例变量? - 2

    我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击

  5. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

  6. ruby-on-rails - Rails HTML 请求渲染 JSON - 2

    在我的Controller中,我通过以下方式在我的index方法中支持HTML和JSON:respond_todo|format|format.htmlformat.json{renderjson:@user}end在浏览器中拉起它时,它会自然地以HTML呈现。但是,当我对/user资源进行内容类型为application/json的curl调用时(因为它是索引方法),我仍然将HTML作为响应。如何获取JSON作为响应?我还需要说明什么? 最佳答案 您应该将.json附加到请求的url,提供的格式在routes.rb的路径中定义。这

  7. ruby - 通过 RVM (OSX Mountain Lion) 安装 Ruby 2.0.0-p247 时遇到问题 - 2

    我的最终目标是安装当前版本的RubyonRails。我在OSXMountainLion上运行。到目前为止,这是我的过程:已安装的RVM$\curl-Lhttps://get.rvm.io|bash-sstable检查已知(我假设已批准)安装$rvmlistknown我看到当前的稳定版本可用[ruby-]2.0.0[-p247]输入命令安装$rvminstall2.0.0-p247注意:我也试过这些安装命令$rvminstallruby-2.0.0-p247$rvminstallruby=2.0.0-p247我很快就无处可去了。结果:$rvminstall2.0.0-p247Search

  8. ruby-on-rails - 使用 Sublime Text 3 突出显示 HTML 背景语法中的 ERB? - 2

    所以我在关注Railscast,我注意到在html.erb文件中,ruby代码有一个微弱的背景高亮效果,以区别于其他代码HTML文档。我知道Ryan使用TextMate。我正在使用SublimeText3。我怎样才能达到同样的效果?谢谢! 最佳答案 为SublimeText安装ERB包。假设您安装了SublimeText包管理器*,只需点击cmd+shift+P即可获得命令菜单,然后键入installpackage并选择PackageControl:InstallPackage获取包管理器菜单。在该菜单中,键入ERB并在看到包时选择

  9. ruby-on-rails - Enumerator.new 如何处理已通过的 block ? - 2

    我在理解Enumerator.new方法的工作原理时遇到了一些困难。假设文档中的示例:fib=Enumerator.newdo|y|a=b=1loopdoy[1,1,2,3,5,8,13,21,34,55]循环中断条件在哪里,它如何知道循环应该迭代多少次(因为它没有任何明确的中断条件并且看起来像无限循环)? 最佳答案 Enumerator使用Fibers在内部。您的示例等效于:require'fiber'fiber=Fiber.newdoa=b=1loopdoFiber.yieldaa,b=b,a+bendend10.times.m

  10. ruby-on-rails - 如何在 ruby​​ 交互式 shell 中有多行? - 2

    这可能是个愚蠢的问题。但是,我是一个新手......你怎么能在交互式ruby​​shell中有多行代码?好像你只能有一条长线。按回车键运行代码。无论如何我可以在不运行代码的情况下跳到下一行吗?再次抱歉,如果这是一个愚蠢的问题。谢谢。 最佳答案 这是一个例子:2.1.2:053>a=1=>12.1.2:054>b=2=>22.1.2:055>a+b=>32.1.2:056>ifa>b#Thecode‘if..."startsthedefinitionoftheconditionalstatement.2.1.2:057?>puts"f

随机推荐