自动发现项目中的所有URL
我的rbac组件,是想用到任何一个,项目中的。 so 问题来了。
- 问题: 拿到一个项目。 怎样获取到,当前项目中, 所有的URL 以及 每个URL的别名name, 还有是有 namespace 命名空间。
- 实现思路:
1. 先要确定我们 根级 路由在哪里。 就是和项目文件同名的包中, 的 urls.py 中的 urlpatterns=[....] 这个路由的位置在我们的settings中是有进行配置的
ROOT_URLCONF = 'learn_formset.urls' 当然这个是可以进行修改的。
那个怎么才能获得,这个 urlpatterns 列表中的所有的对象呢?
直接导入的话,也是可以的。 但是更好的是 通过settings 中 ROOT_URLCONF 的字符串来进行导入。
Django中有一个模块就是来做这件事的, from django.utils.module_loading import import_string 这个模块就是用来根据一个字符串来导入相应的模块。
他的返回值,就是一个模块对象 <class 'module'> 。 可以通过 点 语法。获取到其中的 urlpatterns 列表。

看一看,打印 urlpatterns 里面的没一个数据, 能得到啥:
from django.conf import settings
from django.utils.module_loading import import_string def get_all_url_dict():
'''
递归获取项目中,所有的url 保存到字典
:return:
'''
md = import_string(settings.ROOT_URLCONF) # <class 'module'>
for item in md.urlpatterns:
print(item) # <URLResolver <URLPattern list> (admin:admin) '^admin/'> # <URLResolver <module 'web.urls' from 'D:\\crm_learn\\web\\urls.py'> (None:None) '^'> # <URLResolver <module 'rbac.urls' from 'D:\\crm_learn\\rbac\\urls.py'> (rbac:rbac) '^rbac/'> def multi_permissions(request):
'''
批量操作权限
:param request:
:return:
'''
# 获取项目中,所有的URL
get_all_url_dict()
return HttpResponse("OK")
可以看到的是,每一个item 都是一个 URLResolver 对象。这表示这是一个 路由分发的url。
但是 还有一种是 对应的视图函数的。URLPattern 对象。表示他对应的是一个 视图函数。
看代码吧! 这是 Django 2.0 版本的。 和 1 版本有不小的改动。
from collections import OrderedDict
from django.conf import settings
from django.utils.module_loading import import_string
from django.urls.resolvers import URLResolver, URLPattern def check_url_exclude(url):
'''自定制,过滤一下。 以 xxx 为前缀的 url'''
exclude_url = [
"/admin/.*",
"/login/",
]
for regex in exclude_url:
if re.match(regex, url):
return True def recursion_urls(pre_namespace, pre_url, urlpatterns, url_ordered_dict):
'''
递归获取,所有的url
:param pre_namespace: namespace前缀,用于拼接name (namespace:name)
:param pre_url: url的前缀, 用于拼接url
:param urlpatterns: 路由关系列表
:param url_ordered_dict: 用于保存递归中获取的所有的路由
:return:
'''
for item in urlpatterns:
if isinstance(item, URLPattern): # 表示一个 非路由分发。将路由添加到字典中
if not item.name: # 判断这个url 有没有,name别名
continue
name = item.name
if pre_namespace: # 判断当前这个url是不是有namespace前缀。也就是:是否是某一个命名空间中的 name别名
name = "%s:%s" % (pre_namespace, item.name)
url = (pre_url + str(item.pattern)).replace("^", "").replace("$", "")
if check_url_exclude(url): # 在这里进行自定制的过滤。 过露出我不想要的 哪些url
continue
url_ordered_dict[name] = {"name": name, "url": url} elif isinstance(item, URLResolver): # 表示这是一个路由分发。 这里就需要递归了
namespace = pre_namespace
if pre_namespace: # 如果有前缀
if item.namespace: # 自己有没有namespace
namespace = "%s:%s" % (pre_namespace, item.namespace)# 把之前的pre_namespace 和当前的 item.namespace 拼接。 传给下一次的递归函数。继续进行拼接
else:
if item.namespace:
namespace = item.namespace
recursion_urls(namespace, pre_url + str(item.pattern), item.url_patterns, url_ordered_dict)
# 进入下一次循环之前,pre_url + str(item.pattern) 要拼接上这一次循环的 url。
# item.url_patterns这一次是 URLResolver 对象的 url_patterns。 中间要加一个 _ 烦得很。 第一次是通过导入拿到的 模块对象。
# 但是 递归中的不是 模块对象。是一个URLResolver对象。 所以要加一个 _ 。下划线 def get_all_url_dict():
'''
获取项目中,所有的url 保存到字典(前提是,每个url必须有name别名)
:return:
'''
url_ordered_dict = OrderedDict()
md = import_string(settings.ROOT_URLCONF) recursion_urls(None, "/", md.urlpatterns, url_ordered_dict)
# 递归的去获取所有的路由。
# 第一次循环时,肯定是从 根路由开始, 所以没有前缀 传一个None.
# "/" 也是因为,第一次循环时。 所有的url 都没有前导 的 "/" 手动的加上。
# md.urlpatterns 要循环的这个列表。
# url_ordered_dict 保存所有url 的字典。
return url_ordered_dict def multi_permissions(request):
'''
批量操作权限
:param request:
:return:
'''
# 获取项目中,所有的URL
all_url_dict = get_all_url_dict()
print(all_url_dict)
return HttpResponse("OK")
这里是1.0版本的。 主要是,几个关键的参数获取的位置上。 有些不同!
from collections import OrderedDict
from django.conf import settings
from django.utils.module_loading import import_string
from django.urls.resolvers import RegexURLResolver, RegexURLPattern def recursion_urls(pre_namespace, pre_url, urlpatterns, url_ordered_dict):
for item in urlpatterns:
if isinstance(item, RegexURLPattern): # 表示一个 非路由分发。将路由添加到字典中
if not item.name: # 判断这个url 有没有,name别名
continue if pre_namespace: # 判断当前这个url是不是有namespace前缀。如果有就说明是通过路由分发过来的,
# 就要加上, 他上级的名称空间
name = "%s:%s" % (pre_namespace, item.name)
else:
name = item.name
url = pre_url + str(item._regex)
url_ordered_dict[name] = {"name": name, "url": url.replace("^", "").replace("$", "")} elif isinstance(item, RegexURLResolver): # 表示这是一个路由分发。 这里就需要递归了
if pre_namespace: # 如果有前缀
if item.namespace: # 自己有没有namespace
namespace = "%s:%s" % (pre_namespace, item.namespace)
else:
namespace = pre_namespace
# 把之前的pre_namespace 和当前的 item.namespace 拼接。 传给下一次的递归函数。继续进行拼接
else:
if item.namespace:
namespace = item.namespace
else:
namespace = None
recursion_urls(namespace, pre_url + str(item._regex.pattern), item.url_patterns, url_ordered_dict) def get_all_url_dict():
url_ordered_dict = OrderedDict()
md = import_string(settings.ROOT_URLCONF) recursion_urls(None, "/", md.urlpatterns, url_ordered_dict)
return url_ordered_dict def multi_permissions(request):
# 获取项目中,所有的URL
all_url_dict = get_all_url_dict()
print(all_url_dict)
return HttpResponse("OK")
未经过实际测试。不想用1版本的
我的这个, 中间做了一个限制, 就是 必须要确保每一个。 url 必须要有一个别名。 没有别名的就直接跳过了。
因为,如果使用我的rbac 组件 我需要这个别名, 进行按钮粒度的控制。 必须要有。
但是我没有让他报错, 因为还是有一些,是不需要起别名的。
然后就是把它放到,一个固定的地方了。 放到rbac这个app 哪里都行。 用的时候就导入 get_all_url_dict() 就行了
自动发现项目中的所有URL的更多相关文章
- django2自动发现项目中的url
根据路飞学城luffycity.com 的crm项目修改的 1 url入口:rbac/urls.py urlpatterns = [ ... # 批量操作权限 re_path(r'^multi/per ...
- 权限组件(12):自动发现项目中有别名的URL
自动发现项目中所有有别名的URL,效果如下: customer_list {'name': 'customer_list', 'url': '/customer/list/'} customer_ad ...
- 自动发现项目中的url
def check_url_exclude(url): """ 判断url是否需要自动被发现,如果不是则移除 :param url: 自动发现的url :return: ...
- 自动发现项目中的URL,django1版本和django2版本
一.django 1 版本 routers.py import re from collections import OrderedDict from django.conf import setti ...
- Django自动获取项目中的全部URL
import re from collections import OrderedDict from django.conf import settings from django.utils.mod ...
- SpringMVC项目中获取所有URL到Controller Method的映射
Spring是一个很好很强大的开源框架,它就像是一个容器,为我们提供了各种Bean组件和服务.对于MVC这部分而言,它里面实现了从Url请求映射控制器方法的逻辑处理,在我们平时的开发工作中并不需要太多 ...
- zabbix自动发现功能实现批量web url监控
需求: 现在有大量url需要监控,形式如http://www.baidu.com ,要求url状态不为200即报警. 需求详细分析: 大量的url,且url经常变化,现在监控用的是zabbix,如果手 ...
- [Vscode插件] 自动编译项目中的Sass文件为CSS
插件名 : Live Sass Compiler 今天在VSCode中发现了一个自动watch项目目录下sass文件的插件,摆脱了在控制台中进行手动watch的繁琐. 安装好以后点击右下角即可自动编译 ...
- 安装使用Entity Framework Power Tool Bate4 (Code First)从已建好的数据自动生成项目中的对应Model(新手贴,望各位大侠给予指点)
从开始学习使用MVC以后,同时也开始接触EF,很多原理都不是太懂,只知道安装了EF以后,点击哪里可以生成数据库对应的Model,不用再自己手写Model.这里记录的就是如何从已建立好的数据库生成项目代 ...
随机推荐
- ZooKeeper自定义数据日志目录
安装版本:zookeeper-3.4.10 问题描述: ZooKeeper在启动时会将zookeeper.out输出到当前目录,不仅不友好,有时候可能会因为目录权限问题引发一些不必要的麻烦. 脚本分析 ...
- MyBatis基础-1
1.Mybatis简介 2.Mybatis环境搭建 3.Mybatis的开发方式 一.什么框架 框架其本质是半成品程序,提供相关规范,并且提供大量可重用的组件. 目的:让开发者开发出结构比较良好,可读 ...
- React/anu实现Touchable
在RN中有一个叫Touchable 的组件,这里我们重演如何实现它. Touchable存在的意义是屏蔽click的问题.移动端与手机的click 在一些浏览器是有差异,比如说著名的300ms延迟. ...
- 吴裕雄 30-MySQL 及 SQL 注入
如果您通过网页获取用户输入的数据并将其插入一个MySQL数据库,那么就有可能发生SQL注入安全的问题.本章节将为大家介绍如何防止SQL注入,并通过脚本来过滤SQL中注入的字符.所谓SQL注入,就是通过 ...
- Hadoop集群(二) HDFS搭建
HDFS只是Hadoop最基本的一个服务,很多其他服务,都是基于HDFS展开的.所以部署一个HDFS集群,是很核心的一个动作,也是大数据平台的开始. 安装Hadoop集群,首先需要有Zookeeper ...
- SAP 表类型
最近看了一下ABAP的表类型相关知识,做一个自己的分析总结. ABAP的表类型似乎我感觉只应该有2种,而不是大家说的3种. 从数据存储的原理上来看的.下面我做一个分析. 一般情况下,我们都说ABAP中 ...
- linux安装node简单方法
1.去官网下载和自己系统匹配的文件: 英文网址:https://nodejs.org/en/download/ 中文网址:http://nodejs.cn/download/ 通过 uname -a ...
- 解决SMARTFORMS文本编辑器不能打开
在DEV打开SMARTFORMS文本编辑器时,出现如下错误 由于宏安全设置,无法找到宏或宏被禁用. 解决方法如下: 在DEV环境新建程序后输入如下代码执行即可. *&------------- ...
- Light Probe Proxy Volume
[Light Probe Proxy Volume] The Light Probe Proxy Volume (LPPV) component allows you to use more ligh ...
- 读取excel表格以及生成自动化报告
数据库读取 标签(空格分隔): 数据库读取 读excel数据xlrd 当登录的账号有多个的时候,我们一般用excel存放测试数据,本节课介绍,python读取excel方法,并保存为字典格式. 1.先 ...