Arcane  v3.16.0.0
Documentation développeur
Chargement...
Recherche...
Aucune correspondance
BasicReaderWriterDatabase.cc
1// -*- tab-width: 2; indent-tabs-mode: nil; coding: utf-8-with-signature -*-
2//-----------------------------------------------------------------------------
3// Copyright 2000-2024 CEA (www.cea.fr) IFPEN (www.ifpenergiesnouvelles.com)
4// See the top-level COPYRIGHT file for details.
5// SPDX-License-Identifier: Apache-2.0
6//-----------------------------------------------------------------------------
7/*---------------------------------------------------------------------------*/
8/* BasicReaderWriterDatabase.cc (C) 2000-2024 */
9/* */
10/* Base de donnée pour le service 'BasicReaderWriter'. */
11/*---------------------------------------------------------------------------*/
12/*---------------------------------------------------------------------------*/
13
14#include "arcane/std/internal/BasicReaderWriterDatabase.h"
15
16#include "arcane/utils/Array.h"
17#include "arcane/utils/PlatformUtils.h"
18#include "arcane/utils/Ref.h"
19#include "arcane/utils/FatalErrorException.h"
20#include "arcane/utils/JSONReader.h"
21#include "arcane/utils/JSONWriter.h"
22#include "arcane/utils/CheckedConvert.h"
23#include "arcane/utils/IDataCompressor.h"
24#include "arcane/utils/SmallArray.h"
25#include "arcane/utils/IHashAlgorithm.h"
26#include "arcane/utils/ITraceMng.h"
27
28#include "arcane/ArcaneException.h"
29
30#include "arcane/std/internal/TextReader2.h"
31#include "arcane/std/internal/TextWriter2.h"
32#include "arcane/std/internal/IHashDatabase.h"
33
34#include <fstream>
35#include <map>
36
37/*---------------------------------------------------------------------------*/
38/*---------------------------------------------------------------------------*/
39
40namespace Arcane::impl
41{
42
43/*---------------------------------------------------------------------------*/
44/*---------------------------------------------------------------------------*/
45
47class Hasher
48{
49 public:
50
51 void computeHash(Span<const std::byte> bytes, Array<Byte>& hash_result)
52 {
53 ARCANE_CHECK_POINTER(m_hash_algorithm);
54 {
56 m_nb_processed_bytes += bytes.size();
57 m_hash_algorithm->computeHash64(bytes, hash_result);
59 m_hash_time += (t2 - t1);
60 }
61 }
62
63 void setHashAlgorithm(IHashAlgorithm* algo)
64 {
65 m_hash_algorithm = algo;
66 }
67
68 void printStats(ITraceMng* tm)
69 {
70 if (m_nb_processed_bytes == 0)
71 return;
72 Real nb_byte_per_second = static_cast<Real>(m_nb_processed_bytes) / (m_hash_time + 1.0e-9);
73 // Pour avoir en Mega-byte par seconde
74 nb_byte_per_second /= 1.0e6;
75 tm->info() << "Hasher:nb_processed=" << m_nb_processed_bytes
76 << " hash_time=" << m_hash_time
77 << " rate=" << static_cast<Int64>(nb_byte_per_second) << " MB/s"
78 << "\n";
79 }
80
81 private:
82
83 IHashAlgorithm* m_hash_algorithm = nullptr;
84 Int64 m_nb_processed_bytes = 0;
85 Real m_hash_time = 0.0;
86};
87
88/*---------------------------------------------------------------------------*/
89/*---------------------------------------------------------------------------*/
90
92class BasicReaderWriterDatabaseEpilogFormat
93{
94 public:
95
96 // La taille de cette structure ne doit pas être modifiée sous peine
97 // de rendre le format incompatible. Pour supporter des évolutions, on fixe
98 // une taille de 128 octets, soit 16 'Int64'
99 static constexpr Int64 STRUCT_SIZE = 128;
100
101 public:
102
103 BasicReaderWriterDatabaseEpilogFormat()
104 {
105 checkStructureSize();
106 for (int i = 0; i < 10; ++i)
107 m_remaining_padding[i] = i;
108 }
109
110 public:
111
112 void setJSONDataInfoOffsetAndSize(Int64 file_offset, Int64 data_size)
113 {
114 m_json_data_info_file_offset = file_offset;
115 m_json_data_info_size = data_size;
116 }
117 Int32 version() const { return m_version; }
118 Int64 jsonDataInfoFileOffset() const { return m_json_data_info_file_offset; }
119 Int64 jsonDataInfoSize() const { return m_json_data_info_size; }
120 Span<std::byte> bytes()
121 {
122 return { reinterpret_cast<std::byte*>(this), STRUCT_SIZE };
123 }
124
125 private:
126
127 // Version de l'epilogue. A ne pas confondre avec la version du fichier
128 // qui est dans l'en-tête.
129 Int32 m_version = 1;
130 Int32 m_padding0 = 0;
131 Int64 m_padding1 = 0;
132 Int64 m_padding2 = 0;
133 Int64 m_padding3 = 0;
134 Int64 m_json_data_info_file_offset = -1;
135 Int64 m_json_data_info_size = 0;
136 Int64 m_remaining_padding[10];
137
138 public:
139
140 static void checkStructureSize()
141 {
142 Int64 s = sizeof(BasicReaderWriterDatabaseEpilogFormat);
143 if (s != STRUCT_SIZE)
144 ARCANE_FATAL("Invalid size for epilog format size={0} expected={1}", s, STRUCT_SIZE);
145 }
146};
147
148/*---------------------------------------------------------------------------*/
149/*---------------------------------------------------------------------------*/
150
152class BasicReaderWriterDatabaseHeaderFormat
153{
154 public:
155
156 // La taille de cette structure ne doit pas être modifiée sous peine
157 // de rendre le format incompatible. Pour supporter des évolutions, on fixe
158 // une taille de 128 octets, soit 16 'Int64'
159 static constexpr Int64 STRUCT_SIZE = 128;
160
161 public:
162
163 BasicReaderWriterDatabaseHeaderFormat()
164 {
165 checkStructureSize();
166 for (int i = 0; i < 12; ++i)
167 m_remaining_padding[i] = i;
168 }
169
170 public:
171
172 Span<std::byte> bytes()
173 {
174 return { reinterpret_cast<std::byte*>(this), STRUCT_SIZE };
175 }
176 void setVersion(Int32 version) { m_version = version; }
177 Int32 version() const { return m_version; }
178 void checkHeader()
179 {
180 // Vérifie que le header est correct.
181 if (m_header_begin[0] != 'A' || m_header_begin[1] != 'C' || m_header_begin[2] != 'R' || m_header_begin[3] != (Byte)39)
182 ARCANE_FATAL("Bad header");
183 // TODO: tester indianess
184 }
185
186 private:
187
188 // 4 premiers octets pour indiquer qu'il s'agit d'un fichier de protections
189 // arcane (ACR pour Arcane Checkpoint Restart)
190 std::array<Byte, 4> m_header_begin = { 'A', 'C', 'R', 39 };
191 // 4 octets suivant pour indiquer l'indianness (pas encore utilisé)
192 Int32 m_endian_int = 0x01020304;
193 // Version du fichier (à modifier par l'appelant via setVersion())
194 Int32 m_version = 0;
195 Int32 m_padding0 = 0;
196 Int64 m_padding1 = 0;
197 Int64 m_padding2 = 0;
198 Int64 m_remaining_padding[12];
199
200 public:
201
202 static void checkStructureSize()
203 {
204 Int64 s = sizeof(BasicReaderWriterDatabaseHeaderFormat);
205 if (s != STRUCT_SIZE)
206 ARCANE_FATAL("Invalid size for header format size={0} expected={1}", s, STRUCT_SIZE);
207 }
208};
209
210/*---------------------------------------------------------------------------*/
211/*---------------------------------------------------------------------------*/
212
213class BasicReaderWriterDatabaseCommon
214: public TraceAccessor
215{
216 public:
217
219 {
220 static constexpr int MAX_SIZE = 8;
221 void fill(SmallSpan<const Int64> v)
222 {
223 nb = v.size();
224 if (nb > MAX_SIZE) {
225 m_large_extents = v;
226 return;
227 }
228 for (Int32 i = 0; i < nb; ++i)
229 sizes[i] = v[i];
230 }
231 Int64ConstArrayView view() const
232 {
233 if (nb <= MAX_SIZE)
234 return Int64ConstArrayView(nb, sizes);
235 return m_large_extents.view();
236 }
237 Int32 size() const { return nb; }
238
239 private:
240
241 Int32 nb = 0;
242
243 public:
244
245 Int64 sizes[MAX_SIZE];
246 UniqueArray<Int64> m_large_extents;
247 };
248
249 struct DataInfo
250 {
251 Int64 m_file_offset = 0;
252 ExtentsInfo m_extents;
253 };
254
255 public:
256
257 BasicReaderWriterDatabaseCommon(ITraceMng* tm, Int32 version)
258 : TraceAccessor(tm)
259 {
260 if (version >= 3) {
261 String hash_directory = platform::getEnvironmentVariable("ARCANE_HASHDATABASE_DIRECTORY");
262 if (!hash_directory.null()) {
263 info() << "Using Hash database at location '" << hash_directory << "'";
264 m_hash_database = createFileHashDatabase(tm, hash_directory);
265 }
266 else {
267 String redis_machine = platform::getEnvironmentVariable("ARCANE_HASHDATABASE_REDIS");
268 if (!redis_machine.null()){
269 info() << "Using Redis database at location '" << redis_machine << "'";
270 m_hash_database = createRedisHashDatabase(tm, redis_machine, 6379);
271 }
272 }
273 }
274 }
275
276 public:
277
278 DataInfo& findData(const String& key_name)
279 {
280 auto x = m_data_infos.find(key_name);
281 if (x == m_data_infos.end())
282 ARCANE_FATAL("Can not find key '{0}' in database", key_name);
283 return x->second;
284 }
285
286 public:
287
288 std::map<String, DataInfo> m_data_infos;
289 Ref<IDataCompressor> m_data_compressor;
290 Ref<IHashAlgorithm> m_hash_algorithm;
291 Ref<IHashDatabase> m_hash_database;
292};
293
294/*---------------------------------------------------------------------------*/
295/*---------------------------------------------------------------------------*/
296
297/*---------------------------------------------------------------------------*/
298/*---------------------------------------------------------------------------*/
299
301: public BasicReaderWriterDatabaseCommon
302{
303 public:
304
305 Impl(ITraceMng* tm, const String& filename, Int32 version)
306 : BasicReaderWriterDatabaseCommon(tm, version)
307 , m_writer(filename)
308 , m_version(version)
309 {
310 if (m_version >= 3)
311 _writeHeader();
312 }
313
314 ~Impl()
315 {
316 if (m_version >= 3)
317 arcaneCallFunctionAndTerminateIfThrow([&]() { _writeEpilog(); });
318 m_hasher.printStats(traceMng());
319 }
320
321 public:
322
323 Int64 fileOffset() { return m_writer.fileOffset(); }
324 void setExtents(const String& key_name, SmallSpan<const Int64> extents);
325 void write(const String& key, Span<const std::byte> values);
326
327 private:
328
329 void _addKey(const String& key, SmallSpan<const Int64> extents);
330 void _writeKey(const String& key);
331 void _writeHeader();
332 void _writeEpilog();
333
334 public:
335
336 TextWriter2 m_writer;
337 Int32 m_version;
338 Hasher m_hasher;
339
340 private:
341
342 void _write2(const String& key, Span<const std::byte> values);
343};
344
345/*---------------------------------------------------------------------------*/
346/*---------------------------------------------------------------------------*/
347
348KeyValueTextWriter::
349KeyValueTextWriter(ITraceMng* tm, const String& filename, Int32 version)
350: TraceAccessor(tm)
351, m_p(new Impl(tm, filename, version))
352{
353}
354
355/*---------------------------------------------------------------------------*/
356/*---------------------------------------------------------------------------*/
357
358KeyValueTextWriter::
359~KeyValueTextWriter()
360{
361 delete m_p;
362}
363
364/*---------------------------------------------------------------------------*/
365/*---------------------------------------------------------------------------*/
366
367/*---------------------------------------------------------------------------*/
368/*---------------------------------------------------------------------------*/
377{
379
380 // Actuellement si on passe dans cette partie de code,
381 // la version utilisée est 3 ou plus.
382 header.setVersion(m_version);
383 binaryWrite(m_writer.stream(), header.bytes());
384}
385
386/*---------------------------------------------------------------------------*/
387/*---------------------------------------------------------------------------*/
388
389void KeyValueTextWriter::Impl::
390_writeEpilog()
391{
392 // Ecrit les méta-données au format JSON.
393 JSONWriter jsw;
394 {
395 JSONWriter::Object main_object(jsw);
396 jsw.writeKey("Data");
397 jsw.beginArray();
398 for (auto& x : m_data_infos) {
399 JSONWriter::Object o(jsw);
400 jsw.write("Name", x.first);
401 jsw.write("FileOffset", x.second.m_file_offset);
402 jsw.write("Extents", x.second.m_extents.view());
403 }
404 jsw.endArray();
405 }
406
407 std::ostream& stream = m_writer.stream();
408
409 // Conserve la position dans le fichier des méta-données
410 // ainsi que leur taille.
411 Int64 file_offset = m_writer.fileOffset();
412 StringView buf = jsw.getBuffer();
413 Int64 meta_data_size = buf.size();
414
415 binaryWrite(stream, asBytes(buf.bytes()));
416
417 {
419 epilog.setJSONDataInfoOffsetAndSize(file_offset, meta_data_size);
420 // Doit toujours être la dernière écriture du fichier
421 binaryWrite(stream, epilog.bytes());
422 }
423}
424
425/*---------------------------------------------------------------------------*/
426/*---------------------------------------------------------------------------*/
427
428void KeyValueTextWriter::Impl::
429setExtents(const String& key_name, SmallSpan<const Int64> extents)
430{
431 if (m_version >= 3) {
432 _addKey(key_name, extents);
433 }
434 else {
435 // Versions 1 et 2.
436 // On sauve directement dans le fichier à la position courante
437 // les valeurs des dimentions. Le nombre de valeur est donné par la taille
438 // de \a extents
439 Integer dimension_array_size = extents.size();
440 if (dimension_array_size != 0) {
441 String true_key_name = "Extents:" + key_name;
442 String comment = String::format("Writing Dim1Size for '{0}'", key_name);
443 if (m_version == 1) {
444 // Sauve les dimensions comme un tableau de Int32
445 UniqueArray<Integer> dims(dimension_array_size);
446 for (Integer i = 0; i < dimension_array_size; ++i)
447 dims[i] = CheckedConvert::toInteger(extents[i]);
448 m_writer.write(asBytes(dims));
449 }
450 else
451 // Sauve les dimensions comme un tableau de Int64
452 m_writer.write(asBytes(extents));
453 }
454 }
455}
456
457/*---------------------------------------------------------------------------*/
458/*---------------------------------------------------------------------------*/
459
460void KeyValueTextWriter::Impl::
461write(const String& key, Span<const std::byte> values)
462{
463 _writeKey(key);
464
465 IDataCompressor* d = m_data_compressor.get();
466 Int64 len = values.size();
467 if (d && len > d->minCompressSize()) {
468 UniqueArray<std::byte> compressed_values;
469 m_data_compressor->compress(values, compressed_values);
470 Int64 compressed_size = compressed_values.largeSize();
471 m_writer.write(asBytes(Span<const Int64>(&compressed_size, 1)));
472 _write2(key, compressed_values);
473 }
474 else
475 _write2(key, values);
476}
477
478/*---------------------------------------------------------------------------*/
479/*---------------------------------------------------------------------------*/
480
481void KeyValueTextWriter::Impl::
482_write2(const String& key, Span<const std::byte> values)
483{
484 if (m_hash_database.get()) {
485 IHashAlgorithm* hash_algo = m_hash_algorithm.get();
486 if (!hash_algo)
487 ARCANE_FATAL("Can not use hash database without hash algorithm");
488
489 SmallArray<Byte, 1024> hash_result;
490 m_hasher.computeHash(values, hash_result);
491 String hash_value = Convert::toHexaString(hash_result);
492
493 HashDatabaseWriteResult result;
494 HashDatabaseWriteArgs args(values, hash_value);
495 args.setKey(key);
496
497 m_hash_database->writeValues(args, result);
498 info(5) << "WRITE_KW_HASH key=" << key << " hash=" << hash_value << " len=" << values.size();
499 m_writer.write(asBytes(hash_result));
500 }
501 else
502 m_writer.write(values);
503}
504
505/*---------------------------------------------------------------------------*/
506/*---------------------------------------------------------------------------*/
507
508String KeyValueTextWriter::
509fileName() const
510{
511 return m_p->m_writer.fileName();
512}
513
514/*---------------------------------------------------------------------------*/
515/*---------------------------------------------------------------------------*/
516
517void KeyValueTextWriter::
518setDataCompressor(Ref<IDataCompressor> ds)
519{
520 m_p->m_data_compressor = ds;
521}
522
523/*---------------------------------------------------------------------------*/
524/*---------------------------------------------------------------------------*/
525
526Ref<IDataCompressor> KeyValueTextWriter::
527dataCompressor() const
528{
529 return m_p->m_data_compressor;
530}
531
532/*---------------------------------------------------------------------------*/
533/*---------------------------------------------------------------------------*/
534
535void KeyValueTextWriter::
536setHashAlgorithm(Ref<IHashAlgorithm> v)
537{
538 m_p->m_hash_algorithm = v;
539 m_p->m_hasher.setHashAlgorithm(v.get());
540}
541
542/*---------------------------------------------------------------------------*/
543/*---------------------------------------------------------------------------*/
544
545Ref<IHashAlgorithm> KeyValueTextWriter::
546hashAlgorithm() const
547{
548 return m_p->m_hash_algorithm;
549}
550
551/*---------------------------------------------------------------------------*/
552/*---------------------------------------------------------------------------*/
553
554Int64 KeyValueTextWriter::
555fileOffset()
556{
557 return m_p->m_writer.fileOffset();
558}
559
560/*---------------------------------------------------------------------------*/
561/*---------------------------------------------------------------------------*/
562
563void KeyValueTextWriter::Impl::
564_addKey(const String& key, SmallSpan<const Int64> extents)
565{
567 d.m_extents.fill(extents);
568 m_data_infos.insert(std::make_pair(key, d));
569}
570
571/*---------------------------------------------------------------------------*/
572/*---------------------------------------------------------------------------*/
573
574void KeyValueTextWriter::Impl::
575_writeKey(const String& key)
576{
577 if (m_version >= 3) {
578 auto x = m_data_infos.find(key);
579 if (x == m_data_infos.end())
580 ARCANE_FATAL("Key '{0}' is not in map. You should call setExtents() before", key);
581 x->second.m_file_offset = fileOffset();
582 }
583}
584
585/*---------------------------------------------------------------------------*/
586/*---------------------------------------------------------------------------*/
587
588/*---------------------------------------------------------------------------*/
589/*---------------------------------------------------------------------------*/
590
591void KeyValueTextWriter::
592setExtents(const String& key_name, SmallSpan<const Int64> extents)
593{
594 m_p->setExtents(key_name, extents);
595}
596
597/*---------------------------------------------------------------------------*/
598/*---------------------------------------------------------------------------*/
599
600void KeyValueTextWriter::
601write(const String& key, Span<const std::byte> values)
602{
603 m_p->write(key, values);
604}
605
606/*---------------------------------------------------------------------------*/
607/*---------------------------------------------------------------------------*/
608
609/*---------------------------------------------------------------------------*/
610/*---------------------------------------------------------------------------*/
611
613: public BasicReaderWriterDatabaseCommon
614{
615 public:
616
617 Impl(ITraceMng* tm, const String& filename, Int32 version)
618 : BasicReaderWriterDatabaseCommon(tm, version)
619 , m_reader(filename)
620 , m_version(version)
621 {
622 if (m_version >= 3) {
623 _readHeader();
624 _readJSON();
625 }
626 }
627
628 public:
629
630 void readIntegers(const String& key, Span<Integer> values);
631 void read(const String& key, Span<std::byte> values);
632
633 public:
634
635 void _readHeader();
636 void _readJSON();
637 void _readDirect(Int64 offset, Span<std::byte> bytes);
638 void _setFileOffset(const String& key_name);
639 void _read2(const String& key_name, Span<std::byte> values);
640
641 public:
642
643 TextReader2 m_reader;
644 Int32 m_version;
645};
646
647/*---------------------------------------------------------------------------*/
648/*---------------------------------------------------------------------------*/
649
650KeyValueTextReader::
651KeyValueTextReader(ITraceMng* tm, const String& filename, Int32 version)
652: TraceAccessor(tm)
653, m_p(new Impl(tm, filename, version))
654{
655}
656
657/*---------------------------------------------------------------------------*/
658/*---------------------------------------------------------------------------*/
659
660KeyValueTextReader::
661~KeyValueTextReader()
662{
663 delete m_p;
664}
665
666/*---------------------------------------------------------------------------*/
667/*---------------------------------------------------------------------------*/
668
669void KeyValueTextReader::Impl::
670_readDirect(Int64 offset, Span<std::byte> bytes)
671{
672 m_reader.setFileOffset(offset);
673 std::istream& s = m_reader.stream();
674 binaryRead(s, bytes);
675 if (s.fail())
676 ARCANE_FATAL("Can not read file part offset={0} length={1}", offset, bytes.length());
677}
678
679/*---------------------------------------------------------------------------*/
680/*---------------------------------------------------------------------------*/
689{
691 Span<std::byte> bytes(header.bytes());
692 _readDirect(0, bytes);
693 header.checkHeader();
694 Int32 version = header.version();
695 if (version != m_version)
696 ARCANE_FATAL("Invalid version for ACR file version={0} expected={1}", version, m_version);
697}
698
699/*---------------------------------------------------------------------------*/
700/*---------------------------------------------------------------------------*/
701
702void KeyValueTextReader::Impl::
703_readJSON()
704{
705 // Les informations sur la position dans le fichier et la longueur du
706 // texte JSON sont sauvgegardées dans l'épilogue.
707 Int64 file_length = m_reader.fileLength();
708 // Vérifie la longueur du fichier par précaution
709 Int64 struct_size = BasicReaderWriterDatabaseEpilogFormat::STRUCT_SIZE;
710 if (file_length < struct_size)
711 ARCANE_FATAL("File is too short length={0} minimum={1}", file_length, struct_size);
712
714
715 // Lit l'épilogue et verifie que la version est supportée.
716 {
717 _readDirect(file_length - struct_size, epilog.bytes());
718 const int expected_version = 1;
719 if (epilog.version() != expected_version)
720 ARCANE_FATAL("Bad version for epilog version={0} expected={1}",
721 epilog.version(), expected_version);
722 }
723
724 UniqueArray<std::byte> json_bytes;
725
726 // Lit les données JSON
727 {
728 Int64 file_offset = epilog.jsonDataInfoFileOffset();
729 Int64 meta_data_size = epilog.jsonDataInfoSize();
730 //std::cout << "FILE_INFO: offset=" << file_offset << " meta_data_size=" << meta_data_size << "\n";
731 json_bytes.resize(meta_data_size);
732 _readDirect(file_offset, json_bytes);
733 }
734
735 // Remplit les infos de la base de données à partir du JSON
736 {
737 JSONDocument json_doc;
738 json_doc.parse(json_bytes);
739 JSONValue root = json_doc.root();
740 JSONValue data = root.child("Data");
741 UniqueArray<Int64> extents;
742 extents.reserve(12);
743 for (JSONValue v : data.valueAsArray()) {
744 String name = v.child("Name").value();
745 Int64 file_offset = v.child("FileOffset").valueAsInt64();
746 //std::cout << "Name=" << name << "\n";
747 //std::cout << "FileOffset=" << file_offset << "\n";
748 JSONValueList extents_info = v.child("Extents").valueAsArray();
749 extents.clear();
750 for (JSONValue v2 : extents_info) {
751 extents.add(v2.valueAsInt64());
752 }
753 Impl::DataInfo x;
754 x.m_file_offset = file_offset;
755 x.m_extents.fill(extents.view());
756 m_data_infos.insert(std::make_pair(name, x));
757 }
758 }
759}
760
761/*---------------------------------------------------------------------------*/
762/*---------------------------------------------------------------------------*/
763
764void KeyValueTextReader::
765getExtents(const String& key_name, SmallSpan<Int64> extents)
766{
767 Integer dimension_array_size = extents.size();
768 if (m_p->m_version >= 3) {
769 Impl::DataInfo& data = m_p->findData(key_name);
770 Impl::ExtentsInfo& exi = data.m_extents;
771 if (extents.size() != exi.size())
772 ARCANE_FATAL("Bad size for extents size={0} expected={1}", extents.size(), exi.size());
773 extents.copy(exi.view());
774 }
775 else {
776 if (m_p->m_version == 1) {
777 // Dans la version 1, les dimensions sont des 'Int32'
779 if (dimension_array_size > 0) {
780 dims.resize(dimension_array_size);
781 m_p->m_reader.read(asWritableBytes(dims));
782 }
783 for (Integer i = 0; i < dimension_array_size; ++i)
784 extents[i] = dims[i];
785 }
786 else {
787 if (dimension_array_size > 0) {
788 m_p->m_reader.read(asWritableBytes(extents));
789 }
790 }
791 }
792}
793
794/*---------------------------------------------------------------------------*/
795/*---------------------------------------------------------------------------*/
796
797void KeyValueTextReader::Impl::
798readIntegers(const String& key, Span<Integer> values)
799{
800 _setFileOffset(key);
801 m_reader.readIntegers(values);
802}
803
804/*---------------------------------------------------------------------------*/
805/*---------------------------------------------------------------------------*/
806
807void KeyValueTextReader::Impl::
808read(const String& key, Span<std::byte> values)
809{
810 _setFileOffset(key);
811
812 IDataCompressor* d = m_data_compressor.get();
813 Int64 len = values.size();
814 if (d && len > d->minCompressSize()) {
815 UniqueArray<std::byte> compressed_values;
816 Int64 compressed_size = 0;
817 m_reader.read(asWritableBytes(Span<Int64>(&compressed_size, 1)));
818 compressed_values.resize(compressed_size);
819 _read2(key, compressed_values);
820 m_data_compressor->decompress(compressed_values, values);
821 }
822 else {
823 _read2(key, values);
824 }
825}
826
827/*---------------------------------------------------------------------------*/
828/*---------------------------------------------------------------------------*/
829
830void KeyValueTextReader::Impl::
831_read2(const String& key, Span<std::byte> values)
832{
833 if (m_hash_database.get()) {
834 IHashAlgorithm* hash_algo = m_hash_algorithm.get();
835 if (!hash_algo)
836 ARCANE_FATAL("Can not use hash database without hash algorithm");
837 Int32 hash_size = hash_algo->hashSize();
838 SmallArray<Byte, 1024> hash_as_bytes;
839 hash_as_bytes.resize(hash_size);
840 m_reader.read(asWritableBytes(hash_as_bytes));
841 String hash_value = Convert::toHexaString(hash_as_bytes);
842 info(5) << "READ_KW_HASH key=" << key << " hash=" << hash_value << " expected_len=" << values.size();
843 HashDatabaseReadArgs args(hash_value, values);
844 m_hash_database->readValues(args);
845 }
846 else
847 m_reader.read(values);
848}
849
850/*---------------------------------------------------------------------------*/
851/*---------------------------------------------------------------------------*/
852
853void KeyValueTextReader::Impl::
854_setFileOffset(const String& key_name)
855{
856 // Avec les versions antérieures à la version 3, c'est l'appelant qui
857 // positionne l'offset car il est le seul à le connaitre.
858 if (m_version >= 3) {
859 Impl::DataInfo& data = findData(key_name);
860 m_reader.setFileOffset(data.m_file_offset);
861 }
862}
863
864/*---------------------------------------------------------------------------*/
865/*---------------------------------------------------------------------------*/
866
867/*---------------------------------------------------------------------------*/
868/*---------------------------------------------------------------------------*/
869
870String KeyValueTextReader::
871fileName() const
872{
873 return m_p->m_reader.fileName();
874}
875
876/*---------------------------------------------------------------------------*/
877/*---------------------------------------------------------------------------*/
878
879void KeyValueTextReader::
880setFileOffset(Int64 v)
881{
882 m_p->m_reader.setFileOffset(v);
883}
884
885/*---------------------------------------------------------------------------*/
886/*---------------------------------------------------------------------------*/
887
888void KeyValueTextReader::
889setDataCompressor(Ref<IDataCompressor> ds)
890{
891 m_p->m_data_compressor = ds;
892}
893
894/*---------------------------------------------------------------------------*/
895/*---------------------------------------------------------------------------*/
896
897Ref<IDataCompressor> KeyValueTextReader::
898dataCompressor() const
899{
900 return m_p->m_data_compressor;
901}
902
903/*---------------------------------------------------------------------------*/
904/*---------------------------------------------------------------------------*/
905
906void KeyValueTextReader::
907setHashAlgorithm(Ref<IHashAlgorithm> v)
908{
909 m_p->m_hash_algorithm = v;
910}
911
912/*---------------------------------------------------------------------------*/
913/*---------------------------------------------------------------------------*/
914
915Ref<IHashAlgorithm> KeyValueTextReader::
916hashAlgorithm() const
917{
918 return m_p->m_hash_algorithm;
919}
920
921/*---------------------------------------------------------------------------*/
922/*---------------------------------------------------------------------------*/
923
924void KeyValueTextReader::
925readIntegers(const String& key, Span<Integer> values)
926{
927 m_p->readIntegers(key, values);
928}
929
930/*---------------------------------------------------------------------------*/
931/*---------------------------------------------------------------------------*/
932
933void KeyValueTextReader::
934read(const String& key, Span<std::byte> values)
935{
936 m_p->read(key, values);
937}
938
939/*---------------------------------------------------------------------------*/
940/*---------------------------------------------------------------------------*/
941
942} // End namespace Arcane::impl
943
944/*---------------------------------------------------------------------------*/
945/*---------------------------------------------------------------------------*/
#define ARCANE_CHECK_POINTER(ptr)
Macro retournant le pointeur ptr s'il est non nul ou lancant une exception s'il est nul.
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
Integer size() const
Nombre d'éléments du vecteur.
Tableau d'items de types quelconques.
void clear()
Supprime les éléments du tableau.
void resize(Int64 s)
Change le nombre d'éléments du tableau à s.
void reserve(Int64 new_capacity)
Réserve le mémoire pour new_capacity éléments.
ArrayView< T > view() const
Vue mutable sur ce tableau.
void add(ConstReferenceType val)
Ajoute l'élément val à la fin du tableau.
Interface d'un algorithme de hashage.
Interface du gestionnaire de traces.
virtual TraceMessage info()=0
Flot pour un message d'information.
Gestion d'un document JSON.
Definition JSONReader.h:230
void parse(Span< const Byte > bytes)
Lit le fichier au format UTF-8.
JSONValue root() const
Elément racine.
Liste de valeurs d'un document JSON.
Definition JSONReader.h:199
Représente une valeur JSON.
Definition JSONReader.h:43
JSONValue child(StringView name) const
Valeur fille de nom name. Retourne une valeur nulle si non trouvé.
Ecrivain au format JSON.
Definition JSONWriter.h:33
Vue d'un tableau d'éléments de type T.
Definition Span.h:673
constexpr __host__ __device__ SizeType size() const noexcept
Retourne la taille du tableau.
Definition Span.h:212
constexpr __host__ __device__ SizeType length() const noexcept
Nombre d'éléments du tableau.
Definition Span.h:216
Vue d'un tableau d'éléments de type T.
Definition Span.h:513
Vue sur une chaîne de caractères UTF-8.
Definition StringView.h:47
constexpr Int64 size() const ARCCORE_NOEXCEPT
Longueur en octet de la chaîne de caractères.
Definition StringView.h:102
constexpr Span< const Byte > bytes() const ARCCORE_NOEXCEPT
Retourne la conversion de l'instance dans l'encodage UTF-8.
Definition StringView.h:96
Chaîne de caractères unicode.
bool null() const
Retourne true si la chaîne est nulle.
Definition String.cc:304
TraceAccessor(ITraceMng *m)
Construit un accesseur via le gestionnaire de trace m.
TraceMessage info() const
Flot pour un message d'information.
ITraceMng * traceMng() const
Gestionnaire de trace.
Vecteur 1D de données avec sémantique par valeur (style STL).
Classe pour calculer le hash d'un tableau.
void _readHeader()
En-tête du format de fichier.
void _writeHeader()
En-tête du format de fichier.
Classe d'écriture d'un fichier texte pour les protections/reprises.
Definition TextReader2.h:38
Classe d'écriture d'un fichier texte pour les protections/reprises.
Definition TextWriter2.h:38
Integer len(const char *s)
Retourne la longueur de la chaîne s.
Integer toInteger(Real r)
Converti un Int64 en un Integer.
String toHexaString(ByteConstArrayView input)
Converti un tableau d'octet en sa représentation hexadécimale.
Definition Convert.cc:75
ARCCORE_BASE_EXPORT Real getRealTime()
Temps Real utilisé en secondes.
ARCCORE_BASE_EXPORT String getEnvironmentVariable(const String &name)
Variable d'environnement du nom name.
impl::SpanTypeFromSize< std::byte, SizeType >::SpanType asWritableBytes(const SpanImpl< DataType, SizeType, Extent > &s)
Converti la vue en un tableau d'octets modifiables.
Definition Span.h:919
std::int64_t Int64
Type entier signé sur 64 bits.
Int32 Integer
Type représentant un entier.
ConstArrayView< Int64 > Int64ConstArrayView
Equivalent C d'un tableau à une dimension d'entiers 64 bits.
Definition UtilsTypes.h:567
impl::SpanTypeFromSize< conststd::byte, SizeType >::SpanType asBytes(const SpanImpl< DataType, SizeType, Extent > &s)
Converti la vue en un tableau d'octets non modifiables.
Definition Span.h:884
void arcaneCallFunctionAndTerminateIfThrow(std::function< void()> function)
void binaryRead(std::istream &istr, const Span< std::byte > &bytes)
Lit en binaire le contenu de bytes depuis le flot istr.
Definition ArrayView.cc:70
double Real
Type représentant un réel.
unsigned char Byte
Type d'un octet.
Definition BaseTypes.h:43
UniqueArray< Integer > IntegerUniqueArray
Tableau dynamique à une dimension d'entiers.
Definition UtilsTypes.h:434
void binaryWrite(std::ostream &ostr, const Span< const std::byte > &bytes)
Ecrit en binaire le contenu de bytes sur le flot ostr.
Definition ArrayView.cc:60
std::int32_t Int32
Type entier signé sur 32 bits.