casbin-test.php 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440
  1. <?php
  2. require_once '../../vendor/autoload.php';
  3. require_once '../config.php';
  4. use Casbin\Enforcer;
  5. use CasbinAdapter\Medoo\Adapter as DatabaseAdapter;
  6. define("IDPrefixTranslation" , "it_");
  7. define("IDPrefixChannel", "ic_");
  8. define("IDPrefixArticle", "ia_");
  9. define("IDPrefixUser" , "iu_");
  10. define("IDPrefixOrg" , "io_");
  11. define("IDPrefixOrgGroup", "iog_");
  12. // 资源分组:版本风格
  13. define("ResStudio" , "s_studio");
  14. // 资源分组:版本风格
  15. define("ResChannel" , "s_channel");
  16. // 资源分组:文章
  17. define("ResArticle" , "s_article");
  18. // 资源分组:文集
  19. define("ResArticle" , "s_collection");
  20. // 资源分组:译文(版本风格 + 文章)句子库
  21. define("ResTranslation" , "s_translation");
  22. // 资源分组:逐词译段落
  23. define("ResArticle" , "s_wbw");
  24. // 资源分组:术语
  25. define("ResArticle" , "s_term");
  26. // 资源分组:用户字典
  27. define("ResArticle" , "s_userdict");
  28. // 组织拥有者 转让
  29. define("RoleOrgOwner" , "r_owner");
  30. // 组织管理员 创建 删除 修改 文章/文集模版 channel group
  31. define("RoleOrgAdmin" , "r_admin");
  32. // 组织编辑 修改译文
  33. define("RoleOrgEditor" , "r_editor");
  34. // 组织成员 读取任意资源
  35. define("RoleOrgMember" , "r_member");
  36. // 组织访客(比如未注册用户) 只读取公开资源
  37. define("RoleOrgVisitor" , "r_visitor");
  38. // 权限:角色分组
  39. define("GroupRole" , "g");
  40. // 权限:资源分组
  41. define("GroupRes" , "g2");
  42. // 权限:阅读权限
  43. define("PermRead" , "p_read");
  44. // 权限:翻译权限
  45. define("PermTrans" , "p_trans");
  46. // 权限:修改权限
  47. define("PermWrite" , "p_write");
  48. // 权限:创建
  49. define("PermCreate" , "p_create");
  50. // 权限:删除
  51. define("PermDelete" , "p_delete");
  52. // 权限:修改
  53. define("PermUpdate" , "p_update");
  54. $config = [
  55. 'database_type' => Database["type"],
  56. 'server' => Database['server'],
  57. 'database_name' => Database['name'],
  58. 'username' => Database['user'],
  59. 'password' => Database['password'],
  60. 'port' => Database['port'],
  61. ];
  62. $adapter = DatabaseAdapter::newAdapter($config);
  63. $e = new Enforcer('rbac.model.conf', $adapter);
  64. {
  65. /*
  66. * 按照业务逻辑测试验证
  67. */
  68. echo("// 创建新用户 zhang3,等于同时创建了 Org: zhang3,只是 OrgID 和 UserID 相同");
  69. CreateOrg("zhang3", "zhang3", $e);
  70. echo("// 用户 zhang3 创建了 版本风格 chinese-01");
  71. CreateChannel("chinese-01", "zhang3", $e);
  72. echo("// 将channel chinese-01 加入到组织 zhang3/group1 里");
  73. AddChannelReader1("chinese-01", "wang5", "group1", $e);
  74. echo("// 用户 zhang3 创建了 版本风格 chinese-02");
  75. CreateChannel("chinese-02", "zhang3", $e);
  76. echo("// 用户 zhang3 在组织 zhang3 下创建了 文章模板 article-01");
  77. CreateArticle("article-01", "zhang3", $e);
  78. echo("// 用户 zhang3 基于 版本风格 chinese-01 和 文章 article-01 创建了 译文 chinses-01+article-01");
  79. CreateTranslation("chinese-01", "article-01", "zhang3", $e);
  80. echo("// 用户 zhang3 基于 版本风格 chinese-02 和 文章 article-01 创建了 译文 chinses-02+article-01");
  81. CreateTranslation("chinese-02", "article-01", "zhang3", $e);
  82. testPermission("zhang3", $e);
  83. echo("// 创建新用户 li4,等于同时创建了 Org: li4,只是 OrgID 和 UserID 相同");
  84. CreateOrg("li4", "li4", $e);
  85. testPermission("li4", $e);
  86. //echo("// 将用户 li4 加入到组织 zhang3 里");
  87. //AddOrgMember("zhang3", "li4", $e);
  88. echo("// 将用户 li4 加入到组织 zhang3/group1 里");
  89. AddOrgGroup("zhang3","group1", "li4", $e );
  90. testPermission("li4", $e);
  91. echo("// 创建新用户 wang5,等于同时创建了 Org: wang5,只是 OrgID 和 UserID 相同");
  92. CreateOrg("wang5", "wang5", $e);
  93. testPermission("wang5", $e);
  94. echo("// 将 chinese-01 分享给 wang5,只读 ");
  95. AddChannelReader("chinese-01", "zhang3", "wang5", $e);
  96. echo("// 将 article-01 分享给 wang5,只读 ");
  97. AddArticleReader("article-01", "zhang3", "wang5", $e);
  98. testPermission("wang5", $e);
  99. echo("// 将 chinese-01 分享给 wang5,可以翻译 ");
  100. AddChannelTranslator("chinese-01", "zhang3", "wang5", $e);
  101. testPermission("wang5", $e);
  102. echo("// 创建新用户 zhao6,等于同时创建了 Org: zhao6,只是 OrgID 和 UserID 相同");
  103. CreateOrg("zhao6", "zhao6", $e);
  104. testPermission("zhao6", $e);
  105. echo("// 将 译文 chinese-01+article-01 分享给 zhao6,翻译权限");
  106. AddTranslationTranslator("chinese-01", "article-01", "zhang3", "zhao6", $e);
  107. echo("// 将 译文 chinese-02+article-01 分享给 zhao6,查看权限");
  108. AddTranslationReader("chinese-02", "article-01", "zhang3", "zhao6", $e);
  109. testPermission("zhao6", $e);
  110. }
  111. function testPermission( $userID , $e ) {
  112. echo("\n");
  113. echo("// ---- $userID 是否有权限 查看 组织 zhang3 的文章模板 article-01?");
  114. $r_a = UserCanReadArticle($userID, "zhang3", "article-01", $e);
  115. echo($r_a."\n");
  116. echo("// ---- $userID 是否有权限 查看 组织 zhang3 的版本风格 chinese-01?");
  117. $r_c= UserCanReadChannel($userID, "zhang3", "chinese-01", $e);
  118. echo($r_c ." \n" );
  119. echo("// ---- $userID 是否有权限 修改 组织 zhang3 的版本风格 chinese-01?");
  120. $r_c_w= UserCanWriteChannel($userID, "zhang3", "chinese-01", $e);
  121. echo($r_c_w." \n");
  122. echo("// ---- $userID 是否有权限 查看 组织 zhang3 的版本风格 chinese-02?");
  123. $r_c02= UserCanReadChannel($userID, "zhang3", "chinese-02", $e);
  124. echo(" $r_c02\n");
  125. echo("// ---- $userID 是否有权限 修改 组织 zhang3 的版本风格 chinese-02?");
  126. $r_c_w02= UserCanWriteChannel($userID, "zhang3", "chinese-02", $e);
  127. echo(" $r_c_w02 \n");
  128. echo("// ---- $userID 是否有权限 基于 组织 zhang3 的版本风格 chinese-01 进行翻译?");
  129. $r_t= UserCanTranslateChannel($userID, "zhang3", "chinese-01", $e);
  130. echo(" $r_t \n");
  131. echo("// ---- $userID 是否能查看 组织 zhang3 的译文 chinese-01+article-01?");
  132. $r_tt= UserCanReadTranslation($userID, "zhang3", "chinese-01", "article-01", $e);
  133. echo(" $r_tt \n");
  134. echo("// ---- $userID 是否能翻译 组织 zhang3 的译文 chinese-01+article-01?");
  135. $r_tt1= UserCanTranslateTranslation($userID, "zhang3", "chinese-01", "article-01", $e);
  136. echo(" $r_tt1 \n" );
  137. echo("// ---- $userID 是否能查看 组织 zhang3 的译文 chinese-02+article-01?");
  138. $r1_tt= UserCanReadTranslation($userID, "zhang3", "chinese-02", "article-01", $e);
  139. echo(" $r1_tt \n");
  140. echo("// ---- $userID 是否能翻译 组织 zhang3 的译文 chinese-02+article-01?");
  141. $r1_tt1= UserCanTranslateTranslation($userID, "zhang3", "chinese-02", "article-01", $e);
  142. echo(" $r1_tt1 \n");
  143. echo("\n");
  144. }
  145. function CreateOrg($orgID ,$userID , $e ) {
  146. // 将该用户设置为组织机构的管理员
  147. $e->AddNamedGroupingPolicy(GroupRole, IDPrefixUser.$userID, RoleOrgAdmin, IDPrefixOrg.$orgID);
  148. // 添加 admin 资源操作权限
  149. $e->AddNamedPolicy("p", RoleOrgAdmin, IDPrefixOrg.$orgID, ResArticle, ".*");
  150. $e->AddNamedPolicy("p", RoleOrgAdmin, IDPrefixOrg.$orgID, ResChannel, ".*");
  151. $e->AddNamedPolicy("p", RoleOrgAdmin, IDPrefixOrg.$orgID, ResTranslation, ".*");
  152. // 添加 member 资源操作权限
  153. $e->AddNamedPolicy("p", RoleOrgMember, IDPrefixOrg.$orgID, ResArticle, PermRead);
  154. $e->AddNamedPolicy("p", RoleOrgMember, IDPrefixOrg.$orgID, ResChannel, PermRead);
  155. $e->AddNamedPolicy("p", RoleOrgMember, IDPrefixOrg.$orgID, ResTranslation, PermRead);
  156. }
  157. function AddOrgMemberToGroup($orgID, $groupID, $userID, $e ){
  158. // g, li4, translator, org1
  159. $e->AddNamedGroupingPolicy(GroupRole, IDPrefixUser . $userID, IDPrefixOrgGroup.$groupID, IDPrefixOrg.$orgID);
  160. }
  161. /*
  162. * 「为组织的group添加新成员」
  163. * 加入 group 分组,拥有组织资源读取权限
  164. */
  165. function AddOrgGroup($orgID,$groupId, $userID, $e ){
  166. $e->AddNamedGroupingPolicy(GroupRole, IDPrefixUser . $userID, $groupId, IDPrefixOrg . $orgID);
  167. }
  168. /*
  169. * 「为组织添加新成员」
  170. * 加入 member 分组,拥有组织资源读取权限
  171. */
  172. function AddOrgMember($orgID, $userID, $e ){
  173. $e->AddNamedGroupingPolicy(GroupRole, IDPrefixUser . $userID, RoleOrgMember, IDPrefixOrg . $orgID);
  174. }
  175. /*
  176. * 「为组织添加新管理员」
  177. * 加入 admin 分组,拥有组织资源一切权限 除了转让
  178. */
  179. function AddOrgAdmin($orgID, $userID, $e ){
  180. $e->AddNamedGroupingPolicy(GroupRole, IDPrefixUser . $userID, RoleOrgAdmin, IDPrefixOrg . $orgID);
  181. }
  182. function CreateChannel($channelID, $orgID, $e ){
  183. // 将该 Channel 资源放入本组织的 channel 分组
  184. $e->AddNamedGroupingPolicy(GroupRes, IDPrefixChannel . $channelID, ResChannel, IDPrefixOrg . $orgID);
  185. }
  186. /*
  187. * 「为版本风格添加只读用户」,也即是「分享版本风格」-「查看者」
  188. * 操作之后,该用户可以访问此版本风格下的所有译文
  189. */
  190. function AddChannelReader($channelID, $orgID, $userID, $e ){
  191. $e->AddNamedPolicy("p", IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermRead);
  192. }
  193. function AddChannelReader1($channelID, $orgID, $RoleID, $e ){
  194. $e->AddNamedPolicy("p", $RoleID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermRead);
  195. }
  196. function AddChannelReaderGroup($channelID, $orgID, $groupID, $e ){
  197. $e->AddNamedPolicy("p", IDPrefixOrgGroup . $groupID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermRead);
  198. }
  199. /*
  200. * 「为版本风格添加翻译用户」,也即是「分享版本风格」-「编辑者」
  201. * 操作之后,该用户可以编辑此版本风格下的所有译文
  202. * 注意:该权限并不能编辑「版本风格」本身
  203. */
  204. function AddChannelTranslator($channelID, $orgID, $userID, $e ){
  205. $e->AddNamedPolicy("p", IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermTrans);
  206. }
  207. function AddChannelTranslatorGroup($channelID, $orgID, $groupID, $e ){
  208. $e->AddNamedPolicy("p", IDPrefixOrgGroup . $groupID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermTrans);
  209. }
  210. /*
  211. * 「为版本风格添加编辑用户」
  212. * 操作之后,该用户可以编辑此版本风格本身,比如:
  213. * 分享此版本风格、修改版本风格的描述等等
  214. * //TODO: 此功能是否需要?
  215. */
  216. function AddChannelWriter($channelID, $orgID, $userID, $e ) {
  217. $e->AddNamedPolicy("p", IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermWrite);
  218. //return "", errors.New("Do we realy need this functiontion?")
  219. }
  220. function AddChannelWriterGroup($channelID, $orgID, $groupID, $e ) {
  221. $e->AddNamedPolicy("p", IDPrefixOrgGroup . $groupID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermWrite);
  222. //return "", errors.New("Do we realy need this functiontion?")
  223. }
  224. function CreateArticle($articleID, $orgID, $e ){
  225. // 将该 Article 资源放入本组织的 article 分组
  226. $e->AddNamedGroupingPolicy("g2", IDPrefixArticle.$articleID, ResArticle, IDPrefixOrg . $orgID);
  227. }
  228. /*
  229. * 「为文章添加只读用户」,也即是「分享文章」-「查看者」
  230. * 操作之后,该用户可以访问此文章模板,
  231. * 对于能否查看对应译文,需要由对应 channel 的权限来判定
  232. */
  233. function AddArticleReader($articleID, $orgID, $userID, $e ){
  234. $e->AddNamedPolicy("p", IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixArticle.$articleID, PermRead);
  235. }
  236. function AddArticleReaderGroup($articleID, $orgID, $groupID, $e ){
  237. $e->AddNamedPolicy("p", IDPrefixOrgGroup . $groupID, IDPrefixOrg . $orgID, IDPrefixArticle.$articleID, PermRead);
  238. }
  239. /*
  240. * 「为文章添加编辑用户」,也即是「分享文章」-「编辑者」
  241. * 操作之后,该用户可以编辑此文章模板,
  242. * 对于能否查看、或修改对应译文,需要由对应 channel 的权限来判定
  243. */
  244. function AddArticleWriter($articleID, $orgID, $userID, $e ){
  245. $e->AddNamedPolicy("p", IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixArticle.$articleID, PermWrite);
  246. }
  247. function AddArticleWriterGroup($articleID, $orgID,$groupID, $e ){
  248. $e->AddNamedPolicy("p", IDPrefixOrgGroup . $groupID, IDPrefixOrg . $orgID, IDPrefixArticle.$articleID, PermWrite);
  249. }
  250. /*
  251. * 创建译文 = 版本风格 + 文章模板
  252. * 所谓 “译文是由 版本风格 和 文章模板 组成”,这是一个幻象。
  253. * 当我们参与翻译的时候,最终的翻译结果是对应的经文句子编号,如:{{125-2347-2-14}},
  254. * 而不是 “翻译了这篇文章”,也就是说当前的翻译结果,可以在任何引用了该句子编号的文章里找到。
  255. *
  256. * 所以,与其说是 “创建了译文”,不如说是:
  257. * “通过在当前「文章模板」下基于某「版本风格」翻译该「句子编号」,
  258. * 导致该「文章模板」与该「版本风格」发生了「我曾经在该文章下使用该风格翻译了某句话」的关系”
  259. *
  260. * 那么,我们为什么要创建还要单独译文呢?
  261. * 1. 将「版本风格」与「文章模板」绑定,方便译者查找自己未完成的工作
  262. * 2. 单独创建的译文,可以单独分享,而不用分享整个「版本风格」加上单独分享「文章模板」
  263. * 2.1 如果单独分享了「版本风格」,那么该用户使用该风格翻译的其他「句子」也会被同时分享
  264. * 2.2 如果单独分享「文章模板」以及「版本风格」,两者并没有发生关联,
  265. * 那么被分享者也难以找到如此配对的资源
  266. *
  267. * 那么,如何判断当前访问的资源,是「文章对应的译文」还是「圣典对应的译文」呢?
  268. * 1. app/article?... 对应的资源便是「文章,以及对应的译文」
  269. * 2. app/reader?... 对应的资源便是「圣典,以及对应的译文」
  270. *
  271. * 那么,我们如何判断他们的权限呢?
  272. * 0. 如果以上两个资源都没有传递 channel 参数,则表示「不关特别联任何译文」
  273. * 则不需要做任何权限判断(可选择公开的版本风格显示译文)
  274. * 1. app/article?id=01&channel=01... article - 01 关联了 channel - 01
  275. * 1.1 首先该用户是否具有 channel - 01 和 article - 01 组成的译文的权限,如果有则授予
  276. 这样以来,用户可以单独分享「译文」,而不必分享「版本风格」+「文章模板」了,
  277. 同时也满足了「仅仅希望与其他同学一起编辑某一篇文章」的需求
  278. * 1.2 判断该用户是否拥有 article - 01 的权限,如果有,则进行 1.3 判断
  279. * 1.3 判断该用户是否具有 channel - 01 的权限,如果有,则授予,否则即没有权限
  280. * 2. app/reader?channel=01... 关联了 channel - 01
  281. * 2.1 判断该用户是否拥有 channel - 01 的权限即可
  282. *
  283. * 以上方法对于「写入译文」时的权限判断,同理可推得。推不得的话,我们讨论讨论。
  284. */
  285. function CreateTranslation($channelID, $articleID, $orgID, $e ){
  286. // 将该 Translation 资源放入本组织的 Translation 分组
  287. // ID 由 channelID + articleID 构成
  288. $e->AddNamedGroupingPolicy("g2", IDPrefixTranslation . $channelID."+".$articleID, ResTranslation, $orgID);
  289. }
  290. /*
  291. * 「为译文添加只读用户」,也即是仅分享「某版本风格」对应的「某篇文章模板」
  292. * 操作之后,该用户可以访问此译文,但
  293. * 并不能:单独访问该「文章模板」,或,访问该「版本风格」对应的其他译文,
  294. * 以上两种权限不在此处授予。
  295. */
  296. function AddTranslationReader($channelID, $articleID, $orgID, $userID, $e ){
  297. $e->AddNamedPolicy("p", IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixTranslation . $channelID."+".$articleID, PermRead);
  298. }
  299. function AddTranslationReaderGroup($channelID, $articleID, $orgID, $groupID, $e ){
  300. $e->AddNamedPolicy("p", IDPrefixOrgGroup . $groupID, IDPrefixOrg . $orgID, IDPrefixTranslation . $channelID."+".$articleID, PermRead);
  301. }
  302. /*
  303. * 「为译文添加翻译用户」,也即是仅分享「某版本风格」对应的「某篇文章模板」,并允许翻译
  304. * 操作之后,该用户可以访问、修改此译文,但
  305. * 并不能:单独访问该「文章模板」,或,访问或修改该「版本风格」对应的其他译文,
  306. * 以上两种权限不在此处授予。
  307. */
  308. function AddTranslationTranslator($channelID, $articleID, $orgID, $userID, $e ){
  309. $e->AddNamedPolicy("p", IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixTranslation . $channelID."+".$articleID, PermTrans);
  310. }
  311. function AddTranslationTranslatorGroup($channelID, $articleID, $orgID, $groupID, $e ){
  312. $e->AddNamedPolicy("p", IDPrefixOrgGroup . $groupID, IDPrefixOrg . $orgID, IDPrefixTranslation . $channelID."+".$articleID, PermTrans);
  313. }
  314. /*
  315. * //TODO 是否有这个需求?
  316. * 好像不需要将译文的修改权限分享出去,用户需要的是翻译。
  317. * 即便分享出去,能做什么呢?解除绑定?
  318. */
  319. function AddTranslationWriter($channelID, $articleID, $orgID, $userID, $e ) {
  320. //return "", errors.New("Do we realy need this functiontion?")
  321. }
  322. function AddTranslationWriterGroup($channelID, $articleID, $orgID, $groupID, $e ) {
  323. //return "", errors.New("Do we realy need this functiontion?")
  324. }
  325. function UserCanReadArticle($userID, $orgID, $articleID, $e ) {
  326. return $e->Enforce(IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixArticle.$articleID, PermRead.PermWrite);
  327. }
  328. function UserCanReadChannel($userID, $orgID, $channelID, $e ) {
  329. return $e->Enforce(IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermRead.PermWrite.PermTrans);
  330. }
  331. function UserCanWriteChannel($userID, $orgID, $channelID, $e ) {
  332. return $e->Enforce(IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermWrite);
  333. }
  334. function UserCanTranslateChannel($userID, $orgID, $channelID, $e ) {
  335. return $e->Enforce(IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixChannel . $channelID, PermTrans);
  336. }
  337. function UserCanReadTranslation($userID, $orgID, $channelID, $articleID, $e ) {
  338. // 先基于译文判断
  339. $r = $e->Enforce(IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixTranslation . $channelID."+".$articleID, PermRead.PermTrans);
  340. if (!$r) {
  341. // 再基于 版本风格 和 文章模板联合判断
  342. $r_channel = UserCanReadChannel($userID, $orgID, $channelID, $e);
  343. $r_article = UserCanReadArticle($userID, $orgID, $articleID, $e);
  344. if ($r_channel && $r_article) {
  345. return true;
  346. }
  347. }
  348. return $r;
  349. }
  350. function UserCanTranslateTranslation($userID, $orgID, $channelID, $articleID, $e ) {
  351. // 先基于译文判断
  352. $r = $e->Enforce(IDPrefixUser . $userID, IDPrefixOrg . $orgID, IDPrefixTranslation . $channelID."+".$articleID, PermTrans);
  353. if (!$r) {
  354. // 再基于 版本风格 和 文章模板联合判断
  355. $r_channel = UserCanTranslateChannel($userID, $orgID, $channelID, $e);
  356. $r_article = UserCanReadArticle($userID, $orgID, $articleID, $e);
  357. if ($r_channel && $r_article) {
  358. return true;
  359. }
  360. }
  361. return $r;
  362. }