- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
如何在 Haskell 中定义一个简单的分层访问控制系统?
我的角色是 Public > Contributor > Owner
,这些角色在层次结构中。 Public
能做到的一切也可以通过 Contributor
完成和 Owner
;等等。
同样,操作也在层次结构中:None > View > Edit
.如果允许角色编辑,它也应该能够查看。
data Role = Public | Contributor | Owner
data Operation = None | View | Edit
newtype Policy = Policy (Role -> Operation)
publicEditable :: Policy
publicEditable = Policy $ const Edit
Public
到
Edit
但拒绝对
Owner
的任何访问):
stupidPolicy :: Policy
stupidPolicy = Policy check where
check Public = Edit
check Contributor = View
check Owner = None
最佳答案
任何有权访问 Policy
的人的构造函数可以采用 Policy
分开并重新组合在一起,可能是一种 absurd 的方式。不要暴露 Policy
此模块之外的构造函数。相反,请提供 smart constructor创建保证格式正确的策略并公开 Monoid
接口(interface)来组合它们而不破坏不变量。保留Policy
type abstract 确保所有可能导致无意义策略的代码都保存在此模块中。
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
module Policy (
Role(..),
Level(..),
Policy, -- keep Policy abstract by not exposing the constructor
can
) where
import Data.Semigroup (Semigroup, Max(..))
data Role = Public | Contributor | Owner
deriving (Eq, Ord, Bounded, Enum, Show, Read)
data Level = None | View | Edit
deriving (Eq, Ord, Bounded, Enum, Show, Read)
GeneralizedNewtypeDeriving
借一双
Monoid
来自
base
的实例:
the monoid for functions ,它通过函数箭头逐点提升另一个幺半群,和
the Max
newtype , 变成
Ord
实例为
Monoid
例如,始终选择
mappend
中的较大者的论据。
Policy
的
Monoid
实例将自动管理
Level
的排序制定策略时:在给定角色制定两个具有冲突级别的策略时,我们总是会选择更宽松的一个。这使得
<>
附加操作:通过向“默认”策略
mempty
添加权限来定义策略,这是不授予任何人权限的一种。
newtype Policy = Policy (Role -> Max Level) deriving (Semigroup, Monoid)
grant
是一个智能构造函数,它生成尊重
Role
的排序属性的策略和
Level
.请注意,我将角色与
>=
进行比较确保向角色授予权限也会将该权限授予更多特权角色。
grant :: Role -> Level -> Policy
grant r l = Policy (Max . pol)
where pol r'
| r' >= r = l
| otherwise = None
can
是一个观察结果,它告诉您策略是否将给定的访问级别授予给定的角色。我再次使用
>=
以确保更宽松的水平意味着更宽松的水平。
can :: Role -> Level -> Policy -> Bool
(r `can` l) (Policy f) = getMax (f r) >= l
deriving
机制,尤其是
GeneralizedNewtypeDeriving
, 是让类型负责“无聊”代码的一种非常好的方式,这样您就可以专注于重要的事情。
module Client where
import Data.Monoid ((<>))
import Policy
Monoid
类从简单的策略中构建复杂的策略。
ownerEdit, contributorView, myPolicy :: Policy
ownerEdit = grant Owner Edit
contributorView = grant Contributor View
myPolicy = ownerEdit <> contributorView
can
用于测试策略的功能。
canPublicView :: Policy -> Bool
canPublicView = Public `can` View
ghci> canPublicView myPolicy
False
关于haskell - 类型化分级访问控制系统,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41522159/
我在 vscode 中使用带有 TypeScript 的 Svelte,并且在 vscode 中安装了 Svelte 扩展。 在我的 App.svelte 中有 // a bunch of co
我想延长go-validator返回更好的类型: type Error map[string][]error // Will output the first error when stringifi
在 python 中,您可以定义具有自动值的类型化枚举: import enum from enum import auto class Ordinals(enum.IntEnum): FIRST
我有一个 custom set我想在打字 Racket 中使用它。一世 要求它使用 require/typed与 #:opaque custom-set?操作说明。它工作,除了代码在运行时失败,当我
下面2种设置HttpClient的场景有什么区别吗? 我应该更喜欢一个吗? 输入客户端: public class CatalogService { private readonly Http
我正在尝试创建一个 dbTyped 和大小的 SqlParameters 数组。这工作正常,但如果我需要另一列,则会导致更改两个地方的代码。 SqlParameter[] parameters = {
我有一个用例,其中复杂的 UI 层次结构需要在 iframe 中呈现,但处理它的逻辑(创建、输入文档、事件处理、退出文档)需要在主框架/应用程序中。 我在获取对 iframe 的 Document 实
我最近将 VS 2005 升级到了 2010 年,对 LinQ 还很陌生。也许有人可以把我放在正确的方式。 背景 : 我有一个类型化数据集,并且使用 Table AccessRule 扩展了标准 SQ
我问这个只是为了澄清我的想法是否正确。 静态/动态类型如果变量的类型在编译时已知,则语言是静态类型的。这实际上意味着您作为程序员必须指定每个变量的类型。示例:Java、C、C++。 如果在运行时解释变
当我使用 CultureInfo Typed DataMember 调用我的 WCF 服务的方法时,它抛出 CommunicationException。 我该如何解决这个问题? The InnerE
我想将项目转换为字符串数组或用于填充 ListBox.DataSource 的类型。该类型已覆盖 ToString(),但我似乎无法将其转换,甚至无法转换为 String[]。 String[] a
如何获取/打印(键入的)查询后面的 JPQL 查询字符串,即设置之后参数? (例如,用于调试目的) 一个简单的 toString() 似乎并不能解决问题... 谢谢 最佳答案 没有“最终被翻译成最终
这是 Scala 2.8.0 beta 对这个问题的跟进: What is a proper way to manage flexible, typed, immutable data structu
我是一名优秀的程序员,十分优秀!