librats 0.3.1 → 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +405 -405
- package/binding.gyp +95 -95
- package/lib/index.d.ts +522 -522
- package/lib/index.js +82 -82
- package/package.json +68 -68
- package/scripts/build-librats.js +194 -194
- package/scripts/postinstall.js +52 -52
- package/scripts/prepare-package.js +91 -91
- package/scripts/verify-installation.js +119 -119
- package/src/librats_node.cpp +1174 -1174
|
@@ -1,91 +1,91 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
const fs = require('fs');
|
|
4
|
-
const path = require('path');
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* Prepare package script
|
|
8
|
-
*
|
|
9
|
-
* This script runs before packing and publishing to ensure the package
|
|
10
|
-
* is ready for distribution. It checks that all necessary files are present.
|
|
11
|
-
*/
|
|
12
|
-
|
|
13
|
-
console.log('Preparing package for distribution...');
|
|
14
|
-
|
|
15
|
-
const projectRoot = path.resolve(__dirname, '..', '..');
|
|
16
|
-
const nodejsRoot = path.resolve(__dirname, '..');
|
|
17
|
-
|
|
18
|
-
// Critical files that must be included in the package
|
|
19
|
-
const criticalFiles = [
|
|
20
|
-
// Node.js binding files
|
|
21
|
-
{ path: path.join(nodejsRoot, 'binding.gyp'), desc: 'Node.js binding configuration' },
|
|
22
|
-
{ path: path.join(nodejsRoot, 'lib', 'index.js'), desc: 'JavaScript wrapper' },
|
|
23
|
-
{ path: path.join(nodejsRoot, 'lib', 'index.d.ts'), desc: 'TypeScript definitions' },
|
|
24
|
-
{ path: path.join(nodejsRoot, 'src', 'librats_node.cpp'), desc: 'Node.js binding source' },
|
|
25
|
-
{ path: path.join(nodejsRoot, 'scripts', 'build-librats.js'), desc: 'Build script' },
|
|
26
|
-
{ path: path.join(nodejsRoot, 'scripts', 'postinstall.js'), desc: 'Post-install script' },
|
|
27
|
-
|
|
28
|
-
// Librats C++ library files
|
|
29
|
-
{ path: path.join(projectRoot, 'CMakeLists.txt'), desc: 'CMake configuration' },
|
|
30
|
-
{ path: path.join(projectRoot, 'src', 'librats.cpp'), desc: 'Librats main source' },
|
|
31
|
-
{ path: path.join(projectRoot, 'src', 'librats.h'), desc: 'Librats main header' },
|
|
32
|
-
{ path: path.join(projectRoot, 'src', 'librats_c.cpp'), desc: 'Librats C API' },
|
|
33
|
-
{ path: path.join(projectRoot, 'src', 'librats_c.h'), desc: 'Librats C API header' },
|
|
34
|
-
];
|
|
35
|
-
|
|
36
|
-
let allFilesPresent = true;
|
|
37
|
-
let missingFiles = [];
|
|
38
|
-
|
|
39
|
-
console.log('\nChecking required files:');
|
|
40
|
-
for (const file of criticalFiles) {
|
|
41
|
-
if (fs.existsSync(file.path)) {
|
|
42
|
-
console.log(` ✓ ${file.desc}`);
|
|
43
|
-
} else {
|
|
44
|
-
console.log(` ✗ ${file.desc} - MISSING`);
|
|
45
|
-
allFilesPresent = false;
|
|
46
|
-
missingFiles.push(file);
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
if (!allFilesPresent) {
|
|
51
|
-
console.error('\n❌ ERROR: Some required files are missing!');
|
|
52
|
-
console.error('\nMissing files:');
|
|
53
|
-
missingFiles.forEach(file => {
|
|
54
|
-
console.error(` - ${file.path}`);
|
|
55
|
-
console.error(` (${file.desc})`);
|
|
56
|
-
});
|
|
57
|
-
console.error('\nThe package cannot be published without these files.');
|
|
58
|
-
process.exit(1);
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
// Check that lib directory exists and has content
|
|
62
|
-
const libDir = path.join(nodejsRoot, 'lib');
|
|
63
|
-
if (!fs.existsSync(libDir)) {
|
|
64
|
-
console.error('\n❌ ERROR: lib directory does not exist!');
|
|
65
|
-
console.error('Create it with the JavaScript wrapper and TypeScript definitions.');
|
|
66
|
-
process.exit(1);
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
// Check that scripts directory exists
|
|
70
|
-
const scriptsDir = path.join(nodejsRoot, 'scripts');
|
|
71
|
-
if (!fs.existsSync(scriptsDir)) {
|
|
72
|
-
console.error('\n❌ ERROR: scripts directory does not exist!');
|
|
73
|
-
console.error('This directory is required for the build process.');
|
|
74
|
-
process.exit(1);
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
// Count source files
|
|
78
|
-
const srcDir = path.join(projectRoot, 'src');
|
|
79
|
-
if (fs.existsSync(srcDir)) {
|
|
80
|
-
const sourceFiles = fs.readdirSync(srcDir)
|
|
81
|
-
.filter(f => f.endsWith('.cpp') || f.endsWith('.h'));
|
|
82
|
-
console.log(`\n✓ Found ${sourceFiles.length} source files in src/`);
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
console.log('\n✅ Package is ready for distribution!\n');
|
|
86
|
-
console.log('When users install this package via npm:');
|
|
87
|
-
console.log(' 1. The preinstall script will build the librats C++ library');
|
|
88
|
-
console.log(' 2. The install script will build the Node.js native addon');
|
|
89
|
-
console.log(' 3. The postinstall script will verify the installation');
|
|
90
|
-
console.log('\nNo manual build steps are required by users!\n');
|
|
91
|
-
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
const fs = require('fs');
|
|
4
|
+
const path = require('path');
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Prepare package script
|
|
8
|
+
*
|
|
9
|
+
* This script runs before packing and publishing to ensure the package
|
|
10
|
+
* is ready for distribution. It checks that all necessary files are present.
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
console.log('Preparing package for distribution...');
|
|
14
|
+
|
|
15
|
+
const projectRoot = path.resolve(__dirname, '..', '..');
|
|
16
|
+
const nodejsRoot = path.resolve(__dirname, '..');
|
|
17
|
+
|
|
18
|
+
// Critical files that must be included in the package
|
|
19
|
+
const criticalFiles = [
|
|
20
|
+
// Node.js binding files
|
|
21
|
+
{ path: path.join(nodejsRoot, 'binding.gyp'), desc: 'Node.js binding configuration' },
|
|
22
|
+
{ path: path.join(nodejsRoot, 'lib', 'index.js'), desc: 'JavaScript wrapper' },
|
|
23
|
+
{ path: path.join(nodejsRoot, 'lib', 'index.d.ts'), desc: 'TypeScript definitions' },
|
|
24
|
+
{ path: path.join(nodejsRoot, 'src', 'librats_node.cpp'), desc: 'Node.js binding source' },
|
|
25
|
+
{ path: path.join(nodejsRoot, 'scripts', 'build-librats.js'), desc: 'Build script' },
|
|
26
|
+
{ path: path.join(nodejsRoot, 'scripts', 'postinstall.js'), desc: 'Post-install script' },
|
|
27
|
+
|
|
28
|
+
// Librats C++ library files
|
|
29
|
+
{ path: path.join(projectRoot, 'CMakeLists.txt'), desc: 'CMake configuration' },
|
|
30
|
+
{ path: path.join(projectRoot, 'src', 'librats.cpp'), desc: 'Librats main source' },
|
|
31
|
+
{ path: path.join(projectRoot, 'src', 'librats.h'), desc: 'Librats main header' },
|
|
32
|
+
{ path: path.join(projectRoot, 'src', 'librats_c.cpp'), desc: 'Librats C API' },
|
|
33
|
+
{ path: path.join(projectRoot, 'src', 'librats_c.h'), desc: 'Librats C API header' },
|
|
34
|
+
];
|
|
35
|
+
|
|
36
|
+
let allFilesPresent = true;
|
|
37
|
+
let missingFiles = [];
|
|
38
|
+
|
|
39
|
+
console.log('\nChecking required files:');
|
|
40
|
+
for (const file of criticalFiles) {
|
|
41
|
+
if (fs.existsSync(file.path)) {
|
|
42
|
+
console.log(` ✓ ${file.desc}`);
|
|
43
|
+
} else {
|
|
44
|
+
console.log(` ✗ ${file.desc} - MISSING`);
|
|
45
|
+
allFilesPresent = false;
|
|
46
|
+
missingFiles.push(file);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
if (!allFilesPresent) {
|
|
51
|
+
console.error('\n❌ ERROR: Some required files are missing!');
|
|
52
|
+
console.error('\nMissing files:');
|
|
53
|
+
missingFiles.forEach(file => {
|
|
54
|
+
console.error(` - ${file.path}`);
|
|
55
|
+
console.error(` (${file.desc})`);
|
|
56
|
+
});
|
|
57
|
+
console.error('\nThe package cannot be published without these files.');
|
|
58
|
+
process.exit(1);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// Check that lib directory exists and has content
|
|
62
|
+
const libDir = path.join(nodejsRoot, 'lib');
|
|
63
|
+
if (!fs.existsSync(libDir)) {
|
|
64
|
+
console.error('\n❌ ERROR: lib directory does not exist!');
|
|
65
|
+
console.error('Create it with the JavaScript wrapper and TypeScript definitions.');
|
|
66
|
+
process.exit(1);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Check that scripts directory exists
|
|
70
|
+
const scriptsDir = path.join(nodejsRoot, 'scripts');
|
|
71
|
+
if (!fs.existsSync(scriptsDir)) {
|
|
72
|
+
console.error('\n❌ ERROR: scripts directory does not exist!');
|
|
73
|
+
console.error('This directory is required for the build process.');
|
|
74
|
+
process.exit(1);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// Count source files
|
|
78
|
+
const srcDir = path.join(projectRoot, 'src');
|
|
79
|
+
if (fs.existsSync(srcDir)) {
|
|
80
|
+
const sourceFiles = fs.readdirSync(srcDir)
|
|
81
|
+
.filter(f => f.endsWith('.cpp') || f.endsWith('.h'));
|
|
82
|
+
console.log(`\n✓ Found ${sourceFiles.length} source files in src/`);
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
console.log('\n✅ Package is ready for distribution!\n');
|
|
86
|
+
console.log('When users install this package via npm:');
|
|
87
|
+
console.log(' 1. The preinstall script will build the librats C++ library');
|
|
88
|
+
console.log(' 2. The install script will build the Node.js native addon');
|
|
89
|
+
console.log(' 3. The postinstall script will verify the installation');
|
|
90
|
+
console.log('\nNo manual build steps are required by users!\n');
|
|
91
|
+
|
|
@@ -1,119 +1,119 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Installation Verification Script
|
|
5
|
-
*
|
|
6
|
-
* Checks if librats was installed correctly and all components are working.
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
const fs = require('fs');
|
|
10
|
-
const path = require('path');
|
|
11
|
-
|
|
12
|
-
console.log('🔍 Verifying librats installation...\n');
|
|
13
|
-
|
|
14
|
-
let allChecksPassed = true;
|
|
15
|
-
|
|
16
|
-
// Check 1: Can we load the module?
|
|
17
|
-
console.log('1. Loading librats module...');
|
|
18
|
-
try {
|
|
19
|
-
const librats = require('../lib/index.js');
|
|
20
|
-
console.log(' ✅ Module loaded successfully\n');
|
|
21
|
-
|
|
22
|
-
// Check 2: Version info
|
|
23
|
-
console.log('2. Checking version info...');
|
|
24
|
-
try {
|
|
25
|
-
const versionString = librats.getVersionString();
|
|
26
|
-
const version = librats.getVersion();
|
|
27
|
-
console.log(` ✅ Version: ${versionString}`);
|
|
28
|
-
console.log(` ✅ Components: ${version.major}.${version.minor}.${version.patch}.${version.build}\n`);
|
|
29
|
-
} catch (err) {
|
|
30
|
-
console.log(' ❌ Failed to get version info:', err.message);
|
|
31
|
-
allChecksPassed = false;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
// Check 3: Constants
|
|
35
|
-
console.log('3. Checking constants...');
|
|
36
|
-
try {
|
|
37
|
-
if (typeof librats.ConnectionStrategy.DIRECT_ONLY === 'number') {
|
|
38
|
-
console.log(' ✅ ConnectionStrategy constants defined');
|
|
39
|
-
}
|
|
40
|
-
if (typeof librats.ErrorCodes.SUCCESS === 'number') {
|
|
41
|
-
console.log(' ✅ ErrorCodes constants defined\n');
|
|
42
|
-
}
|
|
43
|
-
} catch (err) {
|
|
44
|
-
console.log(' ❌ Constants check failed:', err.message);
|
|
45
|
-
allChecksPassed = false;
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
// Check 4: Can we create a client?
|
|
49
|
-
console.log('4. Testing client creation...');
|
|
50
|
-
try {
|
|
51
|
-
const RatsClient = librats.RatsClient;
|
|
52
|
-
const testPort = 19999;
|
|
53
|
-
const client = new RatsClient(testPort);
|
|
54
|
-
console.log(' ✅ Client created successfully\n');
|
|
55
|
-
|
|
56
|
-
// Check 5: Can we start and stop?
|
|
57
|
-
console.log('5. Testing start/stop...');
|
|
58
|
-
try {
|
|
59
|
-
const started = client.start();
|
|
60
|
-
if (started) {
|
|
61
|
-
console.log(' ✅ Client started successfully');
|
|
62
|
-
|
|
63
|
-
const peerId = client.getOurPeerId();
|
|
64
|
-
if (peerId && typeof peerId === 'string') {
|
|
65
|
-
console.log(` ✅ Got peer ID: ${peerId}`);
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
const peerCount = client.getPeerCount();
|
|
69
|
-
console.log(` ✅ Peer count: ${peerCount}`);
|
|
70
|
-
|
|
71
|
-
client.stop();
|
|
72
|
-
console.log(' ✅ Client stopped successfully\n');
|
|
73
|
-
} else {
|
|
74
|
-
console.log(' ⚠️ Warning: Client failed to start (port may be in use)\n');
|
|
75
|
-
}
|
|
76
|
-
} catch (err) {
|
|
77
|
-
console.log(' ❌ Start/stop test failed:', err.message);
|
|
78
|
-
allChecksPassed = false;
|
|
79
|
-
}
|
|
80
|
-
} catch (err) {
|
|
81
|
-
console.log(' ❌ Client creation failed:', err.message);
|
|
82
|
-
allChecksPassed = false;
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
// Check 6: TypeScript definitions
|
|
86
|
-
console.log('6. Checking TypeScript definitions...');
|
|
87
|
-
const tsDefsPath = path.join(__dirname, '..', 'lib', 'index.d.ts');
|
|
88
|
-
if (fs.existsSync(tsDefsPath)) {
|
|
89
|
-
console.log(' ✅ TypeScript definitions found\n');
|
|
90
|
-
} else {
|
|
91
|
-
console.log(' ⚠️ Warning: TypeScript definitions not found\n');
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
} catch (err) {
|
|
95
|
-
console.log(' ❌ Failed to load module:', err.message);
|
|
96
|
-
console.log('\nError details:');
|
|
97
|
-
console.log(err.stack);
|
|
98
|
-
allChecksPassed = false;
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
// Summary
|
|
102
|
-
console.log('━'.repeat(60));
|
|
103
|
-
if (allChecksPassed) {
|
|
104
|
-
console.log('✅ All checks passed! Librats is installed correctly.\n');
|
|
105
|
-
console.log('You can now use librats in your project:');
|
|
106
|
-
console.log(" const { RatsClient } = require('librats');");
|
|
107
|
-
console.log(' const client = new RatsClient(8080);');
|
|
108
|
-
console.log(' client.start();\n');
|
|
109
|
-
process.exit(0);
|
|
110
|
-
} else {
|
|
111
|
-
console.log('❌ Some checks failed. Installation may be incomplete.\n');
|
|
112
|
-
console.log('Troubleshooting:');
|
|
113
|
-
console.log(' 1. Try rebuilding: npm rebuild librats');
|
|
114
|
-
console.log(' 2. Check build tools are installed (CMake, C++ compiler)');
|
|
115
|
-
console.log(' 3. Check the logs above for specific errors');
|
|
116
|
-
console.log(' 4. See: https://github.com/librats/librats/tree/main/nodejs\n');
|
|
117
|
-
process.exit(1);
|
|
118
|
-
}
|
|
119
|
-
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Installation Verification Script
|
|
5
|
+
*
|
|
6
|
+
* Checks if librats was installed correctly and all components are working.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
const fs = require('fs');
|
|
10
|
+
const path = require('path');
|
|
11
|
+
|
|
12
|
+
console.log('🔍 Verifying librats installation...\n');
|
|
13
|
+
|
|
14
|
+
let allChecksPassed = true;
|
|
15
|
+
|
|
16
|
+
// Check 1: Can we load the module?
|
|
17
|
+
console.log('1. Loading librats module...');
|
|
18
|
+
try {
|
|
19
|
+
const librats = require('../lib/index.js');
|
|
20
|
+
console.log(' ✅ Module loaded successfully\n');
|
|
21
|
+
|
|
22
|
+
// Check 2: Version info
|
|
23
|
+
console.log('2. Checking version info...');
|
|
24
|
+
try {
|
|
25
|
+
const versionString = librats.getVersionString();
|
|
26
|
+
const version = librats.getVersion();
|
|
27
|
+
console.log(` ✅ Version: ${versionString}`);
|
|
28
|
+
console.log(` ✅ Components: ${version.major}.${version.minor}.${version.patch}.${version.build}\n`);
|
|
29
|
+
} catch (err) {
|
|
30
|
+
console.log(' ❌ Failed to get version info:', err.message);
|
|
31
|
+
allChecksPassed = false;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
// Check 3: Constants
|
|
35
|
+
console.log('3. Checking constants...');
|
|
36
|
+
try {
|
|
37
|
+
if (typeof librats.ConnectionStrategy.DIRECT_ONLY === 'number') {
|
|
38
|
+
console.log(' ✅ ConnectionStrategy constants defined');
|
|
39
|
+
}
|
|
40
|
+
if (typeof librats.ErrorCodes.SUCCESS === 'number') {
|
|
41
|
+
console.log(' ✅ ErrorCodes constants defined\n');
|
|
42
|
+
}
|
|
43
|
+
} catch (err) {
|
|
44
|
+
console.log(' ❌ Constants check failed:', err.message);
|
|
45
|
+
allChecksPassed = false;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
// Check 4: Can we create a client?
|
|
49
|
+
console.log('4. Testing client creation...');
|
|
50
|
+
try {
|
|
51
|
+
const RatsClient = librats.RatsClient;
|
|
52
|
+
const testPort = 19999;
|
|
53
|
+
const client = new RatsClient(testPort);
|
|
54
|
+
console.log(' ✅ Client created successfully\n');
|
|
55
|
+
|
|
56
|
+
// Check 5: Can we start and stop?
|
|
57
|
+
console.log('5. Testing start/stop...');
|
|
58
|
+
try {
|
|
59
|
+
const started = client.start();
|
|
60
|
+
if (started) {
|
|
61
|
+
console.log(' ✅ Client started successfully');
|
|
62
|
+
|
|
63
|
+
const peerId = client.getOurPeerId();
|
|
64
|
+
if (peerId && typeof peerId === 'string') {
|
|
65
|
+
console.log(` ✅ Got peer ID: ${peerId}`);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
const peerCount = client.getPeerCount();
|
|
69
|
+
console.log(` ✅ Peer count: ${peerCount}`);
|
|
70
|
+
|
|
71
|
+
client.stop();
|
|
72
|
+
console.log(' ✅ Client stopped successfully\n');
|
|
73
|
+
} else {
|
|
74
|
+
console.log(' ⚠️ Warning: Client failed to start (port may be in use)\n');
|
|
75
|
+
}
|
|
76
|
+
} catch (err) {
|
|
77
|
+
console.log(' ❌ Start/stop test failed:', err.message);
|
|
78
|
+
allChecksPassed = false;
|
|
79
|
+
}
|
|
80
|
+
} catch (err) {
|
|
81
|
+
console.log(' ❌ Client creation failed:', err.message);
|
|
82
|
+
allChecksPassed = false;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// Check 6: TypeScript definitions
|
|
86
|
+
console.log('6. Checking TypeScript definitions...');
|
|
87
|
+
const tsDefsPath = path.join(__dirname, '..', 'lib', 'index.d.ts');
|
|
88
|
+
if (fs.existsSync(tsDefsPath)) {
|
|
89
|
+
console.log(' ✅ TypeScript definitions found\n');
|
|
90
|
+
} else {
|
|
91
|
+
console.log(' ⚠️ Warning: TypeScript definitions not found\n');
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
} catch (err) {
|
|
95
|
+
console.log(' ❌ Failed to load module:', err.message);
|
|
96
|
+
console.log('\nError details:');
|
|
97
|
+
console.log(err.stack);
|
|
98
|
+
allChecksPassed = false;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// Summary
|
|
102
|
+
console.log('━'.repeat(60));
|
|
103
|
+
if (allChecksPassed) {
|
|
104
|
+
console.log('✅ All checks passed! Librats is installed correctly.\n');
|
|
105
|
+
console.log('You can now use librats in your project:');
|
|
106
|
+
console.log(" const { RatsClient } = require('librats');");
|
|
107
|
+
console.log(' const client = new RatsClient(8080);');
|
|
108
|
+
console.log(' client.start();\n');
|
|
109
|
+
process.exit(0);
|
|
110
|
+
} else {
|
|
111
|
+
console.log('❌ Some checks failed. Installation may be incomplete.\n');
|
|
112
|
+
console.log('Troubleshooting:');
|
|
113
|
+
console.log(' 1. Try rebuilding: npm rebuild librats');
|
|
114
|
+
console.log(' 2. Check build tools are installed (CMake, C++ compiler)');
|
|
115
|
+
console.log(' 3. Check the logs above for specific errors');
|
|
116
|
+
console.log(' 4. See: https://github.com/librats/librats/tree/main/nodejs\n');
|
|
117
|
+
process.exit(1);
|
|
118
|
+
}
|
|
119
|
+
|