Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neoppod
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
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Xavier Thompson
neoppod
Commits
20791999
Commit
20791999
authored
Feb 22, 2024
by
Julien Muchembled
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
client: new ignore-wrong-checksum option
parent
3531ee9e
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
66 additions
and
11 deletions
+66
-11
neo/client/app.py
neo/client/app.py
+23
-5
neo/client/component.xml
neo/client/component.xml
+7
-0
neo/client/exception.py
neo/client/exception.py
+3
-0
neo/storage/handlers/client.py
neo/storage/handlers/client.py
+2
-2
neo/tests/threaded/__init__.py
neo/tests/threaded/__init__.py
+5
-4
neo/tests/threaded/test.py
neo/tests/threaded/test.py
+26
-0
No files found.
neo/client/app.py
View file @
20791999
...
...
@@ -36,7 +36,8 @@ from neo.lib.connection import MTClientConnection, ConnectionClosed
from
neo.lib.exception
import
NodeNotReady
from
.
import
TransactionMetaData
from
.exception
import
(
NEOStorageError
,
NEOStorageCreationUndoneError
,
NEOStorageReadRetry
,
NEOStorageNotFoundError
,
NEOPrimaryMasterLost
)
NEOStorageReadRetry
,
NEOStorageNotFoundError
,
NEOStorageWrongChecksum
,
NEOPrimaryMasterLost
)
from
.handlers
import
storage
,
master
from
neo.lib.threaded_app
import
ThreadedApplication
from
.cache
import
ClientCache
...
...
@@ -70,7 +71,7 @@ class Application(ThreadedApplication):
wait_for_pack
=
False
def
__init__
(
self
,
master_nodes
,
name
,
compress
=
True
,
cache_size
=
None
,
**
kw
):
ignore_wrong_checksum
=
False
,
**
kw
):
super
(
Application
,
self
).
__init__
(
parseMasterList
(
master_nodes
),
name
,
**
kw
)
# Internal Attributes common to all thread
...
...
@@ -106,6 +107,7 @@ class Application(ThreadedApplication):
self
.
_connecting_to_storage_node
=
Lock
()
self
.
_node_failure_dict
=
{}
self
.
compress
=
getCompress
(
compress
)
self
.
ignore_wrong_checksum
=
ignore_wrong_checksum
def
__getattr__
(
self
,
attr
):
if
attr
in
(
'last_tid'
,
'pt'
):
...
...
@@ -459,6 +461,7 @@ class Application(ThreadedApplication):
return
data
,
tid
,
next_tid
def
_loadFromStorage
(
self
,
oid
,
at_tid
,
before_tid
):
wrong_checksum
=
[]
# Py3
def
askStorage
(
conn
,
packet
):
tid
,
next_tid
,
compression
,
checksum
,
data
,
data_tid
\
=
self
.
_askStorage
(
conn
,
packet
)
...
...
@@ -466,13 +469,28 @@ class Application(ThreadedApplication):
if
checksum
!=
makeChecksum
(
data
):
logging
.
error
(
'wrong checksum from %s for %s@%s'
,
conn
,
dump
(
oid
),
dump
(
tid
))
wrong_checksum
.
append
((
tid
,
next_tid
,
compression
,
checksum
,
data
,
data_tid
))
raise
NEOStorageReadRetry
(
False
)
return
(
decompress_list
[
compression
](
data
),
tid
,
next_tid
,
data_tid
)
raise
NEOStorageCreationUndoneError
(
dump
(
oid
))
return
self
.
_askStorageForRead
(
oid
,
Packets
.
AskObject
(
oid
,
at_tid
,
before_tid
),
askStorage
)
try
:
return
self
.
_askStorageForRead
(
oid
,
Packets
.
AskObject
(
oid
,
at_tid
,
before_tid
),
askStorage
)
except
NEOStorageError
:
if
not
wrong_checksum
:
raise
tid
,
next_tid
,
compression
,
checksum
,
data
,
data_tid
=
\
wrong_checksum
[
0
]
if
self
.
ignore_wrong_checksum
:
try
:
data
=
decompress_list
[
compression
](
data
)
except
Exception
:
data
=
''
return
data
,
tid
,
next_tid
,
data_tid
raise
NEOStorageWrongChecksum
(
oid
,
tid
)
def
tpc_begin
(
self
,
storage
,
transaction
,
tid
=
None
,
status
=
' '
):
"""Begin a new transaction."""
...
...
neo/client/component.xml
View file @
20791999
...
...
@@ -51,6 +51,13 @@
be added/removed without requiring a config change each time.
</description>
</key>
<key
name=
"ignore-wrong-checksum"
datatype=
"boolean"
>
<description>
If true whereas checksum does not match, return whatever is stored
instead of raising. When compression is enabled, decompression is
likely to fail and an empty record is returned.
</description>
</key>
<key
name=
"ca"
datatype=
"existing-file"
>
<description>
Certificate authority in PEM format.
...
...
neo/client/exception.py
View file @
20791999
...
...
@@ -25,6 +25,9 @@ class NEOStorageReadRetry(NEOStorageError):
class
NEOStorageNotFoundError
(
NEOStorageError
):
pass
class
NEOStorageWrongChecksum
(
NEOStorageError
):
pass
class
NEOStorageDoesNotExistError
(
NEOStorageNotFoundError
):
"""
This error is a refinement of NEOStorageNotFoundError: this means
...
...
neo/storage/handlers/client.py
View file @
20791999
...
...
@@ -125,8 +125,8 @@ class ClientOperationHandler(BaseHandler):
# register the transaction
self
.
app
.
tm
.
register
(
conn
,
ttid
)
if
data
or
checksum
!=
ZERO_HASH
:
# TODO: return an appropriate error packet
assert
makeChecksum
(
data
)
==
checksum
if
makeChecksum
(
data
)
!=
checksum
:
raise
ProtocolError
(
'invalid checksum'
)
else
:
checksum
=
data
=
None
try
:
...
...
neo/tests/threaded/__init__.py
View file @
20791999
...
...
@@ -949,13 +949,14 @@ class NEOCluster(object):
for
node
in
getattr
(
self
,
node_type
+
'_list'
):
node
.
resetNode
(
**
reset_kw
)
def
_newClient
(
self
):
def
_newClient
(
self
,
**
kw
):
kw
.
setdefault
(
'compress'
,
self
.
compress
)
return
ClientApplication
(
name
=
self
.
name
,
master_nodes
=
self
.
master_nodes
,
compress
=
self
.
compress
,
ssl
=
self
.
SSL
)
ssl
=
self
.
SSL
,
**
kw
)
@
contextmanager
def
newClient
(
self
,
with_db
=
False
):
x
=
self
.
_newClient
()
def
newClient
(
self
,
with_db
=
False
,
**
kw
):
x
=
self
.
_newClient
(
**
kw
)
try
:
t
=
x
.
poll_thread
closed
=
[]
...
...
neo/tests/threaded/test.py
View file @
20791999
...
...
@@ -40,6 +40,7 @@ from .. import Patch, TransactionalResource, getTransactionMetaData
from
.
import
ClientApplication
,
ConnectionFilter
,
LockLock
,
NEOCluster
,
\
NEOThreadedTest
,
RandomConflictDict
,
Serialized
,
ThreadId
,
with_cluster
from
neo.lib.util
import
add64
,
makeChecksum
,
p64
,
u64
from
neo.client
import
exception
from
neo.client.exception
import
NEOPrimaryMasterLost
,
NEOStorageError
from
neo.client.handlers.storage
import
_DeadlockPacket
from
neo.client.transactions
import
Transaction
...
...
@@ -2881,6 +2882,31 @@ class Test(NEOThreadedTest):
storage
.
tpc_vote
(
txn
)
self
.
assertEqual
(
add64
(
tid
,
1
),
storage
.
tpc_finish
(
txn
))
@
with_cluster
()
def
testCorruptedData
(
self
,
cluster
):
def
holdData
(
orig
,
*
args
):
args
=
list
(
args
)
args
[
2
]
=
'!'
+
args
[
2
]
return
orig
(
*
args
)
data
=
'foo'
*
10
tid
=
None
for
compress
in
False
,
True
:
with
cluster
.
newClient
(
ignore_wrong_checksum
=
True
,
compress
=
compress
)
as
client
,
\
Patch
(
cluster
.
storage
.
dm
,
holdData
=
holdData
):
storage
=
cluster
.
getZODBStorage
(
client
=
client
)
txn
=
transaction
.
Transaction
()
storage
.
tpc_begin
(
txn
)
storage
.
store
(
ZERO_OID
,
tid
,
data
,
''
,
txn
)
storage
.
tpc_vote
(
txn
)
tid
=
storage
.
tpc_finish
(
txn
)
storage
.
_cache
.
clear
()
self
.
assertEqual
((
''
if
compress
else
'!'
+
data
,
tid
),
storage
.
load
(
ZERO_OID
))
with
self
.
assertRaises
(
exception
.
NEOStorageWrongChecksum
)
as
cm
:
cluster
.
client
.
load
(
ZERO_OID
)
self
.
assertEqual
(
tid
,
cm
.
exception
.
args
[
1
])
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