Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Z
ZEO
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
ZEO
Commits
1c26a9ac
Commit
1c26a9ac
authored
Jun 10, 2002
by
Shane Hathaway
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Merged shane-activity-monitoring-branch.
parent
828c9df6
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
245 additions
and
6 deletions
+245
-6
src/ZODB/ActivityMonitor.py
src/ZODB/ActivityMonitor.py
+107
-0
src/ZODB/Connection.py
src/ZODB/Connection.py
+17
-2
src/ZODB/DB.py
src/ZODB/DB.py
+14
-4
src/ZODB/tests/testActivityMonitor.py
src/ZODB/tests/testActivityMonitor.py
+107
-0
No files found.
src/ZODB/ActivityMonitor.py
0 → 100644
View file @
1c26a9ac
##############################################################################
#
# Copyright (c) 2001, 2002 Zope Corporation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
# FOR A PARTICULAR PURPOSE
#
##############################################################################
"""ZODB transfer activity monitoring
$Id: ActivityMonitor.py,v 1.2 2002/06/10 20:20:44 shane Exp $"""
__version__
=
'$Revision: 1.2 $'
[
11
:
-
2
]
import
time
class
ActivityMonitor
:
"""ZODB load/store activity monitor
This simple implementation just keeps a small log in memory
and iterates over the log when getActivityAnalysis() is called.
It assumes that log entries are added in chronological sequence,
which is only guaranteed because DB.py holds a lock when calling
the closedConnection() method.
"""
def
__init__
(
self
,
history_length
=
3600
):
self
.
history_length
=
history_length
# Number of seconds
self
.
log
=
[]
# [(time, loads, stores)]
def
closedConnection
(
self
,
conn
):
log
=
self
.
log
now
=
time
.
time
()
loads
,
stores
=
conn
.
getTransferCounts
(
1
)
log
.
append
((
now
,
loads
,
stores
))
self
.
trim
(
now
)
def
trim
(
self
,
now
):
log
=
self
.
log
cutoff
=
now
-
self
.
history_length
n
=
0
loglen
=
len
(
log
)
while
n
<
loglen
and
log
[
n
][
0
]
<
cutoff
:
n
=
n
+
1
if
n
:
del
log
[:
n
]
def
setHistoryLength
(
self
,
history_length
):
self
.
history_length
=
history_length
self
.
trim
(
time
.
time
())
def
getHistoryLength
(
self
):
return
self
.
history_length
def
getActivityAnalysis
(
self
,
start
=
0
,
end
=
0
,
divisions
=
10
):
res
=
[]
log
=
self
.
log
now
=
time
.
time
()
if
start
==
0
:
start
=
now
-
self
.
history_length
if
end
==
0
:
end
=
now
for
n
in
range
(
divisions
):
res
.
append
({
'start'
:
start
+
(
end
-
start
)
*
n
/
divisions
,
'end'
:
start
+
(
end
-
start
)
*
(
n
+
1
)
/
divisions
,
'loads'
:
0
,
'stores'
:
0
,
})
div
=
res
[
0
]
div_start
=
div
[
'start'
]
div_end
=
div
[
'end'
]
div_index
=
0
total_loads
=
0
total_stores
=
0
for
t
,
loads
,
stores
in
self
.
log
:
if
t
<
start
:
# We could use a binary search to find the start.
continue
elif
t
>
end
:
# We could use a binary search to find the end also.
break
while
t
>
div_end
:
div
[
'loads'
]
=
total_loads
div
[
'stores'
]
=
total_stores
total_loads
=
0
total_stores
=
0
div_index
=
div_index
+
1
if
div_index
<
divisions
:
div
=
res
[
div_index
]
div_start
=
div
[
'start'
]
div_end
=
div
[
'end'
]
total_loads
=
total_loads
+
loads
total_stores
=
total_stores
+
stores
div
[
'stores'
]
=
div
[
'stores'
]
+
total_stores
div
[
'loads'
]
=
div
[
'loads'
]
+
total_loads
return
res
src/ZODB/Connection.py
View file @
1c26a9ac
...
@@ -13,8 +13,8 @@
...
@@ -13,8 +13,8 @@
##############################################################################
##############################################################################
"""Database connection support
"""Database connection support
$Id: Connection.py,v 1.6
6 2002/04/15 18:42:50 jeremy
Exp $"""
$Id: Connection.py,v 1.6
7 2002/06/10 20:20:44 shane
Exp $"""
__version__
=
'$Revision: 1.6
6
$'
[
11
:
-
2
]
__version__
=
'$Revision: 1.6
7
$'
[
11
:
-
2
]
from
cPickleCache
import
PickleCache
,
MUCH_RING_CHECKING
from
cPickleCache
import
PickleCache
,
MUCH_RING_CHECKING
from
POSException
import
ConflictError
,
ReadConflictError
from
POSException
import
ConflictError
,
ReadConflictError
...
@@ -83,6 +83,8 @@ class Connection(ExportImport.ExportImport):
...
@@ -83,6 +83,8 @@ class Connection(ExportImport.ExportImport):
self
.
_invalid
=
d
.
has_key
self
.
_invalid
=
d
.
has_key
self
.
_committed
=
[]
self
.
_committed
=
[]
self
.
_code_timestamp
=
global_code_timestamp
self
.
_code_timestamp
=
global_code_timestamp
self
.
_load_count
=
0
# Number of objects unghosted
self
.
_store_count
=
0
# Number of objects stored
def
_cache_items
(
self
):
def
_cache_items
(
self
):
# find all items on the lru list
# find all items on the lru list
...
@@ -383,6 +385,7 @@ class Connection(ExportImport.ExportImport):
...
@@ -383,6 +385,7 @@ class Connection(ExportImport.ExportImport):
dump
(
state
)
dump
(
state
)
p
=
file
(
1
)
p
=
file
(
1
)
s
=
dbstore
(
oid
,
serial
,
p
,
version
,
transaction
)
s
=
dbstore
(
oid
,
serial
,
p
,
version
,
transaction
)
self
.
_store_count
=
self
.
_store_count
+
1
# Put the object in the cache before handling the
# Put the object in the cache before handling the
# response, just in case the response contains the
# response, just in case the response contains the
# serial number for a newly created object
# serial number for a newly created object
...
@@ -486,6 +489,7 @@ class Connection(ExportImport.ExportImport):
...
@@ -486,6 +489,7 @@ class Connection(ExportImport.ExportImport):
try
:
try
:
p
,
serial
=
self
.
_storage
.
load
(
oid
,
self
.
_version
)
p
,
serial
=
self
.
_storage
.
load
(
oid
,
self
.
_version
)
self
.
_load_count
=
self
.
_load_count
+
1
# XXX this is quite conservative!
# XXX this is quite conservative!
# We need, however, to avoid reading data from a transaction
# We need, however, to avoid reading data from a transaction
...
@@ -692,6 +696,17 @@ class Connection(ExportImport.ExportImport):
...
@@ -692,6 +696,17 @@ class Connection(ExportImport.ExportImport):
def
getDebugInfo
(
self
):
return
self
.
_debug_info
def
getDebugInfo
(
self
):
return
self
.
_debug_info
def
setDebugInfo
(
self
,
*
args
):
self
.
_debug_info
=
self
.
_debug_info
+
args
def
setDebugInfo
(
self
,
*
args
):
self
.
_debug_info
=
self
.
_debug_info
+
args
def
getTransferCounts
(
self
,
clear
=
0
):
"""Returns the number of objects loaded and stored.
Set the clear argument to reset the counters.
"""
res
=
(
self
.
_load_count
,
self
.
_store_count
)
if
clear
:
self
.
_load_count
=
0
self
.
_store_count
=
0
return
res
######################################################################
######################################################################
# Just plain weird. Don't try this at home kids.
# Just plain weird. Don't try this at home kids.
...
...
src/ZODB/DB.py
View file @
1c26a9ac
...
@@ -13,8 +13,8 @@
...
@@ -13,8 +13,8 @@
##############################################################################
##############################################################################
"""Database objects
"""Database objects
$Id: DB.py,v 1.4
1 2002/04/15 18:42:51 jeremy
Exp $"""
$Id: DB.py,v 1.4
2 2002/06/10 20:20:44 shane
Exp $"""
__version__
=
'$Revision: 1.4
1
$'
[
11
:
-
2
]
__version__
=
'$Revision: 1.4
2
$'
[
11
:
-
2
]
import
cPickle
,
cStringIO
,
sys
,
POSException
,
UndoLogCompatible
import
cPickle
,
cStringIO
,
sys
,
POSException
,
UndoLogCompatible
from
Connection
import
Connection
from
Connection
import
Connection
...
@@ -33,7 +33,8 @@ class DB(UndoLogCompatible.UndoLogCompatible):
...
@@ -33,7 +33,8 @@ class DB(UndoLogCompatible.UndoLogCompatible):
or more connections, which manage object spaces. Most of the actual work
or more connections, which manage object spaces. Most of the actual work
of managing objects is done by the connections.
of managing objects is done by the connections.
"""
"""
klass
=
Connection
klass
=
Connection
# Class to use for connections
_activity_monitor
=
None
def
__init__
(
self
,
storage
,
def
__init__
(
self
,
storage
,
pool_size
=
7
,
pool_size
=
7
,
...
@@ -124,6 +125,9 @@ class DB(UndoLogCompatible.UndoLogCompatible):
...
@@ -124,6 +125,9 @@ class DB(UndoLogCompatible.UndoLogCompatible):
"""Return a connection to the pool"""
"""Return a connection to the pool"""
self
.
_a
()
self
.
_a
()
try
:
try
:
am
=
self
.
_activity_monitor
if
am
is
not
None
:
am
.
closedConnection
(
connection
)
version
=
connection
.
_version
version
=
connection
.
_version
pools
,
pooll
=
self
.
_pools
pools
,
pooll
=
self
.
_pools
pool
,
allocated
,
pool_lock
=
pools
[
version
]
pool
,
allocated
,
pool_lock
=
pools
[
version
]
...
@@ -486,6 +490,9 @@ class DB(UndoLogCompatible.UndoLogCompatible):
...
@@ -486,6 +490,9 @@ class DB(UndoLogCompatible.UndoLogCompatible):
})
})
return
r
return
r
def
getActivityMonitor
(
self
):
return
self
.
_activity_monitor
def
pack
(
self
,
t
=
None
,
days
=
0
):
def
pack
(
self
,
t
=
None
,
days
=
0
):
if
t
is
None
:
t
=
time
()
if
t
is
None
:
t
=
time
()
t
=
t
-
(
days
*
86400
)
t
=
t
-
(
days
*
86400
)
...
@@ -516,6 +523,9 @@ class DB(UndoLogCompatible.UndoLogCompatible):
...
@@ -516,6 +523,9 @@ class DB(UndoLogCompatible.UndoLogCompatible):
def
setPoolSize
(
self
,
v
):
def
setPoolSize
(
self
,
v
):
self
.
_pool_size
=
v
self
.
_pool_size
=
v
def
setActivityMonitor
(
self
,
am
):
self
.
_activity_monitor
=
am
def
setVersionCacheDeactivateAfter
(
self
,
v
):
def
setVersionCacheDeactivateAfter
(
self
,
v
):
self
.
_version_cache_deactivate_after
=
v
self
.
_version_cache_deactivate_after
=
v
for
ver
in
self
.
_pools
[
0
].
keys
():
for
ver
in
self
.
_pools
[
0
].
keys
():
...
...
src/ZODB/tests/testActivityMonitor.py
0 → 100644
View file @
1c26a9ac
##############################################################################
#
# Copyright (c) 2001, 2002 Zope Corporation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
# FOR A PARTICULAR PURPOSE.
#
##############################################################################
"""Tests of the default activity monitor.
See ZODB/ActivityMonitor.py
$Id: testActivityMonitor.py,v 1.2 2002/06/10 20:20:44 shane Exp $
"""
import
unittest
import
time
from
ZODB.ActivityMonitor
import
ActivityMonitor
class
FakeConnection
:
loads
=
0
stores
=
0
def
_transferred
(
self
,
loads
,
stores
):
self
.
loads
=
self
.
loads
+
loads
self
.
stores
=
self
.
stores
+
stores
def
getTransferCounts
(
self
,
clear
=
0
):
res
=
self
.
loads
,
self
.
stores
if
clear
:
self
.
loads
=
self
.
stores
=
0
return
res
class
Tests
(
unittest
.
TestCase
):
def
testAddLogEntries
(
self
):
am
=
ActivityMonitor
(
history_length
=
3600
)
self
.
assertEqual
(
len
(
am
.
log
),
0
)
c
=
FakeConnection
()
c
.
_transferred
(
1
,
2
)
am
.
closedConnection
(
c
)
c
.
_transferred
(
3
,
7
)
am
.
closedConnection
(
c
)
self
.
assertEqual
(
len
(
am
.
log
),
2
)
def
testTrim
(
self
):
am
=
ActivityMonitor
(
history_length
=
0.1
)
c
=
FakeConnection
()
c
.
_transferred
(
1
,
2
)
am
.
closedConnection
(
c
)
time
.
sleep
(
0.2
)
c
.
_transferred
(
3
,
7
)
am
.
closedConnection
(
c
)
self
.
assert_
(
len
(
am
.
log
)
<=
1
)
def
testSetHistoryLength
(
self
):
am
=
ActivityMonitor
(
history_length
=
3600
)
c
=
FakeConnection
()
c
.
_transferred
(
1
,
2
)
am
.
closedConnection
(
c
)
time
.
sleep
(
0.2
)
c
.
_transferred
(
3
,
7
)
am
.
closedConnection
(
c
)
self
.
assertEqual
(
len
(
am
.
log
),
2
)
am
.
setHistoryLength
(
0.1
)
self
.
assertEqual
(
am
.
getHistoryLength
(),
0.1
)
self
.
assert_
(
len
(
am
.
log
)
<=
1
)
def
testActivityAnalysis
(
self
):
am
=
ActivityMonitor
(
history_length
=
3600
)
c
=
FakeConnection
()
c
.
_transferred
(
1
,
2
)
am
.
closedConnection
(
c
)
c
.
_transferred
(
3
,
7
)
am
.
closedConnection
(
c
)
res
=
am
.
getActivityAnalysis
(
start
=
0
,
end
=
0
,
divisions
=
10
)
lastend
=
0
for
n
in
range
(
9
):
div
=
res
[
n
]
self
.
assertEqual
(
div
[
'stores'
],
0
)
self
.
assertEqual
(
div
[
'loads'
],
0
)
self
.
assert_
(
div
[
'start'
]
>
0
)
self
.
assert_
(
div
[
'start'
]
>=
lastend
)
self
.
assert_
(
div
[
'start'
]
<
div
[
'end'
])
lastend
=
div
[
'end'
]
div
=
res
[
9
]
self
.
assertEqual
(
div
[
'stores'
],
9
)
self
.
assertEqual
(
div
[
'loads'
],
4
)
self
.
assert_
(
div
[
'start'
]
>
0
)
self
.
assert_
(
div
[
'start'
]
>=
lastend
)
self
.
assert_
(
div
[
'start'
]
<
div
[
'end'
])
def
test_suite
():
return
unittest
.
makeSuite
(
Tests
)
if
__name__
==
'__main__'
:
unittest
.
main
(
defaultTest
=
'test_suite'
)
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