Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neo
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
2
Merge Requests
2
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Jobs
Commits
Open sidebar
Kirill Smelkov
neo
Commits
fd80cc30
Commit
fd80cc30
authored
Apr 10, 2018
by
Julien Muchembled
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add support for custom compression levels
parent
6f855eef
Changes
9
Hide whitespace changes
Inline
Side-by-side
Showing
9 changed files
with
147 additions
and
35 deletions
+147
-35
neo/client/app.py
neo/client/app.py
+4
-15
neo/client/component.xml
neo/client/component.xml
+7
-3
neo/client/config.py
neo/client/config.py
+8
-0
neo/client/handlers/storage.py
neo/client/handlers/storage.py
+2
-3
neo/lib/compress.py
neo/lib/compress.py
+54
-0
neo/scripts/runner.py
neo/scripts/runner.py
+1
-0
neo/storage/database/importer.py
neo/storage/database/importer.py
+8
-13
neo/tests/threaded/test.py
neo/tests/threaded/test.py
+1
-1
neo/tests/threaded/testConfig.py
neo/tests/threaded/testConfig.py
+62
-0
No files found.
neo/client/app.py
View file @
fd80cc30
...
@@ -15,7 +15,6 @@
...
@@ -15,7 +15,6 @@
# along with this program. If not, see <http://www.gnu.org/licenses/>.
# along with this program. If not, see <http://www.gnu.org/licenses/>.
from
cPickle
import
dumps
,
loads
from
cPickle
import
dumps
,
loads
from
zlib
import
compress
,
decompress
import
heapq
import
heapq
import
time
import
time
...
@@ -26,6 +25,7 @@ if OLD_ZODB:
...
@@ -26,6 +25,7 @@ if OLD_ZODB:
from
persistent.TimeStamp
import
TimeStamp
from
persistent.TimeStamp
import
TimeStamp
from
neo.lib
import
logging
from
neo.lib
import
logging
from
neo.lib.compress
import
decompress_list
,
getCompress
from
neo.lib.protocol
import
NodeTypes
,
Packets
,
\
from
neo.lib.protocol
import
NodeTypes
,
Packets
,
\
INVALID_PARTITION
,
MAX_TID
,
ZERO_HASH
,
ZERO_TID
INVALID_PARTITION
,
MAX_TID
,
ZERO_HASH
,
ZERO_TID
from
neo.lib.util
import
makeChecksum
,
dump
from
neo.lib.util
import
makeChecksum
,
dump
...
@@ -50,7 +50,6 @@ if SignalHandler:
...
@@ -50,7 +50,6 @@ if SignalHandler:
import
signal
import
signal
SignalHandler
.
registerHandler
(
signal
.
SIGUSR2
,
logging
.
reopen
)
SignalHandler
.
registerHandler
(
signal
.
SIGUSR2
,
logging
.
reopen
)
class
Application
(
ThreadedApplication
):
class
Application
(
ThreadedApplication
):
"""The client node application."""
"""The client node application."""
...
@@ -99,7 +98,7 @@ class Application(ThreadedApplication):
...
@@ -99,7 +98,7 @@ class Application(ThreadedApplication):
# _connecting_to_master_node is used to prevent simultaneous master
# _connecting_to_master_node is used to prevent simultaneous master
# node connection attempts
# node connection attempts
self
.
_connecting_to_master_node
=
Lock
()
self
.
_connecting_to_master_node
=
Lock
()
self
.
compress
=
compress
self
.
compress
=
getCompress
(
compress
)
def
__getattr__
(
self
,
attr
):
def
__getattr__
(
self
,
attr
):
if
attr
in
(
'last_tid'
,
'pt'
):
if
attr
in
(
'last_tid'
,
'pt'
):
...
@@ -387,7 +386,7 @@ class Application(ThreadedApplication):
...
@@ -387,7 +386,7 @@ class Application(ThreadedApplication):
logging
.
error
(
'wrong checksum from %s for oid %s'
,
logging
.
error
(
'wrong checksum from %s for oid %s'
,
conn
,
dump
(
oid
))
conn
,
dump
(
oid
))
raise
NEOStorageReadRetry
(
False
)
raise
NEOStorageReadRetry
(
False
)
return
(
decompress
(
data
)
if
compression
else
data
,
return
(
decompress
_list
[
compression
](
data
)
,
tid
,
next_tid
,
data_tid
)
tid
,
next_tid
,
data_tid
)
raise
NEOStorageCreationUndoneError
(
dump
(
oid
))
raise
NEOStorageCreationUndoneError
(
dump
(
oid
))
return
self
.
_askStorageForRead
(
oid
,
return
self
.
_askStorageForRead
(
oid
,
...
@@ -434,17 +433,7 @@ class Application(ThreadedApplication):
...
@@ -434,17 +433,7 @@ class Application(ThreadedApplication):
checksum
=
ZERO_HASH
checksum
=
ZERO_HASH
else
:
else
:
assert
data_serial
is
None
assert
data_serial
is
None
size
=
len
(
data
)
size
,
compression
,
compressed_data
=
self
.
compress
(
data
)
if
self
.
compress
:
compressed_data
=
compress
(
data
)
if
size
<=
len
(
compressed_data
):
compressed_data
=
data
compression
=
0
else
:
compression
=
1
else
:
compression
=
0
compressed_data
=
data
checksum
=
makeChecksum
(
compressed_data
)
checksum
=
makeChecksum
(
compressed_data
)
txn_context
.
data_size
+=
size
txn_context
.
data_size
+=
size
# Store object in tmp cache
# Store object in tmp cache
...
...
neo/client/component.xml
View file @
fd80cc30
...
@@ -14,10 +14,14 @@
...
@@ -14,10 +14,14 @@
Give the name of the cluster
Give the name of the cluster
</description>
</description>
</key>
</key>
<key
name=
"compress"
datatype=
"
boolean
"
>
<key
name=
"compress"
datatype=
"
.compress
"
>
<description>
<description>
If true, data is automatically compressed (unless compressed size is
The value is either of 'boolean' type or an explicit algorithm that
not smaller). This is the default behaviour.
matches the regex 'zlib(=\d+)?', where the optional number is
the compression level.
Any record that is not smaller once compressed is stored uncompressed.
True is the default and its meaning may change over time:
currently, it is the same as 'zlib'.
</description>
</description>
</key>
</key>
<key
name=
"read-only"
datatype=
"boolean"
>
<key
name=
"read-only"
datatype=
"boolean"
>
...
...
neo/client/config.py
View file @
fd80cc30
...
@@ -23,3 +23,11 @@ class NeoStorage(BaseConfig):
...
@@ -23,3 +23,11 @@ class NeoStorage(BaseConfig):
config
=
self
.
config
config
=
self
.
config
return
Storage
(
**
{
k
:
getattr
(
config
,
k
)
return
Storage
(
**
{
k
:
getattr
(
config
,
k
)
for
k
in
config
.
getSectionAttributes
()})
for
k
in
config
.
getSectionAttributes
()})
def
compress
(
value
):
from
ZConfig.datatypes
import
asBoolean
try
:
return
asBoolean
(
value
)
except
ValueError
:
from
neo.lib.compress
import
parseOption
return
parseOption
(
value
)
neo/client/handlers/storage.py
View file @
fd80cc30
...
@@ -14,10 +14,10 @@
...
@@ -14,10 +14,10 @@
# You should have received a copy of the GNU General Public License
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
# along with this program. If not, see <http://www.gnu.org/licenses/>.
from
zlib
import
decompress
from
ZODB.TimeStamp
import
TimeStamp
from
ZODB.TimeStamp
import
TimeStamp
from
neo.lib
import
logging
from
neo.lib
import
logging
from
neo.lib.compress
import
decompress_list
from
neo.lib.protocol
import
Packets
,
uuid_str
from
neo.lib.protocol
import
Packets
,
uuid_str
from
neo.lib.util
import
dump
,
makeChecksum
from
neo.lib.util
import
dump
,
makeChecksum
from
neo.lib.exception
import
NodeNotReady
from
neo.lib.exception
import
NodeNotReady
...
@@ -129,8 +129,7 @@ class StorageAnswersHandler(AnswerBaseHandler):
...
@@ -129,8 +129,7 @@ class StorageAnswersHandler(AnswerBaseHandler):
'wrong checksum while getting back data for'
'wrong checksum while getting back data for'
' object %s during rebase of transaction %s'
' object %s during rebase of transaction %s'
%
(
dump
(
oid
),
dump
(
txn_context
.
ttid
)))
%
(
dump
(
oid
),
dump
(
txn_context
.
ttid
)))
if
compression
:
data
=
decompress_list
[
compression
](
data
)
data
=
decompress
(
data
)
size
=
len
(
data
)
size
=
len
(
data
)
txn_context
.
data_size
+=
size
txn_context
.
data_size
+=
size
if
cached
:
if
cached
:
...
...
neo/lib/compress.py
0 → 100644
View file @
fd80cc30
#
# Copyright (C) 2018 Nexedi SA
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
from
functools
import
partial
import
zlib
decompress_list
=
(
lambda
data
:
data
,
zlib
.
decompress
,
)
def
parseOption
(
value
):
x
=
value
.
split
(
'='
,
1
)
try
:
alg
=
(
'zlib'
,).
index
(
x
[
0
])
if
len
(
x
)
==
1
:
return
alg
,
None
level
=
int
(
x
[
1
])
except
Exception
:
raise
ValueError
(
"not a valid 'compress' option: %r"
%
value
)
if
0
<
level
<=
zlib
.
Z_BEST_COMPRESSION
:
return
alg
,
level
raise
ValueError
(
"invalid compression level: %r"
%
level
)
def
getCompress
(
value
):
if
value
:
alg
,
level
=
(
0
,
None
)
if
value
is
True
else
value
_compress
=
zlib
.
compress
if
level
:
_compress
=
partial
(
_compress
,
level
=
level
)
alg
+=
1
assert
0
<
alg
<
len
(
decompress_list
),
'invalid compression algorithm'
def
compress
(
data
):
size
=
len
(
data
)
compressed
=
_compress
(
data
)
if
len
(
compressed
)
<
size
:
return
size
,
alg
,
compressed
return
size
,
0
,
data
compress
.
_compress
=
_compress
# for testBasicStore
return
compress
return
lambda
data
:
(
len
(
data
),
0
,
data
)
neo/scripts/runner.py
View file @
fd80cc30
...
@@ -65,6 +65,7 @@ UNIT_TEST_MODULES = [
...
@@ -65,6 +65,7 @@ UNIT_TEST_MODULES = [
'
neo
.
tests
.
client
.
testZODBURI
',
'
neo
.
tests
.
client
.
testZODBURI
',
# light functional tests
# light functional tests
'
neo
.
tests
.
threaded
.
test
',
'
neo
.
tests
.
threaded
.
test
',
'
neo
.
tests
.
threaded
.
testConfig
',
'
neo
.
tests
.
threaded
.
testImporter
',
'
neo
.
tests
.
threaded
.
testImporter
',
'
neo
.
tests
.
threaded
.
testReplication
',
'
neo
.
tests
.
threaded
.
testReplication
',
'
neo
.
tests
.
threaded
.
testSSL
',
'
neo
.
tests
.
threaded
.
testSSL
',
...
...
neo/storage/database/importer.py
View file @
fd80cc30
...
@@ -25,7 +25,7 @@ from ZODB.POSException import POSKeyError
...
@@ -25,7 +25,7 @@ from ZODB.POSException import POSKeyError
from
.
import
buildDatabaseManager
from
.
import
buildDatabaseManager
from
.manager
import
DatabaseManager
from
.manager
import
DatabaseManager
from
neo.lib
import
logging
,
patch
,
util
from
neo.lib
import
compress
,
logging
,
patch
,
util
from
neo.lib.exception
import
DatabaseFailure
from
neo.lib.exception
import
DatabaseFailure
from
neo.lib.interfaces
import
implements
from
neo.lib.interfaces
import
implements
from
neo.lib.protocol
import
BackendNotImplemented
,
MAX_TID
from
neo.lib.protocol
import
BackendNotImplemented
,
MAX_TID
...
@@ -297,7 +297,11 @@ class ImporterDatabaseManager(DatabaseManager):
...
@@ -297,7 +297,11 @@ class ImporterDatabaseManager(DatabaseManager):
main
=
{
'adapter'
:
'MySQL'
,
'wait'
:
0
}
main
=
{
'adapter'
:
'MySQL'
,
'wait'
:
0
}
main
.
update
(
config
.
items
(
sections
.
pop
(
0
)))
main
.
update
(
config
.
items
(
sections
.
pop
(
0
)))
self
.
zodb
=
((
x
,
dict
(
config
.
items
(
x
)))
for
x
in
sections
)
self
.
zodb
=
((
x
,
dict
(
config
.
items
(
x
)))
for
x
in
sections
)
self
.
compress
=
main
.
get
(
'compress'
,
1
)
x
=
main
.
get
(
'compress'
)
try
:
self
.
compress
=
bool
(
x
and
(
'false'
,
'true'
).
index
(
x
))
except
ValueError
:
self
.
compress
=
compress
.
parseOption
(
x
)
self
.
db
=
buildDatabaseManager
(
main
[
'adapter'
],
self
.
db
=
buildDatabaseManager
(
main
[
'adapter'
],
(
main
[
'database'
],
main
.
get
(
'engine'
),
main
[
'wait'
]))
(
main
[
'database'
],
main
.
get
(
'engine'
),
main
[
'wait'
]))
for
x
in
"""getConfiguration _setConfiguration setNumPartitions
for
x
in
"""getConfiguration _setConfiguration setNumPartitions
...
@@ -375,11 +379,7 @@ class ImporterDatabaseManager(DatabaseManager):
...
@@ -375,11 +379,7 @@ class ImporterDatabaseManager(DatabaseManager):
if
self
.
_last_commit
+
1
<
time
.
time
():
if
self
.
_last_commit
+
1
<
time
.
time
():
self
.
commit
()
self
.
commit
()
self
.
zodb_tid
=
u64
(
tid
)
self
.
zodb_tid
=
u64
(
tid
)
if
self
.
compress
:
_compress
=
compress
.
getCompress
(
self
.
compress
)
from
zlib
import
compress
else
:
compress
=
None
compression
=
0
object_list
=
[]
object_list
=
[]
data_id_list
=
[]
data_id_list
=
[]
while
zodb_list
:
while
zodb_list
:
...
@@ -399,12 +399,7 @@ class ImporterDatabaseManager(DatabaseManager):
...
@@ -399,12 +399,7 @@ class ImporterDatabaseManager(DatabaseManager):
if
data_tid
or
r
.
data
is
None
:
if
data_tid
or
r
.
data
is
None
:
data_id
=
None
data_id
=
None
else
:
else
:
data
=
zodb
.
repickle
(
r
.
data
)
_
,
compression
,
data
=
_compress
(
zodb
.
repickle
(
r
.
data
))
if
compress
:
compressed_data
=
compress
(
data
)
compression
=
len
(
compressed_data
)
<
len
(
data
)
if
compression
:
data
=
compressed_data
data_id
=
self
.
holdData
(
util
.
makeChecksum
(
data
),
oid
,
data
,
data_id
=
self
.
holdData
(
util
.
makeChecksum
(
data
),
oid
,
data
,
compression
)
compression
)
data_id_list
.
append
(
data_id
)
data_id_list
.
append
(
data_id
)
...
...
neo/tests/threaded/test.py
View file @
fd80cc30
...
@@ -23,7 +23,6 @@ import unittest
...
@@ -23,7 +23,6 @@ import unittest
from
collections
import
defaultdict
from
collections
import
defaultdict
from
contextlib
import
contextmanager
from
contextlib
import
contextmanager
from
thread
import
get_ident
from
thread
import
get_ident
from
zlib
import
compress
from
persistent
import
Persistent
,
GHOST
from
persistent
import
Persistent
,
GHOST
from
transaction.interfaces
import
TransientError
from
transaction.interfaces
import
TransientError
from
ZODB
import
DB
,
POSException
from
ZODB
import
DB
,
POSException
...
@@ -66,6 +65,7 @@ class Test(NEOThreadedTest):
...
@@ -66,6 +65,7 @@ class Test(NEOThreadedTest):
storage
=
cluster
.
getZODBStorage
()
storage
=
cluster
.
getZODBStorage
()
storage
.
sync
()
storage
.
sync
()
storage
.
app
.
max_reconnection_to_master
=
0
storage
.
app
.
max_reconnection_to_master
=
0
compress
=
storage
.
app
.
compress
.
_compress
data_info
=
{}
data_info
=
{}
compressible
=
'x'
*
20
compressible
=
'x'
*
20
compressed
=
compress
(
compressible
)
compressed
=
compress
(
compressible
)
...
...
neo/tests/threaded/testConfig.py
0 → 100644
View file @
fd80cc30
#
# Copyright (C) 2018 Nexedi SA
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
import
unittest
from
contextlib
import
contextmanager
from
ZConfig
import
ConfigurationSyntaxError
from
ZODB.config
import
databaseFromString
from
..
import
Patch
from
.
import
ClientApplication
,
NEOThreadedTest
,
with_cluster
from
neo.client
import
Storage
def
databaseFromDict
(
**
kw
):
return
databaseFromString
(
"%%import neo.client
\
n
"
"<zodb>
\
n
<NEOStorage>
\
n
%s </NEOStorage>
\
n
</zodb>
\
n
"
%
''
.
join
(
' %s %s
\
n
'
%
x
for
x
in
kw
.
iteritems
()))
class
ConfigTests
(
NEOThreadedTest
):
dummy_required
=
{
'name'
:
'cluster'
,
'master_nodes'
:
'127.0.0.1:10000'
}
@
contextmanager
def
_db
(
self
,
cluster
,
**
kw
):
kw
[
'name'
]
=
cluster
.
name
kw
[
'master_nodes'
]
=
cluster
.
master_nodes
def
newClient
(
_
,
*
args
,
**
kw
):
client
=
ClientApplication
(
*
args
,
**
kw
)
t
.
append
(
client
.
poll_thread
)
return
client
t
=
[]
with
Patch
(
Storage
,
Application
=
newClient
):
db
=
databaseFromDict
(
**
kw
)
try
:
yield
db
finally
:
db
.
close
()
cluster
.
join
(
t
)
@
with_cluster
()
def
testCompress
(
self
,
cluster
):
kw
=
self
.
dummy_required
.
copy
()
valid
=
[
'false'
,
'true'
,
'zlib'
,
'zlib=9'
]
for
kw
[
'compress'
]
in
'9'
,
'best'
,
'zlib=0'
,
'zlib=100'
:
self
.
assertRaises
(
ConfigurationSyntaxError
,
databaseFromDict
,
**
kw
)
for
compress
in
valid
:
with
self
.
_db
(
cluster
,
compress
=
compress
):
pass
if
__name__
==
"__main__"
:
unittest
.
main
()
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