- App.tsx: full navigation (Auth stack + Main tabs with 5 screens) - Auth: LoginScreen, RegisterScreen, ForgotPasswordScreen - HomeScreen: dashboard with IoT metrics, weather widget, alerts, quick actions, sensors - MapScreen: interactive map with layer toggles (6 layers) - MarketplaceScreen: categories (6), products (5), search - ChatScreen: AI chat with quick prompts (4), bot responses - ProfileScreen: user info, stats, menu (9 items), logout - AlertsScreen: alert list with severity, acknowledge - SensorsScreen: sensor list with type filters (6 types), search - ZonesScreen: zone cards with stats - SettingsScreen: language picker (FR/EN/ES/DE), privacy, about - Stores: iotStore (sensors, zones, alerts), notificationStore, uiStore + i18n - Hooks: useSensors, useAlerts, useNotifications, useLocation - Components: Card, Button, LoadingSpinner, ErrorBoundary, Header - Services: iotService, notificationService (with axios API client) - Utils: formatters (temp, AQI, noise, dates), validators (email, password, IBAN) - Theme: colors.ts with full design system (Blue Ocean palette) - Ditto: fixed MongoDB connection, new JWT secrets, official gateway image
51 lines
1.6 KiB
JavaScript
51 lines
1.6 KiB
JavaScript
'use strict';
|
|
|
|
var test = require('tape');
|
|
var parse = require('../').parse;
|
|
|
|
test('parse shell commands', function (t) {
|
|
t.same(parse(''), [], 'parses an empty string');
|
|
|
|
t['throws'](
|
|
function () { parse('${}'); },
|
|
Error,
|
|
'empty substitution throws'
|
|
);
|
|
t['throws'](
|
|
function () { parse('${'); },
|
|
Error,
|
|
'incomplete substitution throws'
|
|
);
|
|
|
|
t.same(parse('a \'b\' "c"'), ['a', 'b', 'c']);
|
|
t.same(
|
|
parse('beep "boop" \'foo bar baz\' "it\'s \\"so\\" groovy"'),
|
|
['beep', 'boop', 'foo bar baz', 'it\'s "so" groovy']
|
|
);
|
|
t.same(parse('a b\\ c d'), ['a', 'b c', 'd']);
|
|
t.same(parse('\\$beep bo\\`op'), ['$beep', 'bo`op']);
|
|
t.same(parse('echo "foo = \\"foo\\""'), ['echo', 'foo = "foo"']);
|
|
t.same(parse(''), []);
|
|
t.same(parse(' '), []);
|
|
t.same(parse('\t'), []);
|
|
t.same(parse('a"b c d"e'), ['ab c de']);
|
|
t.same(parse('a\\ b"c d"\\ e f'), ['a bc d e', 'f']);
|
|
t.same(parse('a\\ b"c d"\\ e\'f g\' h'), ['a bc d ef g', 'h']);
|
|
t.same(parse("x \"bl'a\"'h'"), ['x', "bl'ah"]);
|
|
t.same(parse("x bl^'a^'h'", {}, { escape: '^' }), ['x', "bl'a'h"]);
|
|
t.same(parse('abcH def', {}, { escape: 'H' }), ['abc def']);
|
|
|
|
t.deepEqual(parse('# abc def ghi'), [{ comment: ' abc def ghi' }], 'start-of-line comment content is unparsed');
|
|
t.deepEqual(parse('xyz # abc def ghi'), ['xyz', { comment: ' abc def ghi' }], 'comment content is unparsed');
|
|
|
|
t.deepEqual(parse('-x "" -y'), ['-x', '', '-y'], 'empty string is preserved');
|
|
|
|
t.same(
|
|
parse('2;b', {}, { escape: 'd' }),
|
|
[{ op: '2;b' }],
|
|
'control char in unquoted context mid-token with regex-special escape returns op'
|
|
);
|
|
|
|
t.end();
|
|
});
|