Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in
Toggle navigation
J
jumpserver
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
ops
jumpserver
Commits
26f68bc7
Commit
26f68bc7
authored
May 27, 2019
by
ibuler
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[Update] 修改asset表结构
parent
a5b874e2
Hide whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
86 additions
and
77 deletions
+86
-77
asset.py
apps/assets/api/asset.py
+2
-2
asset.py
apps/assets/models/asset.py
+48
-40
asset.py
apps/assets/serializers/asset.py
+2
-2
tasks.py
apps/assets/tasks.py
+1
-1
asset_asset_user_list.html
apps/assets/templates/assets/asset_asset_user_list.html
+2
-2
asset_detail.html
apps/assets/templates/assets/asset_detail.html
+9
-8
settings.py
apps/jumpserver/settings.py
+4
-4
inventory.py
apps/ops/inventory.py
+2
-2
command_execution_create.html
apps/ops/templates/ops/command_execution_create.html
+5
-2
user_permission.py
apps/perms/api/user_permission.py
+4
-5
asset_permission.py
apps/perms/utils/asset_permission.py
+7
-9
No files found.
apps/assets/api/asset.py
View file @
26f68bc7
...
...
@@ -168,8 +168,8 @@ class AssetGatewayApi(generics.RetrieveAPIView):
asset
=
get_object_or_404
(
Asset
,
pk
=
asset_id
)
if
asset
.
domain
and
\
asset
.
domain
.
gateways
.
filter
(
protocol
=
asset
.
protocol
)
.
exists
():
gateway
=
random
.
choice
(
asset
.
domain
.
gateways
.
filter
(
protocol
=
asset
.
protocol
))
asset
.
domain
.
gateways
.
filter
(
protocol
=
'ssh'
)
.
exists
():
gateway
=
random
.
choice
(
asset
.
domain
.
gateways
.
filter
(
protocol
=
'ssh'
))
serializer
=
serializers
.
GatewayWithAuthSerializer
(
instance
=
gateway
)
return
Response
(
serializer
.
data
)
else
:
...
...
apps/assets/models/asset.py
View file @
26f68bc7
...
...
@@ -48,6 +48,12 @@ class AssetQuerySet(models.QuerySet):
return
self
.
active
()
class
AssetManager
(
OrgManager
):
def
get_queryset
(
self
):
queryset
=
super
()
.
get_queryset
()
.
prefetch_related
(
"nodes"
,
"protocols"
)
return
queryset
class
Protocol
(
models
.
Model
):
PROTOCOL_SSH
=
'ssh'
PROTOCOL_RDP
=
'rdp'
...
...
@@ -127,7 +133,7 @@ class Asset(OrgModelMixin):
date_created
=
models
.
DateTimeField
(
auto_now_add
=
True
,
null
=
True
,
blank
=
True
,
verbose_name
=
_
(
'Date created'
))
comment
=
models
.
TextField
(
max_length
=
128
,
default
=
''
,
blank
=
True
,
verbose_name
=
_
(
'Comment'
))
objects
=
Org
Manager
.
from_queryset
(
AssetQuerySet
)()
objects
=
Asset
Manager
.
from_queryset
(
AssetQuerySet
)()
CONNECTIVITY_CACHE_KEY
=
'_JMS_ASSET_CONNECTIVITY_{}'
UNREACHABLE
,
REACHABLE
,
UNKNOWN
=
range
(
0
,
3
)
CONNECTIVITY_CHOICES
=
(
...
...
@@ -148,19 +154,39 @@ class Asset(OrgModelMixin):
return
True
,
''
return
False
,
warning
def
support_ansible
(
self
):
if
self
.
platform
in
(
"Windows"
,
"Windows2016"
,
"Other"
):
return
False
if
self
.
protocol
!=
'ssh'
:
return
False
return
True
@property
def
protocols_name
(
self
):
names
=
[]
for
protocol
in
self
.
protocols
.
all
():
names
.
append
(
protocol
.
name
)
return
names
def
has_protocol
(
self
,
name
):
return
name
in
self
.
protocols_name
def
get_protocol_by_name
(
self
,
name
):
for
i
in
self
.
protocols
.
all
():
if
i
.
name
.
lower
()
==
name
.
lower
():
return
i
return
None
@property
def
protocol_ssh
(
self
):
return
self
.
get_protocol_by_name
(
"ssh"
)
@property
def
protocol_rdp
(
self
):
return
self
.
get_protocol_by_name
(
"rdp"
)
def
is_unixlike
(
self
):
if
self
.
platform
not
in
(
"Windows"
,
"Windows2016"
):
if
self
.
platform
not
in
(
"Windows"
,
"Windows2016"
,
"Other"
):
return
True
else
:
return
False
def
is_support_ansible
(
self
):
return
self
.
has_protocol
(
'ssh'
)
and
self
.
platform
not
in
(
"Other"
,)
def
get_nodes
(
self
):
from
.node
import
Node
nodes
=
self
.
nodes
.
all
()
or
[
Node
.
root
()]
...
...
@@ -189,6 +215,15 @@ class Asset(OrgModelMixin):
filter_arg
|=
Q
(
Q
(
org_id__isnull
=
True
)
|
Q
(
org_id
=
''
),
hostname__in
=
hosts
)
return
Asset
.
objects
.
filter
(
filter_arg
)
@property
def
cpu_info
(
self
):
info
=
""
if
self
.
cpu_model
:
info
+=
self
.
cpu_model
if
self
.
cpu_count
and
self
.
cpu_cores
:
info
+=
"{}*{}"
.
format
(
self
.
cpu_count
,
self
.
cpu_cores
)
return
info
@property
def
hardware_info
(
self
):
if
self
.
cpu_count
:
...
...
@@ -232,35 +267,6 @@ class Asset(OrgModelMixin):
fake_node
.
is_node
=
False
return
fake_node
def
to_json
(
self
):
info
=
{
'id'
:
self
.
id
,
'hostname'
:
self
.
hostname
,
'ip'
:
self
.
ip
,
'port'
:
self
.
port
,
}
if
self
.
domain
and
self
.
domain
.
gateway_set
.
all
():
info
[
"gateways"
]
=
[
d
.
id
for
d
in
self
.
domain
.
gateway_set
.
all
()]
return
info
def
_to_secret_json
(
self
):
"""
Ansible use it create inventory
Todo: May be move to ops implements it
"""
data
=
self
.
to_json
()
if
self
.
admin_user
:
self
.
admin_user
.
load_specific_asset_auth
(
self
)
admin_user
=
self
.
admin_user
data
.
update
({
'username'
:
admin_user
.
username
,
'password'
:
admin_user
.
password
,
'private_key'
:
admin_user
.
private_key_file
,
'become'
:
admin_user
.
become_info
,
'groups'
:
[
node
.
value
for
node
in
self
.
nodes
.
all
()],
})
return
data
def
as_tree_node
(
self
,
parent_node
):
from
common.tree
import
TreeNode
icon_skin
=
'file'
...
...
@@ -282,9 +288,11 @@ class Asset(OrgModelMixin):
'id'
:
self
.
id
,
'hostname'
:
self
.
hostname
,
'ip'
:
self
.
ip
,
'port'
:
self
.
port
,
'protocols'
:
[
{
"name"
:
p
.
name
,
"port"
:
p
.
port
}
for
p
in
self
.
protocols
.
all
()
],
'platform'
:
self
.
platform
,
'protocol'
:
self
.
protocol
,
}
}
}
...
...
@@ -308,10 +316,10 @@ class Asset(OrgModelMixin):
asset
=
cls
(
ip
=
'.'
.
join
(
ip
),
hostname
=
forgery_py
.
internet
.
user_name
(
True
),
admin_user
=
choice
(
AdminUser
.
objects
.
all
()),
port
=
22
,
created_by
=
'Fake'
)
try
:
asset
.
save
()
asset
.
protocols
.
create
(
name
=
"ssh"
,
port
=
22
)
if
nodes
and
len
(
nodes
)
>
3
:
_nodes
=
random
.
sample
(
nodes
,
3
)
else
:
...
...
apps/assets/serializers/asset.py
View file @
26f68bc7
...
...
@@ -8,7 +8,6 @@ from django.utils.translation import ugettext_lazy as _
from
orgs.mixins
import
OrgResourceSerializerMixin
from
common.mixins
import
BulkSerializerMixin
from
common.serializers
import
AdaptedBulkListSerializer
from
common.validators
import
ProjectUniqueValidator
from
..models
import
Asset
,
Protocol
from
.system_user
import
AssetSystemUserSerializer
...
...
@@ -24,7 +23,8 @@ class ProtocolSerializer(serializers.ModelSerializer):
fields
=
[
"name"
,
"port"
]
class
AssetSerializer
(
BulkSerializerMixin
,
OrgResourceSerializerMixin
,
serializers
.
ModelSerializer
):
class
AssetSerializer
(
BulkSerializerMixin
,
OrgResourceSerializerMixin
,
serializers
.
ModelSerializer
):
protocols
=
ProtocolSerializer
(
many
=
True
)
"""
...
...
apps/assets/tasks.py
View file @
26f68bc7
...
...
@@ -31,7 +31,7 @@ def check_asset_can_run_ansible(asset):
msg
=
_
(
"Asset has been disabled, skipped: {}"
)
.
format
(
asset
)
logger
.
info
(
msg
)
return
False
if
not
asset
.
support_ansible
():
if
not
asset
.
is_
support_ansible
():
msg
=
_
(
"Asset may not be support ansible, skipped: {}"
)
.
format
(
asset
)
logger
.
info
(
msg
)
return
False
...
...
apps/assets/templates/assets/asset_asset_user_list.html
View file @
26f68bc7
...
...
@@ -62,7 +62,7 @@
<div
class=
"panel-body"
>
<table
class=
"table"
>
<tbody>
{% if asset.
protocol == 'ssh'
%}
{% if asset.
is_support_ansible
%}
<tr
class=
"no-borders-tr"
>
<td>
{% trans 'Test connective' %}:
</td>
<td>
...
...
@@ -118,7 +118,7 @@ function initAssetUserTable() {
var
view_btn
=
' <a class="btn btn-xs btn-primary btn-view-auth" data-username="DEFAULT_USERNAME">{% trans "View auth" %}</a>'
.
replace
(
"DEFAULT_USERNAME"
,
cellData
);
var
test_btn
=
' <a class="btn btn-xs btn-info btn-test-connective" data-username="DEFAULT_USERNAME">{% trans "Test" %}</a>'
.
replace
(
"DEFAULT_USERNAME"
,
cellData
);
btn
+=
view_btn
;
{
%
if
asset
.
protocol
==
'ssh'
%
}
{
%
if
asset
.
is_support_ansible
%
}
btn
+=
test_btn
;
{
%
endif
%
}
$
(
td
).
html
(
btn
);
...
...
apps/assets/templates/assets/asset_detail.html
View file @
26f68bc7
...
...
@@ -69,12 +69,13 @@
<td><b>
{{ asset.public_ip|default:"" }}
</b></td>
</tr>
<tr>
<td>
{% trans 'Port' %}:
</td>
<td><b>
{{ asset.port }}
</b></td>
</tr>
<tr>
<td>
{% trans 'Protocol' %}:
</td>
<td><b>
{{ asset.protocol }}
</b></td>
<td>
{% trans 'Protocol' %}
</td>
<td>
{% for protocol in asset.protocols.all %}
<b>
{{ protocol.name }}:
</b>
{{ protocol.port }}
{% endfor %}
</td>
</tr>
<tr>
<td>
{% trans 'Admin user' %}:
</td>
...
...
@@ -94,7 +95,7 @@
</tr>
<tr>
<td>
{% trans 'CPU' %}:
</td>
<td><b>
{{ asset.cpu_
model|default:"" }} {{ asset.cpu_count|default:"" }}*{{ asset.cpu_cores|default:""
}}
</b></td>
<td><b>
{{ asset.cpu_
info
}}
</b></td>
</tr>
<tr>
<td>
{% trans 'Memory' %}:
</td>
...
...
@@ -166,7 +167,7 @@
</span>
</td>
</tr>
{% if asset.
protocol == 'ssh'
%}
{% if asset.
is_support_ansible
%}
<tr>
<td>
{% trans 'Refresh hardware' %}:
</td>
<td>
...
...
apps/jumpserver/settings.py
View file @
26f68bc7
...
...
@@ -300,10 +300,10 @@ LOGGING = {
'handlers'
:
[
'gunicorn_console'
,
'gunicorn_file'
],
'level'
:
'INFO'
,
},
#
'django.db': {
#
'handlers': ['console', 'file'],
#
'level': 'DEBUG'
#
}
'django.db'
:
{
'handlers'
:
[
'console'
,
'file'
],
'level'
:
'DEBUG'
}
}
}
...
...
apps/ops/inventory.py
View file @
26f68bc7
...
...
@@ -21,7 +21,7 @@ class JMSBaseInventory(BaseInventory):
'id'
:
asset
.
id
,
'hostname'
:
asset
.
hostname
,
'ip'
:
asset
.
ip
,
'port'
:
asset
.
port
,
'port'
:
asset
.
p
rotocol_ssh
.
p
ort
,
'vars'
:
dict
(),
'groups'
:
[],
}
...
...
@@ -73,7 +73,7 @@ class JMSInventory(JMSBaseInventory):
"""
def
__init__
(
self
,
assets
,
run_as_admin
=
False
,
run_as
=
None
,
become_info
=
None
):
"""
:param
host_id_list: ["test1", ]
:param
assets: assets
:param run_as_admin: True 是否使用管理用户去执行, 每台服务器的管理用户可能不同
:param run_as: 用户名(添加了统一的资产用户管理器之后AssetUserManager加上之后修改为username)
:param become_info: 是否become成某个用户去执行
...
...
apps/ops/templates/ops/command_execution_create.html
View file @
26f68bc7
...
...
@@ -134,8 +134,11 @@ function getSelectedAssetsNode() {
var
assetsNodeId
=
[];
var
assetsNode
=
[];
nodes
.
forEach
(
function
(
node
)
{
if
(
node
.
meta
.
type
===
'asset'
&&
!
node
.
isHidden
&&
node
.
meta
.
asset
.
protocol
===
'ssh'
)
{
if
(
assetsNodeId
.
indexOf
(
node
.
id
)
===
-
1
)
{
if
(
node
.
meta
.
type
===
'asset'
&&
!
node
.
isHidden
)
{
var
protocols
=
$
.
map
(
node
.
meta
.
asset
.
protocols
,
function
(
v
)
{
return
v
.
name
});
if
(
assetsNodeId
.
indexOf
(
node
.
id
)
===
-
1
&&
protocols
.
indexOf
(
"ssh"
)
>
-
1
)
{
assetsNodeId
.
push
(
node
.
id
);
assetsNode
.
push
(
node
)
}
...
...
apps/perms/api/user_permission.py
View file @
26f68bc7
...
...
@@ -154,7 +154,7 @@ class UserGrantedAssetsApi(UserPermissionCacheMixin, AssetsFilterMixin, ListAPIV
util
=
AssetPermissionUtil
(
user
,
cache_policy
=
self
.
cache_policy
)
assets
=
util
.
get_assets
()
for
k
,
v
in
assets
.
items
():
system_users_granted
=
[
s
for
s
in
v
if
s
.
protocol
==
k
.
protocol
]
system_users_granted
=
[
s
for
s
in
v
if
k
.
has_protocol
(
s
.
protocol
)
]
k
.
system_users_granted
=
system_users_granted
queryset
.
append
(
k
)
return
queryset
...
...
@@ -215,8 +215,7 @@ class UserGrantedNodesWithAssetsApi(UserPermissionCacheMixin, AssetsFilterMixin,
for
node
,
_assets
in
nodes
.
items
():
assets
=
_assets
.
keys
()
for
k
,
v
in
_assets
.
items
():
system_users_granted
=
[
s
for
s
in
v
if
s
.
protocol
==
k
.
protocol
]
system_users_granted
=
[
s
for
s
in
v
if
k
.
has_protocol
(
s
.
protocol
)]
k
.
system_users_granted
=
system_users_granted
node
.
assets_granted
=
assets
queryset
.
append
(
node
)
...
...
@@ -358,7 +357,7 @@ class UserGrantedNodeChildrenApi(UserPermissionCacheMixin, ListAPIView):
for
asset
,
system_users
in
nodes_granted
[
node
]
.
items
():
fake_node
=
asset
.
as_node
()
fake_node
.
assets_amount
=
0
system_users
=
[
s
for
s
in
system_users
if
s
.
protocol
==
asset
.
protocol
]
system_users
=
[
s
for
s
in
system_users
if
asset
.
has_protocol
(
s
.
protocol
)
]
fake_node
.
asset
.
system_users_granted
=
system_users
fake_node
.
key
=
node
.
key
+
':0'
fake_nodes
.
append
(
fake_node
)
...
...
@@ -383,7 +382,7 @@ class UserGrantedNodeChildrenApi(UserPermissionCacheMixin, ListAPIView):
fake_node
=
asset
.
as_node
()
fake_node
.
assets_amount
=
0
system_users
=
[
s
for
s
in
system_users
if
s
.
protocol
==
asset
.
protocol
]
asset
.
has_protocol
(
s
.
protocol
)
]
fake_node
.
asset
.
system_users_granted
=
system_users
fake_node
.
key
=
node
.
key
+
':0'
matched_assets
.
append
(
fake_node
)
...
...
apps/perms/utils/asset_permission.py
View file @
26f68bc7
...
...
@@ -182,7 +182,7 @@ class AssetPermissionUtil:
for
perm
in
permissions
:
for
asset
in
perm
.
assets
.
all
()
.
valid
()
.
prefetch_related
(
'nodes'
):
assets
[
asset
]
.
update
(
perm
.
system_users
.
filter
(
protocol
=
asset
.
protocol
)
perm
.
system_users
.
filter
(
protocol
__in
=
asset
.
protocols_name
)
)
return
assets
...
...
@@ -213,7 +213,7 @@ class AssetPermissionUtil:
_assets
=
node
.
get_all_assets
()
.
valid
()
.
prefetch_related
(
'nodes'
)
for
asset
in
_assets
:
assets
[
asset
]
.
update
(
[
s
for
s
in
system_users
if
s
.
protocol
==
asset
.
protocol
]
[
s
for
s
in
system_users
if
asset
.
has_protocol
(
s
.
protocol
)
]
)
self
.
_assets
=
assets
self
.
_setattr_actions_to_system_user
()
...
...
@@ -226,15 +226,12 @@ class AssetPermissionUtil:
resource
=
resource
)
@property
def
node_key
(
self
):
return
self
.
get_cache_key
(
'NODES_WITH_ASSETS'
)
@property
def
asset_key
(
self
):
return
self
.
get_cache_key
(
'ASSETS'
)
@property
def
system_key
(
self
):
return
self
.
get_cache_key
(
'SYSTEM_USER'
)
...
...
@@ -265,7 +262,8 @@ class AssetPermissionUtil:
tree
=
GenerateTree
()
for
asset
,
system_users
in
assets
.
items
():
tree
.
add_asset
(
asset
,
system_users
)
return
tree
.
get_nodes
()
nodes
=
tree
.
get_nodes
()
return
nodes
def
get_nodes_with_assets_from_cache
(
self
):
cached
=
cache
.
get
(
self
.
node_key
)
...
...
@@ -405,7 +403,7 @@ def parse_node_to_tree_node(node):
def
parse_asset_to_tree_node
(
node
,
asset
,
system_users
):
system_users_protocol_matched
=
[
s
for
s
in
system_users
if
s
.
protocol
==
asset
.
protocol
]
system_users_protocol_matched
=
[
s
for
s
in
system_users
if
asset
.
has_protocol
(
s
.
protocol
)
]
icon_skin
=
'file'
if
asset
.
platform
.
lower
()
==
'windows'
:
icon_skin
=
'windows'
...
...
@@ -438,8 +436,8 @@ def parse_asset_to_tree_node(node, asset, system_users):
'id'
:
asset
.
id
,
'hostname'
:
asset
.
hostname
,
'ip'
:
asset
.
ip
,
'p
ort'
:
asset
.
port
,
'protocol'
:
asset
.
protocol
,
'p
rotocols'
:
[{
"name"
:
p
.
name
,
"port"
:
p
.
port
}
for
p
in
asset
.
protocols
.
all
()]
,
'platform'
:
asset
.
platform
,
'domain'
:
None
if
not
asset
.
domain
else
asset
.
domain
.
id
,
'is_active'
:
asset
.
is_active
,
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment