Merge branch 'main' into main

This commit is contained in:
Eric Dubé 2024-04-01 15:06:16 -04:00 committed by GitHub
commit fdd9f30c39
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
42 changed files with 547 additions and 166 deletions

View File

@ -58,9 +58,38 @@ docker compose up
<br/>
See [Configuration](#configuration) for next steps.
<br/>
## ⚠️ Self-Hosting ⚠️
The self-hosted version of Puter is currently in alpha stage and should not be used in production yet. It is under active development and may contain bugs, other issues. Please exercise caution and use it for testing and evaluation purposes only.
## Configuration
Running the server will generate a configuration file at `volatile/config/config.json`.
### Domain Name
To access Puter on your device, you can simply go to the address printed in
the server console (usually `puter.localhost:4100`).
To access Puter from another device, a domain name must be configured, as well as
an `api` subdomain. For example, `example.local` might be the domain name pointing
to the IP address of the server running puter, and `api.example.com` must point to
this address as well. This domain must be specified in the configuration file
(usually `volatile/config/config.json`) as well.
See [domain configuration](./doc/self-hosters/domains.md) for more information.
### Default User
By default, Puter will create a user called `default_user` with the password
`changeme`. A warning will persist in the dev console until this user's
password is changed. Please login to this user and change the password as
your first step. This user by default has 10GB storage instead of the default
(500MB storage) for new/temporary users.
<br/>
## FAQ

View File

@ -0,0 +1,72 @@
# Configurating Domains for Self-Hosted Puter
## Local Network Configuration
### Prerequisite Conditions
Ensure the hosting device has a static IP address to prevent potential connectivity issues due to IP changes. This setup will enable seamless access to Puter and its services across your local network.
### Using Hosts Files
The hosts file is a straightforward way to map domain names to IP addresses on individual devices. It's simple to set up but requires manual changes on each device that needs access to the domains.
#### Windows
1. Open Notepad as an administrator.
2. Open the file located at `C:\Windows\System32\drivers\etc\hosts`.
3. Add lines for your domain and subdomain with the server's IP address, in the
following format:
```
192.168.1.10 puter.local
192.168.1.10 api.puter.local
```
### For macOS and Linux:
1. Open a terminal.
2. Edit the hosts file with a text editor, e.g., `sudo nano /etc/hosts`.
3. Add lines for your domain and subdomain with the server's IP address, in the
following format:
```
192.168.1.10 puter.local
192.168.1.10 api.puter.local
```
4. Save and exit the editor.
### Using Router Configuration
Some routers allow you to add custom DNS rules, letting you configure domain names network-wide without touching each device.
1. Access your routers admin interface (usually through a web browser).
2. Look for DNS or DHCP settings.
3. Add custom DNS mappings for `puter.local` and `api.puter.local` to the hosting device's IP address.
4. Save the changes and reboot the router if necessary.
This method's availability and steps may vary depending on your router's model and firmware.
### Using Local DNS
Setting up a local DNS server on your network allows for flexible and scalable domain name resolution. This method works across all devices automatically once they're configured to use the DNS server.
#### Options for DNS Software:
- **Pi-hole**: Acts as both an ad-blocker and a DNS server. Ideal for easy setup and maintenance.
- **BIND9**: Offers comprehensive DNS server capabilities for complex setups.
- **Dnsmasq**: Lightweight and suitable for smaller networks or those new to running a DNS server.
**contributors note:** feel free to add any software you're aware of
which might help with this to the list. Also, feel free to add instructions here for specific software; our goal is for Puter to be easy to setup with tools you're already familiar with.
#### General Steps:
1. Choose and install DNS server software on a device within your network.
2. Configure the DNS server to resolve `puter.local` and `api.puter.local` to the IP address of your Puter hosting device.
3. Update your routers DHCP settings to distribute the DNS server's IP address to all devices on the network.
By setting up a local DNS server, you gain the most flexibility and control over your network's domain name resolution, ensuring that all devices can access Puter and its API without manual configuration.
## Production Configuration
Please note the self-hosting feature is still in alpha and a public production
deployment is not recommended at this time. However, if you wish to host
publically you can do so following the same steps you normally would to configure
a domain name and ensuring the `api` subdomain points to the server as well.

40
package-lock.json generated
View File

@ -9637,6 +9637,45 @@
"resolved": "https://registry.npmjs.org/string-hash/-/string-hash-1.1.3.tgz",
"integrity": "sha512-kJUvRUFK49aub+a7T1nNE66EJbZBMnBgoC1UbCZ5n6bsZKBRga4KgBRTMn/pFkeCZSYtNeSyMxPDM0AXWELk2A=="
},
"node_modules/string-length": {
"version": "6.0.0",
"resolved": "https://registry.npmjs.org/string-length/-/string-length-6.0.0.tgz",
"integrity": "sha512-1U361pxZHEQ+FeSjzqRpV+cu2vTzYeWeafXFLykiFlv4Vc0n3njgU8HrMbyik5uwm77naWMuVG8fhEF+Ovb1Kg==",
"dependencies": {
"strip-ansi": "^7.1.0"
},
"engines": {
"node": ">=16"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/string-length/node_modules/ansi-regex": {
"version": "6.0.1",
"resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.0.1.tgz",
"integrity": "sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==",
"engines": {
"node": ">=12"
},
"funding": {
"url": "https://github.com/chalk/ansi-regex?sponsor=1"
}
},
"node_modules/string-length/node_modules/strip-ansi": {
"version": "7.1.0",
"resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz",
"integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==",
"dependencies": {
"ansi-regex": "^6.0.1"
},
"engines": {
"node": ">=12"
},
"funding": {
"url": "https://github.com/chalk/strip-ansi?sponsor=1"
}
},
"node_modules/string-width": {
"version": "4.2.3",
"resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
@ -10807,6 +10846,7 @@
"socket.io": "^4.6.2",
"ssh2": "^1.13.0",
"string-hash": "^1.1.3",
"string-length": "^6.0.0",
"svgo": "^3.0.2",
"tiktoken": "^1.0.11",
"uglify-js": "^3.17.4",

View File

@ -61,6 +61,7 @@
"socket.io": "^4.6.2",
"ssh2": "^1.13.0",
"string-hash": "^1.1.3",
"string-length": "^6.0.0",
"svgo": "^3.0.2",
"tiktoken": "^1.0.11",
"uglify-js": "^3.17.4",

View File

@ -182,6 +182,9 @@ const install = async ({ services, app }) => {
const { EventService } = require('./services/EventService');
services.registerService('event', EventService);
const DefaultUserService = require('./services/DefaultUserService');
services.registerService('__default-user', DefaultUserService);
}
const install_legacy = async ({ services }) => {

View File

@ -95,11 +95,8 @@ class Kernel extends AdvancedBase {
root_context.arun(async () => {
await this._install_modules();
});
(async () => {
await this._boot_services();
})();
});
// Error.stackTraceLimit = Infinity;
@ -183,6 +180,7 @@ class Kernel extends AdvancedBase {
await services.emit('start.webserver');
await services.emit('ready.webserver');
}
}

View File

@ -0,0 +1,32 @@
const { TeePromise } = require('../util/promise');
const es_import_promise = new TeePromise();
let stringLength;
(async () => {
stringLength = (await import('string-length')).default;
es_import_promise.resolve();
// console.log('STRING LENGTH', stringLength);
// process.exit(0);
})();
const surrounding_box = (col, lines, lengths) => {
if ( ! lengths ) {
lengths = lines.map(line => stringLength(line));
}
const max_length = Math.max(...lengths);
const c = str => `\x1b[${col}m${str}\x1b[0m`;
const bar = c(Array(max_length + 4).fill('━').join(''));
for ( let i = 0 ; i < lines.length ; i++ ) {
while ( stringLength(lines[i]) < max_length ) {
lines[i] += ' ';
}
lines[i] = `${c('┃ ')} ${lines[i]} ${c(' ┃')}`;
}
lines.unshift(`${c('┏')}${bar}${c('┓')}`);
lines.push(`${c('┗')}${bar}${c('┛')}`);
};
module.exports = {
surrounding_box,
es_import_promise,
};

View File

@ -0,0 +1,124 @@
const { surrounding_box } = require("../fun/dev-console-ui-utils");
const { get_user, generate_system_fsentries, invalidate_cached_user } = require("../helpers");
const { Context } = require("../util/context");
const { asyncSafeSetInterval } = require("../util/promise");
const BaseService = require("./BaseService");
const { Actor, UserActorType } = require("./auth/Actor");
const { DB_WRITE } = require("./database/consts");
const DEFAULT_PASSWORD = 'changeme';
const USERNAME = 'default_user';
class DefaultUserService extends BaseService {
static MODULES = {
bcrypt: require('bcrypt'),
uuidv4: require('uuid').v4,
}
async _init () {
}
async ['__on_ready.webserver'] () {
// check if a user named `default-user` exists
let user = await get_user({ username: USERNAME, cached: false });
if ( ! user ) user = await this.create_default_user_();
// check if user named `default-user` is using default password
const require = this.require;
const tmp_password = await this.get_tmp_password_(user);
console.log(`second input [${tmp_password}]`);
const bcrypt = require('bcrypt');
console.log(...[
'THESE ARE THE ARGS',
tmp_password,
// password_hashed,
user.password
].map(l => l + '\n'));
const is_default_password = await bcrypt.compare(
tmp_password,
user.password
);
if ( ! is_default_password ) return;
// show console widget
this.default_user_widget = () => {
const lines = [
`Your default user has been created!`,
`\x1B[31;1musername:\x1B[0m ${USERNAME}`,
`\x1B[32;1mpassword:\x1B[0m ${tmp_password}`,
`(change the password to remove this message)`
];
surrounding_box('31;1', lines);
return lines;
};
this.start_poll_({ tmp_password, user });
const svc_devConsole = this.services.get('dev-console');
svc_devConsole.add_widget(this.default_user_widget);
}
start_poll_ ({ tmp_password, user }) {
const interval = 1000 * 3; // 3 seconds
const poll_interval = asyncSafeSetInterval(async () => {
const user = await get_user({ username: USERNAME });
const require = this.require;
const bcrypt = require('bcrypt');
const is_default_password = await bcrypt.compare(
tmp_password,
user.password
);
if ( ! is_default_password ) {
const svc_devConsole = this.services.get('dev-console');
svc_devConsole.remove_widget(this.default_user_widget);
clearInterval(poll_interval);
return;
}
}, interval);
}
async create_default_user_ () {
const db = this.services.get('database').get(DB_WRITE, 'default-user');
await db.write(
`
INSERT INTO user (uuid, username, free_storage)
VALUES (?, ?, ?)
`,
[
this.modules.uuidv4(),
USERNAME,
1024 * 1024 * 1024 * 10, // 10 GB
],
);
const user = await get_user({ username: USERNAME });
const tmp_password = await this.get_tmp_password_(user);
console.log(`first input [${tmp_password}]`);
const bcrypt = require('bcrypt');
const password_hashed = await bcrypt.hash(tmp_password, 8);
await db.write(
`UPDATE user SET password = ? WHERE id = ?`,
[
password_hashed,
user.id,
],
);
user.password = password_hashed;
await generate_system_fsentries(user);
invalidate_cached_user(user);
await new Promise(rslv => setTimeout(rslv, 2000));
return user;
}
async get_tmp_password_ (user) {
const actor = await Actor.create(UserActorType, { user });
return await Context.get().sub({ actor }).arun(async () => {
const svc_driver = this.services.get('driver');
const driver_response = await svc_driver.call(
'puter-kvstore', 'get', { key: 'tmp_password' });
if ( driver_response.result ) return driver_response.result.value;
const tmp_password = require('crypto').randomBytes(4).toString('hex');
await svc_driver.call(
'puter-kvstore', 'set', {
key: 'tmp_password',
value: tmp_password });
return tmp_password;
});
}
}
module.exports = DefaultUserService;

View File

@ -27,6 +27,7 @@ var http = require('http');
const fs = require('fs');
const auth = require('../middleware/auth');
const { osclink } = require('../util/strutil');
const { surrounding_box, es_import_promise } = require('../fun/dev-console-ui-utils');
class WebServerService extends BaseService {
static MODULES = {
@ -42,6 +43,7 @@ class WebServerService extends BaseService {
};
async ['__on_start.webserver'] () {
await es_import_promise;
// error handling middleware goes last, as per the
// expressjs documentation:
@ -122,17 +124,7 @@ class WebServerService extends BaseService {
lines[2].length,
0,
];
const max_length = Math.max(...lengths);
const c = str => `\x1b[34;1m${str}\x1b[0m`;
const bar = c(Array(max_length + 4).fill('━').join(''));
for ( let i = 0 ; i < lines.length ; i++ ) {
while ( lines[i].length < max_length ) {
lines[i] += ' ';
}
lines[i] = `${c('┃ ')} ${lines[i]} ${c(' ┃')}`;
}
lines.unshift(`${c('┏')}${bar}${c('┓')}`);
lines.push(`${c('┗')}${bar}${c('┛')}`);
surrounding_box('34;1', lines, lengths);
return lines;
};
{
@ -360,7 +352,7 @@ class WebServerService extends BaseService {
const pad = (width - last_logo.sz) / 2;
const asymmetrical = pad % 1 !== 0;
const pad_left = Math.floor(pad);
const pad_right = Math.ceil(pad) + (asymmetrical ? 1 : 0);
const pad_right = Math.ceil(pad);
for ( let i = 0 ; i < lines.length ; i++ ) {
lines[i] = ' '.repeat(pad_left) + lines[i] + ' '.repeat(pad_right);
}

View File

@ -84,15 +84,28 @@ class DBKVStore extends BaseImplementation {
const db = this.services.get('database').get(DB_WRITE, 'kvstore');
const key_hash = this.modules.murmurhash.v3(key);
await db.write(
`INSERT INTO kv
(user_id, app, kkey_hash, kkey, value)
VALUES
(?, ?, ?, ?, ?)
ON DUPLICATE KEY UPDATE
value = ?`,
[ user.id, app?.uid ?? 'global', key_hash, key, value, value ]
);
try {
await db.write(
`INSERT INTO kv (user_id, app, kkey_hash, kkey, value)
VALUES (?, ?, ?, ?, ?) ` +
db.case({
mysql: 'ON DUPLICATE KEY UPDATE value = ?',
sqlite: ' ',
// sqlite: 'ON CONFLICT(user_id, app, kkey_hash) DO UPDATE SET value = ?',
}),
[
user.id, app?.uid ?? 'global', key_hash, key, value,
...db.case({ mysql: [value], otherwise: [] }),
]
);
} catch (e) {
// if ( e.code !== 'SQLITE_ERROR' && e.code !== 'SQLITE_CONSTRAINT_PRIMARYKEY' ) throw e;
// The "ON CONFLICT" clause isn't currently working.
await db.write(
`UPDATE kv SET value = ? WHERE user_id=? AND app=? AND kkey_hash=?`,
[ value, user.id, app?.uid ?? 'global', key_hash ]
);
}
return true;
},

View File

@ -34,18 +34,41 @@ class MonthlyUsageService extends BaseService {
const maybe_app_id = actor.type.app?.id;
if ( this.db.case({ sqlite: true, otherwise: false }) ) {
return;
}
// UPSERT increment count
await this.db.write(
'INSERT INTO `service_usage_monthly` (`year`, `month`, `key`, `count`, `user_id`, `app_id`, `extra`) ' +
'VALUES (?, ?, ?, 1, ?, ?, ?) ' +
'ON DUPLICATE KEY UPDATE `count` = `count` + 1',
[
year, month, key,
actor.type.user?.id || null,
maybe_app_id || null,
JSON.stringify(extra)
]
);
try {
await this.db.write(
'INSERT INTO `service_usage_monthly` (`year`, `month`, `key`, `count`, `user_id`, `app_id`, `extra`) ' +
'VALUES (?, ?, ?, 1, ?, ?, ?) ' +
this.db.case({
mysql: 'ON DUPLICATE KEY UPDATE `count` = `count` + 1, `extra` = ?',
sqlite: ' ',
// sqlite: 'ON CONFLICT(`year`, `month`, `key`, `user_id`, `app_id`) ' +
// 'DO UPDATE SET `count` = `count` + 1 AND `extra` = ?',
}),
[
year, month, key, actor.type.user.id, maybe_app_id, JSON.stringify(extra),
...this.db.case({ mysql: [JSON.stringify(extra)], otherwise: [] }),
]
);
} catch (e) {
// if ( e.code !== 'SQLITE_ERROR' && e.code !== 'SQLITE_CONSTRAINT_PRIMARYKEY' ) throw e;
// The "ON CONFLICT" clause isn't currently working.
await this.db.write(
'UPDATE `service_usage_monthly` ' +
'SET `count` = `count` + 1, `extra` = ? ' +
'WHERE `year` = ? AND `month` = ? AND `key` = ? ' +
'AND `user_id` = ? AND `app_id` = ?',
[
JSON.stringify(extra),
year, month, key, actor.type.user.id, maybe_app_id,
]
);
}
}
async check (actor, specifiers) {

View File

@ -1217,7 +1217,7 @@ $(document).on('click', '.user-options-menu-btn', async function(e){
message: `<p>${i18n('confirm_open_apps_log_out')}</p>`,
buttons:[
{
label: i18n('close_all_Windows_and_log_out'),
label: i18n('close_all_windows_and_log_out'),
value: 'close_and_log_out',
type: 'primary',
},

View File

@ -783,7 +783,7 @@ function UIItem(options){
// Donwload
// -------------------------------------------
menu_items.push({
html: i18n('Download'),
html: i18n('download'),
onClick: async function(){
let items = [];
for (let index = 0; index < $selected_items.length; index++) {
@ -987,7 +987,7 @@ function UIItem(options){
)
){
const alert_resp = await UIAlert({
message: `${i18n('change_allways_open_with')} ` + html_encode(suggested_app.title) + '?',
message: `${i18n('change_always_open_with')} ` + html_encode(suggested_app.title) + '?',
body_icon: suggested_app.icon,
buttons:[
{
@ -1113,11 +1113,11 @@ function UIItem(options){
});
}
// -------------------------------------------
// Donwload
// Download
// -------------------------------------------
if(!is_trash && !is_trashed && (options.associated_app_name === null || options.associated_app_name === undefined)){
menu_items.push({
html: i18n('Download'),
html: i18n('download'),
disabled: options.is_dir && !window.feature_flags.download_directory,
onClick: async function(){
if(options.is_dir)

View File

@ -38,7 +38,7 @@ function UIPrompt(options){
if(!options.buttons || options.buttons.length === 0){
options.buttons = [
{label: i18n('cancel'), value: false, type: 'default'},
{label: i18n('OK'), value: true, type: 'primary'},
{label: i18n('ok'), value: true, type: 'primary'},
]
}
@ -53,7 +53,7 @@ function UIPrompt(options){
if(options.buttons && options.buttons.length > 0){
h += `<div style="overflow:hidden; margin-top:20px; float:right;">`;
h += `<button class="button button-default prompt-resp-button prompt-resp-btn-cancel" data-label="${i18n('cancel')}" style="padding: 0 20px;">${i18n('cancel')}</button>`;
h += `<button class="button button-primary prompt-resp-button prompt-resp-btn-ok" data-label="${i18n('OK')}" data-value="true" autofocus>${i18n('OK')}</button>`;
h += `<button class="button button-primary prompt-resp-button prompt-resp-btn-ok" data-label="${i18n('ok')}" data-value="true" autofocus>${i18n('ok')}</button>`;
h += `</div>`;
}

View File

@ -1956,7 +1956,7 @@ async function UIWindow(options) {
if (window.user.is_temp &&
!await UIWindowSaveAccount({
send_confirmation_code: true,
message: i18n('save_account_to_publish_website'),
message: i18n('save_account_to_publish'),
window_options: {
backdrop: true,
close_on_backdrop_click: false,

View File

@ -60,7 +60,7 @@ async function UIWindowFontPicker(options){
h += `</div>`;
// Select
h += `<button class="select-btn button button-primary button-block button-normal">${i18n('Select')}</button>`
h += `<button class="select-btn button button-primary button-block button-normal">${i18n('select')}</button>`
h += `</form>`;
h += `</div>`;
h += `</div>`;

View File

@ -41,7 +41,7 @@ async function UIWindowUploadProgress(options){
h += `</div>`;
const el_window = await UIWindow({
title: i18n('Upload'),
title: i18n('upload'),
icon: window.icons[`app-icon-uploader.svg`],
uid: null,
is_dir: false,

View File

@ -20,6 +20,18 @@ import translations from './translations/translations.js';
window.listSupportedLanguages = () => Object.keys(translations).map(lang => translations[lang]);
const variables = {
docs: "https://docs.puter.com/",
terms: "https://puter.com/terms",
privacy: "https://puter.com/privacy"
};
function ReplacePlaceholders(str) {
str = str.replace(/{{link=(.*?)}}(.*?){{\/link}}/g, (_, key, text) => `<a href="${variables[key]}" target="_blank">${text}</a>`);
str = str.replace(/{{(.*?)}}/g, (_, key) => variables[key]);
return str;
}
window.i18n = function (key, replacements = [], encode_html = true) {
if(Array.isArray(replacements) === false){
replacements = [replacements];
@ -31,6 +43,7 @@ window.i18n = function (key, replacements = [], encode_html = true) {
if (!str) {
str = key;
}
str = ReplacePlaceholders(str);
str = encode_html ? html_encode(str) : str;
// replace %% occurrences with the values in replacements
// %% is for simple text replacements

View File

@ -110,7 +110,7 @@ const ar = {
paste_into_folder: "الصق داخل الملف",
pick_name_for_website: "اختيار اسم لموقع الويب ",
picture: "صورة ",
powered_by_puter_js: `مشغل بواسطة<a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `مشغل بواسطة{{link=docs}}Puter.js{{/link}}`,
preparing: "إعداده",
preparing_for_upload: "التحضير للتحميل ",
properties: "ملكيات ",
@ -148,7 +148,7 @@ const ar = {
start: 'إبدأ ',
taking_longer_than_usual: 'يستغرق وقتا أطول من المعتاد ',
text_document: 'وثيقة نصية',
tos_fineprint: `بالنقر على "إنشاء حساب مجاني"، فإنك توافق على <a href="https://puter.com/terms" target="_blank">شروط الاستخدام</a> و <a href="https://puter.com/privacy" target="_blank">حماية البيانات</a>`,
tos_fineprint: `بالنقر على "إنشاء حساب مجاني"، فإنك توافق على {{link=terms}}شروط الاستخدام{{/link}} و {{link=privacy}}حماية البيانات{{/link}}`,
trash: 'نفاية',
type: 'اكتب',
undo: 'الغاء التحميل',

View File

@ -109,7 +109,7 @@ const bn = {
paste_into_folder: "ফোল্ডার আয়ে পেস্ট করুন",
pick_name_for_website: "আপনার ওয়েবসাইটের জন্য একটি নাম পছন্দ করুন:",
picture: "ছবি",
powered_by_puter_js: `দ্বারা চালিত <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `দ্বারা চালিত {{link=docs}}Puter.js{{/link}}`,
preparing: "প্রস্তুত হচ্ছে...",
preparing_for_upload: "আপলোডের জন্য প্রস্তুত হচ্ছে...",
properties: "বৈশিষ্ট্য",
@ -147,7 +147,7 @@ const bn = {
start: 'শুরু করুন',
taking_longer_than_usual: 'স্বাভাবিকের চেয়ে একটু বেশি সময় নিচ্ছে। অনুগ্রহপূর্বক অপেক্ষা করুন...',
text_document: 'পাঠ্য নথি',
tos_fineprint: `'ফ্রি অ্যাকাউন্ট তৈরি করুন'-এ ক্লিক করার মাধ্যমে আপনি Puter-এর <a href="https://puter.com/terms" target="_blank">পরিষেবার শর্তাবলী</a> এবং <a href="https://puter-এর সাথে সম্মত হন .com/privacy" target="_blank">গোপনীয়তা নীতি</a>`,
tos_fineprint: `'ফ্রি অ্যাকাউন্ট তৈরি করুন'-এ ক্লিক করার মাধ্যমে আপনি Puter-এর {{link=terms}}পরিষেবার শর্তাবলী{{/link}} এবং {{link=privacy}}গোপনীয়তা নীতি{{/link}}`,
trash: 'আবর্জনা',
type: 'টাইপ',
undo: 'পূর্বাবস্থায় ফেরান',

View File

@ -39,7 +39,7 @@ const br = {
change_username: "Alterar Nome de Utilizador",
close_all_windows: "Fechar Todas as Janelas",
close_all_windows_and_log_out: 'Fechar Janelas e Sair',
change_allways_open_with: "Quer sempre abrir arquivos deste tipo com",
change_always_open_with: "Quer sempre abrir arquivos deste tipo com",
color: 'Cor',
confirm_account_for_free_referral_storage_c2a: 'Crie uma conta e confirme o endereço do email para receber 1 GB de armazenamento gratuito. Seu amigo receberá 1 GB de armazenamento gratuito também.',
confirm_delete_multiple_items: 'Quer apagar estes itens permanentemente?',
@ -132,7 +132,7 @@ const br = {
pick_name_for_website: "Escolha um nome para seu site:",
picture: "Imagem",
plural_suffix: 's',
powered_by_puter_js: `Criado por <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Criado por {{link=docs}}Puter.js{{/link}}`,
preparing: "Preparando...",
preparing_for_upload: "Preparando para o envio...",
privacy: "Privacidade",
@ -184,7 +184,7 @@ const br = {
taking_longer_than_usual: 'Está a levar mais tempo que o usual. Por favor, aguarde...',
terms: "Termos",
text_document: 'Documento de Texto',
tos_fineprint: `Clicando em 'Criar Conta Gratuita' você concorda com os <a href="https://puter.com/terms" target="_blank">Termos de Serviço</a> e <a href="https://puter.com/privacy" target="_blank">Política de Privacidade</a> do Puter.`,
tos_fineprint: `Clicando em 'Criar Conta Gratuita' você concorda com os {{link=terms}}Termos de Serviço{{/link}} e {{link=privacy}}Política de Privacidade{{/link}} do Puter.`,
trash: 'Lixo',
type: 'Tipo',
type_confirm_to_delete_account: "Digite 'confirm' para excluir sua conta.",

View File

@ -110,7 +110,7 @@ const da = {
paste_into_folder: "Indsæt i mappe",
pick_name_for_website: "Vælg et navn til dit websted:",
picture: "Billede",
powered_by_puter_js: "Drevet af <a href=\"https://docs.puter.com/\" target=\"_blank\">Puter.js</a>",
powered_by_puter_js: "Drevet af {{link=docs}}Puter.js{{/link}}",
preparing: "Forbereder...",
preparing_for_upload: "Forbereder upload...",
properties: "Egenskaber",
@ -148,7 +148,7 @@ const da = {
start: "Start",
taking_longer_than_usual: "Dette tager længere tid end sædvanligt. Vent venligst...",
text_document: "Tekstdokument",
tos_fineprint: "Ved at klikke på 'Opret gratis konto' accepterer du Puters <a href=\"https://puter.com/terms\" target=\"_blank\">servicevilkår</a> og <a href=\"https://puter.com/privacy\" target=\"_blank\">privatlivspolitik</a>.",
tos_fineprint: "Ved at klikke på 'Opret gratis konto' accepterer du Puters {{link=terms}}servicevilkår{{/link}} og {{link=privacy}}privatlivspolitik{{/link}}.",
trash: "Papirkurv",
type: "Type",
undo: "Fortryd",

View File

@ -110,7 +110,7 @@ const de = {
paste_into_folder: "In Ordner einfügen",
pick_name_for_website: "Wählen Sie einen Namen für Ihre Webseite:",
picture: "Bild",
powered_by_puter_js: `Betrieben von <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Betrieben von {{link=docs}}Puter.js{{/link}}`,
preparing: "Bereitet vor...",
preparing_for_upload: "Bereitet für das Hochladen vor...",
properties: "Einstellungen",
@ -148,7 +148,7 @@ const de = {
start: 'Start',
taking_longer_than_usual: 'Dauert etwas länger als gewöhnlich. Bitte warten...',
text_document: 'Textdokument',
tos_fineprint: `Indem Sie auf „Kostenloses Konto erstellen“ klicken, stimmen Sie den <a href="https://puter.com/terms" target="_blank">Nutzungsbedingungen</a> und der <a href="https://puter.com/privacy" target="_blank">Datenschutzerklärung</a> von Puter zu.`,
tos_fineprint: `Indem Sie auf „Kostenloses Konto erstellen“ klicken, stimmen Sie den {{link=terms}}Nutzungsbedingungen{{/link}} und der {{link=privacy}}Datenschutzerklärung{{/link}} von Puter zu.`,
trash: 'Papierkorb',
type: 'Typ',
undo: 'Zurück',

View File

@ -34,12 +34,13 @@ const en = {
cancel: 'Cancel',
center: 'Center',
change_desktop_background: 'Change desktop background…',
change_email: "Change Email",
change_language: "Change Language",
change_password: "Change Password",
change_username: "Change Username",
close_all_windows: "Close All Windows",
close_all_windows_and_log_out: 'Close Windows and Log Out',
change_allways_open_with: "Do you want to always open this type of file with",
change_always_open_with: "Do you want to always open this type of file with",
color: 'Color',
confirm_account_for_free_referral_storage_c2a: 'Create an account and confirm your email address to receive 1 GB of free storage. Your friend will get 1 GB of free storage too.',
confirm_delete_multiple_items: 'Are you sure you want to permanently delete these items?',
@ -47,6 +48,7 @@ const en = {
confirm_open_apps_log_out: 'You have open apps. Are you sure you want to log out?',
confirm_new_password: "Confirm New Password",
confirm_delete_user: "Are you sure you want to delete your account? All your files and data will be permanently deleted. This action cannot be undone.",
confirm_delete_user_title: "Delete Account?",
contact_us: "Contact Us",
contain: 'Contain',
continue: "Continue",
@ -86,6 +88,7 @@ const en = {
feedback: "Feedback",
feedback_c2a: "Please use the form below to send us your feedback, comments, and bug reports.",
feedback_sent_confirmation: "Thank you for contacting us. If you have an email associated with your account, you will hear back from us as soon as possible.",
fit: "Fit",
forgot_pass_c2a: "Forgot password?",
from: "From",
general: "General",
@ -94,6 +97,7 @@ const en = {
hide_all_windows: "Hide All Windows",
html_document: 'HTML document',
image: 'Image',
incorrect_password: "Incorrect password",
invite_link: "Invite Link",
item: 'item',
items_in_trash_cannot_be_renamed: `This item can't be renamed because it's in the trash. To rename this item, first drag it out of the Trash.`,
@ -136,7 +140,7 @@ const en = {
pick_name_for_website: "Pick a name for your website:",
picture: "Picture",
plural_suffix: 's',
powered_by_puter_js: `Powered by <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Powered by {{link=docs}}Puter.js{{/link}}`,
preparing: "Preparing...",
preparing_for_upload: "Preparing for upload...",
privacy: "Privacy",
@ -189,7 +193,7 @@ const en = {
taking_longer_than_usual: 'Taking a little longer than usual. Please wait...',
terms: "Terms",
text_document: 'Text document',
tos_fineprint: `By clicking 'Create Free Account' you agree to Puter's <a href="https://puter.com/terms" target="_blank">Terms of Service</a> and <a href="https://puter.com/privacy" target="_blank">Privacy Policy</a>.`,
tos_fineprint: `By clicking 'Create Free Account' you agree to Puter's {{link=terms}}Terms of Service{{/link}} and {{link=privacy}}Privacy Policy{{/link}}.`,
trash: 'Trash',
type: 'Type',
type_confirm_to_delete_account: "Type 'confirm' to delete your account.",

View File

@ -110,7 +110,7 @@ const es = {
paste_into_folder: "Pegar en la Carpeta",
pick_name_for_website: "Escoge un nombre para tu página web:",
picture: "Imagen",
powered_by_puter_js: `Creado por <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Creado por {{link=docs}}Puter.js{{/link}}`,
preparing: "Preparando...",
preparing_for_upload: "Preparando para la subida...",
properties: "Propiedades",
@ -148,7 +148,7 @@ const es = {
start: 'Inicio',
taking_longer_than_usual: 'Tardando un poco más de lo usual. Por favor, espere...',
text_document: 'Documento de Texto',
tos_fineprint: `Pulsando sobre 'Crear una cuenta gratuita' aceptas los <a href="https://puter.com/terms" target="_blank"> términos del servicio</a> de Puter y <a href="https://puter.com/privacy" target="_blank">la política de privacidad</a>.`,
tos_fineprint: `Pulsando sobre 'Crear una cuenta gratuita' aceptas los {{link=terms}}términos del servicio{{/link}} de Puter y {{link=privacy}}la política de privacidad{{/link}}.`,
trash: 'Papelera',
type: 'Tipo',
undo: 'Deshacer',

View File

@ -111,7 +111,7 @@ const fa = {
paste_into_folder: "چسباندن در پوشه",
pick_name_for_website: "یک نام برای وبسایت خود انتخاب کنید:",
picture: "تصویر",
powered_by_puter_js: `پشتیبانی شده توسط <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `پشتیبانی شده توسط {{link=docs}}Puter.js{{/link}}`,
preparing: "در حال آماده سازی...",
preparing_for_upload: "آماده سازی برای بارگذاری...",
properties: "ویژگی ها",
@ -149,7 +149,7 @@ const fa = {
start: 'شروع',
taking_longer_than_usual: 'کمی بیشتر از معمول طول می کشد. لطفا صبر کنید...',
text_document: 'سند متنی',
tos_fineprint: `با کلیک بر روی 'ایجاد حساب کاربری رایگان' شما با <a href="https://puter.com/terms" target="_blank">شرایط خدمات</a> و <a href="https://puter.com/privacy" target="_blank">سیاست حفظ حریم خصوصی</a> Puter موافقت می کنید.`,
tos_fineprint: `با کلیک بر روی 'ایجاد حساب کاربری رایگان' شما با {{link=terms}}شرایط خدمات{{/link}} و {{link=privacy}}سیاست حفظ حریم خصوصی{{/link}} Puter موافقت می کنید.`,
trash: 'سطل زباله',
type: 'نوع',
undo: 'بازگشت',

View File

@ -154,7 +154,7 @@ const fi = {
paste_into_folder: "Liitä Kansioon",
pick_name_for_website: "Valitse nimi verkkosivustollesi:",
picture: "Kuva",
powered_by_puter_js: `Tämän Mahdollistaa <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Tämän Mahdollistaa {{link=docs}}Puter.js{{/link}}`,
preparing: "Valmistellaan...",
preparing_for_upload: "Valmistellaan latausta...",
properties: "Ominaisuudet",

View File

@ -109,7 +109,7 @@ const fr = {
paste_into_folder: "Coller dans le dossier",
pick_name_for_website: "Choisissez un nom pour votre site web :",
picture: "Image",
powered_by_puter_js: "Propulsé par <a href=\"https://docs.puter.com/\" target=\"_blank\">Puter.js</a>",
powered_by_puter_js: "Propulsé par {{link=docs}}Puter.js{{/link}}",
preparing: "Préparation en cours...",
preparing_for_upload: "Préparation du téléchargement en cours...",
properties: "Propriétés",
@ -147,7 +147,7 @@ const fr = {
start: "Démarrer",
taking_longer_than_usual: "Cela prend un peu plus de temps que d'habitude. Veuillez patienter...",
text_document: "Document texte",
tos_fineprint: "En cliquant sur 'Créer un compte gratuit', vous acceptez les <a href=\"https://puter.com/terms\" target=\"_blank\">Conditions d'utilisation</a> et la <a href=\"https://puter.com/privacy\" target=\"_blank\">Politique de confidentialité</a> de Puter.",
tos_fineprint: "En cliquant sur 'Créer un compte gratuit', vous acceptez les {{link=terms}}Conditions d'utilisation{{/link}} et la {{link=privacy}}Politique de confidentialité{{/link}} de Puter.",
trash: "Corbeille",
type: "Type",
undo: "Annuler",

View File

@ -110,7 +110,7 @@ const hy = {
paste_into_folder: "Տեղադրել պանակում",
pick_name_for_website: "Ընտրել անուն ձեր կայքի համար",
picture: "Նկար",
powered_by_puter_js: `Աջակցվում է <a href="https://docs.puter.com/" target="_blank">Puter.js</a>-ի կողմից`,
powered_by_puter_js: `Աջակցվում է {{link=docs}}Puter.js{{/link}}-ի կողմից`,
preparing: "Պատրաստվում է...",
preparing_for_upload: "Պատրաստվում է վերբեռնել...",
properties: "Հատկություններ",
@ -148,7 +148,7 @@ const hy = {
start: "Սկսել",
taking_longer_than_usual: "Սովորականից մի փոքր ավելի երկար է տևում: Խնդրում ենք սպասել...",
text_document: "Text նիշք",
tos_fineprint: `Սեղմելով «Ստեղծել անվճար հաշիվ»՝ դուք համաձայնում եք Փութերի <a href="https://puter.com/terms" target="_blank">ծառայությունների պայմաններին</a> և <a href="https://puter.com/privacy" target="_blank">գաղտնիության քաղաքականությանը</a>:`,
tos_fineprint: `Սեղմելով «Ստեղծել անվճար հաշիվ»՝ դուք համաձայնում եք Փութերի {{link=terms}}ծառայությունների պայմաններին{{/link}} և {{link=privacy}}գաղտնիության քաղաքականությանը{{/link}}:`,
trash: "Աղբաման",
type: "Տեսակ",
undo: "Հետարկել",

View File

@ -110,7 +110,7 @@ const it = {
paste_into_folder: "Incolla nella cartella",
pick_name_for_website: "Scegli un nome per il tuo sito web:",
picture: "Immagine",
powered_by_puter_js: `Powered by <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Powered by {{link=docs}}Puter.js{{/link}}`,
preparing: "Preparazione in corso...",
preparing_for_upload: "Preparazione per lupload...",
properties: "Proprietà",
@ -148,7 +148,7 @@ const it = {
start: 'Start',
taking_longer_than_usual: 'Il processo in corso ci sta mettendo più del solito. Attendere prego...',
text_document: 'Documento di testo',
tos_fineprint: `Cliccando su 'Crea un account gratis' accetti i <a href="https://puter.com/terms" target="_blank">Termini di Servizio</a> e l'<a href="https://puter.com/privacy" target="_blank">Informativa sulla Privacy</a> di Puter.`,
tos_fineprint: `Cliccando su 'Crea un account gratis' accetti i {{link=terms}}Termini di Servizio{{/link}} e l'{{link=privacy}}Informativa sulla Privacy{{/link}} di Puter.`,
trash: 'Cestino',
type: 'Tipo',
undo: 'Annulla',

View File

@ -110,7 +110,7 @@ const ko = {
paste_into_folder: "폴더에 붙여넣기",
pick_name_for_website: "웹사이트 이름을 선택하세요:",
picture: "사진",
powered_by_puter_js: `Powered by <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Powered by {{link=docs}}Puter.js{{/link}}`,
preparing: "준비 중...",
preparing_for_upload: "업로드 준비 중...",
properties: "속성",
@ -148,7 +148,7 @@ const ko = {
start: '시작',
taking_longer_than_usual: '보통보다 조금 더 오래 걸립니다. 잠시만 기다려 주십시오...',
text_document: '텍스트 문서',
tos_fineprint: `무료 계정 생성을 클릭하면 Puter의 <a href="https://puter.com/terms" target="_blank">서비스 약관</a>과 <a href="https://puter.com/privacy" target="_blank">개인정보 보호정책</a>에 동의하는 것입니다.`,
tos_fineprint: `무료 계정 생성을 클릭하면 Puter의 {{link=terms}}서비스 약관{{/link}}과 {{link=privacy}}개인정보 보호정책{{/link}}에 동의하는 것입니다.`,
trash: '휴지통',
type: '유형',
undo: '실행 취소',

View File

@ -36,7 +36,7 @@ const nb = {
change_username: "Endre brukernavn",
close_all_windows: "Lukk alle vinduer",
close_all_windows_and_log_out: 'Lukk alle vinduer og logg ut',
change_allways_open_with: "Ønsker du å alltid åpne denne filtypen med",
change_always_open_with: "Ønsker du å alltid åpne denne filtypen med",
color: "Farge",
confirm_account_for_free_referral_storage_c2a: "Opprett en konto og bekreft e-postadressen din for å motta 1 GB gratis lagringsplass. Din venn vil også få 1 GB gratis lagringsplass.",
confirm_delete_multiple_items: 'Er du sikker på at du vil slette disse elementene permanent?',
@ -120,7 +120,7 @@ const nb = {
paste_into_folder: "Lim inn i mappe",
pick_name_for_website: "Velg et navn for nettstedet ditt:",
picture: "Bilde",
powered_by_puter_js: "Drevet av <a href=\"https://docs.puter.com/\" target=\"_blank\">Puter.js</a>",
powered_by_puter_js: "Drevet av {{link=docs}}Puter.js{{/link}}",
preparing: "Forbereder...",
preparing_for_upload: "Forbereder opplasting...",
properties: "Egenskaper",
@ -166,7 +166,7 @@ const nb = {
start: "Start",
taking_longer_than_usual: "Dette tar litt lenger tid enn vanlig. Vennligst vent...",
text_document: "Tekstdokument",
tos_fineprint: "Ved å klikke på 'Opprett gratis konto' godtar du Puters <a href=\"https://puter.com/terms\" target=\"_blank\">tjenestevilkår</a> og <a href=\"https://puter.com/privacy\" target=\"_blank\">personvernpolicy</a>.",
tos_fineprint: "Ved å klikke på 'Opprett gratis konto' godtar du Puters {{link=terms}}tjenestevilkår{{/link}} og {{link=privacy}}personvernpolicy{{/link}}.",
trash: "Papirkurv",
type: "Type",
undo: "Angre",

View File

@ -39,7 +39,7 @@ const nl = {
change_username: "Gebruikersnaam veranderen",
close_all_windows: "Alle schermen sluiten",
close_all_windows_and_log_out: 'Alle schermen sluiten en afmelden',
change_allways_open_with: "Dit type bestand altijd openen met",
change_always_open_with: "Dit type bestand altijd openen met",
color: 'Kleur',
confirm_account_for_free_referral_storage_c2a: 'Maak een account en bevestig uw emailadres om 1 GB aan gratis opslag te ontvangen. Uw vriend krijgt ook 1 GB aan gratis opslag.',
confirm_delete_multiple_items: 'Weet u zeker dat u deze bestanden permanent wilt verwijderen?',
@ -132,7 +132,7 @@ const nl = {
paste_into_folder: "Plakken in Map",
pick_name_for_website: "Kies een naam voor uw website:",
picture: "Foto",
powered_by_puter_js: `Aangedreven door <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Aangedreven door {{link=docs}}Puter.js{{/link}}`,
preparing: "Voorbereiden...",
preparing_for_upload: "Upload voorbereiden...",
proceed_to_login: 'Doorgaan naar Inloggen',
@ -184,7 +184,7 @@ const nl = {
taking_longer_than_usual: 'Het duurt iets langer dan normaal. Even geduld aub...',
terms: "Voorwaarden",
text_document: 'Tekst document',
tos_fineprint: `Door te klikken op 'Maak Gratis Account' gaat u akkoord met Puter's <a href="https://puter.com/terms" target="_blank">Gebruiksvoorwaarden</a> en <a href="https://puter.com/privacy" target="_blank">Privacybeleid</a>.`,
tos_fineprint: `Door te klikken op 'Maak Gratis Account' gaat u akkoord met Puter's {{link=terms}}Gebruiksvoorwaarden{{/link}} en {{link=privacy}}Privacybeleid{{/link}}.`,
trash: 'Prullenbak',
type: 'Type',
type_confirm_to_delete_account: "Type 'bevestig' om uw account te verwijderen.",

View File

@ -110,7 +110,7 @@ const nn = {
paste_into_folder: "Lim inn i mappe",
pick_name_for_website: "Vel eit namn for nettstaden din:",
picture: "Bilete",
powered_by_puter_js: "Dreve av <a href=\"https://docs.puter.com/\" target=\"_blank\">Puter.js</a>",
powered_by_puter_js: "Dreve av {{link=docs}}Puter.js{{/link}}",
preparing: "Førebur…",
preparing_for_upload: "Førebur opplasting…",
properties: "Eigenskapar",
@ -148,7 +148,7 @@ const nn = {
start: "Start",
taking_longer_than_usual: "Dette tar litt lengre tid enn vanleg. Vennligst vent...",
text_document: "Tekstdokument",
tos_fineprint: "Ved å klikke på 'Opprett gratis konto' godtek du Puters <a href=\"https://puter.com/terms\" target=\"_blank\">tenestevilkår</a> og <a href=\"https://puter.com/privacy\" target=\"_blank\">personvernpolitikk</a>.",
tos_fineprint: "Ved å klikke på 'Opprett gratis konto' godtek du Puters {{link=terms}}tenestevilkår{{/link}} og {{link=privacy}}personvernpolitikk{{/link}}.",
trash: "Papirkorg",
type: "Type",
undo: "Angra",

View File

@ -36,7 +36,7 @@ const pl = {
change_username: "Zmień użytkownika",
close_all_windows: "Zamknij wszystkie okna",
close_all_windows_and_log_out: 'Zamknij wszystkie okna i wyloguj',
change_allways_open_with: "Czy chcesz zawsze otwierać ten typ pliku używając",
change_always_open_with: "Czy chcesz zawsze otwierać ten typ pliku używając",
color: 'Kolor',
confirm_account_for_free_referral_storage_c2a: 'Stwórz konto i potwierdź swój adres e-mail, żeby dostać 1 GB darmowego miejsca. Twój znajomy również dostanie 1 GB darmowego miejsca.',
confirm_delete_multiple_items: 'Czy na pewno chcesz na zawsze usunąć te przedmioty?',
@ -120,7 +120,7 @@ const pl = {
paste_into_folder: "Wklej do folderu",
pick_name_for_website: "Wybierz nazwę dla swojej strony:",
picture: "Obraz",
powered_by_puter_js: `Zasilane za pomocą <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Zasilane za pomocą {{link=docs}}Puter.js{{/link}}`,
preparing: "Przygotowywanie...",
preparing_for_upload: "Przygotowywanie do wgrania...",
proceed_to_login: 'Przejdź do logowania',
@ -166,7 +166,7 @@ const pl = {
start: 'Start',
taking_longer_than_usual: 'To trwa chwilę dłużej niż zwyklę. Prosimy poczekać...',
text_document: 'Dokument tekstowy',
tos_fineprint: `Klikając 'Stwórz darmowe konto' Zgadzasz się z <a href="https://puter.com/terms" target="_blank">Warunkami Obsługi</a> i <a href="https://puter.com/privacy" target="_blank">Polityką Prywatności</a>.`,
tos_fineprint: `Klikając 'Stwórz darmowe konto' Zgadzasz się z {{link=terms}}Warunkami Obsługi{{/link}} i {{link=privacy}}Polityką Prywatności{{/link}}.`,
trash: 'Kosz',
type: 'Wpisz',
undo: 'Cofnij',

View File

@ -23,183 +23,187 @@ const pt = {
dictionary: {
about: "Sobre",
account: "Conta",
access_granted_to: "Acesso garantido a",
add_existing_account: "Incluir Conta Existente",
access_granted_to: "Acesso Dado A",
add_existing_account: "Adicionar Conta Existente",
all_fields_required: 'Todos os campos são obrigatórios.',
apply: "Aplicar",
ascending: 'Ascendente',
auto_arrange: 'Auto Organizar',
background: "Fundo",
browse: "Pesquisar",
browse: "Explorar",
cancel: 'Cancelar',
center: 'Centrar',
change_desktop_background: 'Alterar o fundo…',
change_language: "Alterar a Língua",
change_password: "Alterar a Palavra Passe",
change_username: "Alterar Nome de Utilizador",
change_password: "Alterar a Password",
change_username: "Alterar o Nome de Utilizador",
close_all_windows: "Fechar Todas as Janelas",
close_all_windows_and_log_out: 'Fechar Janelas e Sair',
change_allways_open_with: "Quer sempre abrir ficheiros deste tipo com",
change_always_open_with: "Queres que ficheiros deste tipo abram sempre com",
color: 'Cor',
confirm_account_for_free_referral_storage_c2a: 'Cries uma conta e confirme o endereço do email para receber 1 GB de armazenamento gratuito. Vosso amigo receberá 1 GB de armazenamento gratuito também.',
confirm_delete_multiple_items: 'Queres apagar estes itens permanentemente?',
confirm_account_for_free_referral_storage_c2a: 'Cria uma conta e confirma o endereço do email para receber 1 GB de armazenamento gratuito. O teu amigo também receberá 1 GB de armazenamento gratuito.',
confirm_delete_multiple_items: 'Tens a certeza que queres apagar estes itens permanentemente?',
confirm_delete_single_item: 'Queres apagar este item permanentemente?',
confirm_open_apps_log_out: 'Possui aplicações abertas. Queres mesmo fechar vossa sessão?',
confirm_new_password: "Confirme vossa Nova Palavra Passe",
confirm_delete_user: "Queres excluir vossa conta? Todos os ficheiros e informações serão destruídas permanentemente. Esta operação não pode ser desfeita.",
contact_us: "Contacte-nos",
confirm_open_apps_log_out: 'Tens aplicações abertas. Queres mesmo terminar a sessão?',
confirm_new_password: "Confirma a Nova Password",
confirm_delete_user: "Tens a certeza que queres apagar a tua conta? Todos os ficheiros e dados serão apagados permanentemente. Esta operação é final.",
contact_us: "Contacta-nos",
contain: 'Contém',
continue: "Continua",
copy: 'Copia',
copy_link: "Copia Link",
copying: "Copiando",
copying: "A copiar",
cover: 'Capa',
create_account: "Criar Conta",
create_free_account: "Criar Conta Gratuita",
create_shortcut: "Criar Atalho",
credits: "Créditos",
current_password: "Palavra Passe Atual",
current_password: "Password Atual",
cut: 'Cortar',
clock: 'Relógio',
clock_visible_hide: 'Esconder - Sempre escondido',
clock_visible_show: 'Mostrar - Sempre visível',
clock_visible_auto: 'Auto - Por defeito, mostra apenas em modo full-screen',
date_modified: 'Data alterada',
delete: 'Excluir',
delete_account: "Excluir Conta",
delete_permanently: "Excluir Permanentemente",
delete: 'Apagar',
delete_account: "Apagar Conta",
delete_permanently: "Apagar Permanentemente",
deploy_as_app: 'Publicar como aplicativo',
descending: 'Descendente',
desktop_background_fit: "Caber",
developers: "Desenvolvedores",
dir_published_as_website: `%strong% foi publicado para:`,
desktop_background_fit: "Ajustado",
developers: "Developers",
dir_published_as_website: `%strong% foi publicado em:`,
disassociate_dir: "Desassociar Diretório",
download: 'Descarregar',
download_file: 'Descarregar Ficheiro',
downloading: "Efectuando a Descarga",
downloading: "Fazendo a descarga",
email: "Email",
email_or_username: "Email ou Nome de Utilizador",
empty_trash: 'Esvaziar Lixo',
empty_trash_confirmation: `Queres apagar os itens do Lixo permanentemente?`,
emptying_trash: 'Deitando o Lixo fora…',
enter_password_to_confirm_delete_user: "Entre vossa palavra passe para confirmar a exclusão da conta",
enter_password_to_confirm_delete_user: "Insere a Password para confirmar a remoção da conta",
feedback: "Feedback",
feedback_c2a: "Favor usares o formulário abaixo para enviar vossos comentários e comunicados.",
feedback_sent_confirmation: "Obrigado por contactar-nos. Se tiveres email associado a esta conta, esperamos ver-nos novamente em breve.",
feedback_c2a: "Pff usa o formulário abaixo para enviar feedback, comentários e bugs.",
feedback_sent_confirmation: "Obrigado por nos contactares. Se tiveres um email associado a esta conta, receberás notícias o mais brevemente que nos seja possível.",
forgot_pass_c2a: "Esqueceste a senha?",
from: "De",
general: "Geral",
get_a_copy_of_on_puter: `Obter uma cópia de '%%' no Puter.com!`,
get_a_copy_of_on_puter: `Obter uma cópia de '%%' em Puter.com!`,
get_copy_link: 'Copiar Link',
hide_all_windows: "Ocultar Todas as Janelas",
html_document: 'Documento HTML',
image: 'Imagem',
invite_link: "Link do Convite",
item: 'item',
items_in_trash_cannot_be_renamed: `Item não pode ser renomeado porque está no lixo. Para renomear, arraste-o para fora do Lixo.`,
items_in_trash_cannot_be_renamed: `Este item não pode ser renomeado porque está no lixo. Para alterar o nome, primeiro arrasta-o para fora do Lixo.`,
jpeg_image: 'Imagem JPEG',
keep_in_taskbar: 'Armazenar na Barra de Tarefas',
keep_in_taskbar: 'Manter na Barra de Tarefas',
language: "Língua",
license: "Licença",
loading: 'Carregando',
log_in: "Entrar",
log_into_another_account_anyway: 'Entrar com outra conta de qualquer maneira',
log_into_another_account_anyway: 'Entrar com outra conta na mesma',
log_out: 'Sair',
move: 'Mover',
moving: "Movendo",
my_websites: "Meus Sites",
name: 'Nome',
name_cannot_be_empty: 'Nome não pode ser vazio.',
name_cannot_contain_double_period: "Nome não pode conter o caracters '..'.",
name_cannot_contain_period: "Nome não pode conter o caracter '.'.",
name_cannot_contain_slash: "Nome não pode conter o caracter '/'.",
name_cannot_contain_double_period: "Nome não pode conter o caractere '..'.",
name_cannot_contain_period: "Nome não pode conter o caractere '.'.",
name_cannot_contain_slash: "Nome não pode conter o caractere '/'.",
name_must_be_string: "Nome tem que ser apenas texto.",
name_too_long: `Nome não pode ter mais que %% characters.`,
name_too_long: `Nome não pode ter mais que %% caracteres.`,
new: 'Novo',
new_folder: 'Nova Pasta',
new_password: "Nova Senha",
new_username: "Novo Utilizador",
new_password: "Nova Password",
new_username: "Novo Nome de Utilizador",
no: 'Não',
no_dir_associated_with_site: 'Não existe diretório associado com este endereço.',
no_websites_published: "Ainda não publicaste sites.",
no_websites_published: "Ainda não tens sites publicados.",
ok: 'OK',
open: "Abrir",
open_in_new_tab: "Abrir em Nova Aba",
open_in_new_window: "Abrir em Nova Janela",
open_with: "Abrir Com",
oss_code_and_content: "Software de Código Aberto",
password: "Palavra Passe",
password_changed: "Palavra Passe alterada.",
passwords_do_not_match: '`Nova Palavra Passe` e `Confirmação de Nova Palavra Passe` não conferem como idênticas.',
password: "Password",
password_changed: "Password alterada.",
passwords_do_not_match: '`Nova Password` e `Confirmação de Nova Password` são diferentes.',
paste: 'Colar',
paste_into_folder: "Cole na Pasta",
paste_into_folder: "Cola na Pasta",
pick_name_for_website: "Escolha um nome para seu site:",
picture: "Imagem",
plural_suffix: 's',
powered_by_puter_js: `Criado por <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Criado com {{link=docs}}Puter.js{{/link}}`,
preparing: "A preparar...",
preparing_for_upload: "A preparar o envio...",
preparing_for_upload: "A preparar o upload...",
privacy: "Privacidade",
proceed_to_login: 'Proceguir para a entrada',
proceed_with_account_deletion: "Prosseguir com Exclusão da Conta",
proceed_to_login: 'Prosseguir para o login',
proceed_with_account_deletion: "Prosseguir com Remoção da Conta",
properties: "Propriedades",
publish: "Publicar",
publish_as_website: 'Publicar como Site',
puter_description: `Puter é uma nuvem pessoal que prioriza a privacidade para manter todos os seus ficheiros, aplicativos e jogos em um local seguro, acessível de qualquer lugar e a qualquer hora.`,
puter_description: `Puter é uma nuvem pessoal que prioriza a privacidade e que mantém todos os teus ficheiros, aplicativos e jogos num local seguro, acessível de qualquer lugar e a qualquer hora.`,
recent: "Recentes",
recover_password: "Recuperar Senha",
refer_friends_c2a: "Obtenhas 1 GB para cada amigo que criar e confirmar uma conta no Puter. Vosso amigo ganhará 1 GB também!",
refer_friends_social_media_c2a: `Obtenhas 1 GB de armazenamento gratuito no Puter.com!`,
recover_password: "Recuperar Password",
refer_friends_c2a: "Ganha 1 GB por cada amigo que criar e confirmar uma conta Puter. Os teus amigos também ganham 1 GB!",
refer_friends_social_media_c2a: `Ganha 1 GB de armazenamento gratuito em Puter.com!`,
refresh: 'Atualizar',
release_address_confirmation: `Desejas liberar este endereço?`,
release_address_confirmation: `Queres libertar este endereço?`,
remove_from_taskbar:'Remover da Barra de Tarefas',
rename: 'Renomear',
repeat: 'Repetir',
replace: 'Substituir',
replace_all: 'Substituir Todas',
replace_all: 'Substituir Todos',
resend_confirmation_code: "Re-enviar o Código de Confirmação",
restore: "Restaurar",
save_account: 'Gravar conta',
save_account_to_get_copy_link: "Favor criares uma conta para prosseguir.",
save_account_to_publish: 'Favor criares uma conta para prosseguir.',
save_account_to_get_copy_link: "Para continuar, pff cria uma conta.",
save_account_to_publish: 'Para continuar, pff cria uma conta.',
save_session: 'Gravar sessão',
save_session_c2a: 'Crie uma conta para gravares a sessão atual e evitar a perda de vosso trabalho.',
scan_qr_c2a: 'Escaneie o código abaixo para entrares nesta sessão com outros dispositivos',
save_session_c2a: 'Cria uma conta para gravares a sessão atual e evitares perder o teu trabalho.',
scan_qr_c2a: 'Digitaliza o código abaixo para entrares nesta sessão com outros dispositivos',
select: "Selecionar",
selected: 'selecionado',
select_color: 'Selecionar cor…',
send: "Enviar",
send_password_recovery_email: "Enviar Email de Recuperação de Senha",
send_password_recovery_email: "Enviar Email de Recuperação de Password",
session_saved: "Obrigado por criares uma conta. Esta sessão foi gravada.",
settings: "Configurações",
set_new_password: "Informar Palavra Passe",
settings: "Definições",
set_new_password: "Definir nova Password",
share_to: "Partilhar com",
show_all_windows: "Exibir Todas as Janelas",
show_hidden: 'Exibir oculto',
sign_in_with_puter: "Entrar no Puter",
show_all_windows: "Mostrar Todas as Janelas",
show_hidden: 'Exibir janelas ocultas',
sign_in_with_puter: "Entrar em Puter",
sign_up: "Registar",
signing_in: "Entrar…",
size: 'Tamanho',
skip: 'Pular',
sort_by: 'Organizar por',
start: 'Início',
skip: 'Passar à frente',
sort_by: 'Ordenar por',
start: 'Iniciar',
status: "Status",
storage_usage: "Uso do Armazenamento",
taking_longer_than_usual: 'Está a levar mais tempo que o usual. Por favor, aguarde...',
taking_longer_than_usual: 'Está a levar mais tempo que o usual. Por favor aguarda...',
terms: "Termos",
text_document: 'Documento de Texto',
tos_fineprint: `Ao clicar em 'Criar Conta Gratuita' concordas com os <a href="https://puter.com/terms" target="_blank">Termos de Serviço</a> e <a href="https://puter.com/privacy" target="_blank">Política de Privacidade</a> do Puter.`,
tos_fineprint: `Ao clicares em 'Criar Conta Gratuita' concordas com os {{link=terms}}Termos de Serviço{{/link}} e {{link=privacy}}Política de Privacidade{{/link}} do Puter.`,
trash: 'Lixo',
type: 'Tipo',
type_confirm_to_delete_account: "Digite 'confirm' para excluires vossa conta.",
undo: 'Desfazer',
type_confirm_to_delete_account: "Escreve 'confirm' para apagares esta conta.",
undo: 'Voltar atrás',
unlimited: 'Ilimitado',
unzip: "Deszipar",
upload: 'Enviar',
upload_here: 'Enviar aqui',
usage: 'Uso',
unzip: "Abrir zip",
upload: 'Carregar',
upload_here: 'Carregar para aqui',
usage: 'Utilização',
username: "Nome de Utilizador",
username_changed: 'Nome de Utilizador atualizado com sucesso.',
username_changed: 'Nome de Utilizador atualizado.',
versions: "Versões",
yes: 'Sim',
yes_release_it: 'Sim, Libere Isto',
you_have_been_referred_to_puter_by_a_friend: "Indicaste o Puter a um amigo!",
yes_release_it: 'Sim, libertar',
you_have_been_referred_to_puter_by_a_friend: "Um amigo teu recomendou-te a Puter.com!",
zip: "Zipar",
}
};

View File

@ -109,7 +109,7 @@ const ro = {
paste_into_folder: "Inserează in folder",
pick_name_for_website: "Alegeți un nume pentru site-ul dvs:",
picture: "Poza",
powered_by_puter_js: `Creat de <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `Creat de {{link=docs}}Puter.js{{/link}}`,
preparing: "Preparare...",
preparing_for_upload: "Preparare pentru încărcare...",
properties: "Proprietăți",
@ -147,7 +147,7 @@ const ro = {
start: 'Start',
taking_longer_than_usual: 'Durează puțin mai mult decât de obicei. Vă rugăm așteptați...',
text_document: 'Document Text',
tos_fineprint: `Făcând clic pe „Creați un cont gratuit”, sunteți de acord cu <a href="https://puter.com/terms" target="_blank">Termenii si conditiile</a> si <a href="https://puter.com/privacy" target="_blank">Politia de Confidentialitate Puter.com</a>.`,
tos_fineprint: `Făcând clic pe „Creați un cont gratuit”, sunteți de acord cu {{link=terms}}Termenii si conditiile{{/link}} si {{link=privacy}}Politia de Confidentialitate Puter.com{{/link}}.`,
trash: 'Coș de gunoi',
type: 'Type',
undo: 'Undo',

View File

@ -110,7 +110,7 @@ const sv = {
paste_into_folder: "Klistra in i mapp",
pick_name_for_website: "Välj ett namn för din webbplats:",
picture: "Bild",
powered_by_puter_js: "Drivs av <a href=\"https://docs.puter.com/\" target=\"_blank\">Puter.js</a>",
powered_by_puter_js: "Drivs av {{link=docs}}Puter.js{{/link}}",
preparing: "Förbereder...",
preparing_for_upload: "Förbereder för uppladdning...",
properties: "Egenskaper",
@ -148,7 +148,7 @@ const sv = {
start: "Start",
taking_longer_than_usual: "Detta tar längre tid än vanligt. Vänligen vänta...",
text_document: "Textdokument",
tos_fineprint: "Genom att klicka på 'Skapa gratis konto' godkänner du Puters <a href=\"https://puter.com/terms\" target=\"_blank\">användarvillkor</a> och <a href=\"https://puter.com/privacy\" target=\"_blank\">integritetspolicy</a>.",
tos_fineprint: "Genom att klicka på 'Skapa gratis konto' godkänner du Puters {{link=terms}}användarvillkor{{/link}} och {{link=privacy}}integritetspolicy{{/link}}.",
trash: "Papperskorg",
type: "Typ",
undo: "Ångra",

View File

@ -119,7 +119,7 @@ const th = {
paste_into_folder: "วางลงในโฟลเดอร์",
pick_name_for_website: "เลือกชื่อสำหรับเว็บไซต์ของคุณ:",
picture: "รูปภาพ",
powered_by_puter_js: `สนับสนุนโดย <a href="https://docs.puter.com/" target="_blank">Puter.js</a>`,
powered_by_puter_js: `สนับสนุนโดย {{link=docs}}Puter.js{{/link}}`,
preparing: "กำลังเตรียม...",
preparing_for_upload: "กำลังเตรียมสำหรับอัปโหลด...",
proceed_to_login: "ดำเนินการเข้าสู่ระบบ",
@ -165,7 +165,7 @@ const th = {
start: "เริ่มต้น",
taking_longer_than_usual: "ใช้เวลานานกว่าปกติเล็กน้อย กรุณารอสักครู่...",
text_document: "เอกสารข้อความ",
tos_fineprint: `การคลิก 'สร้างบัญชีฟรี' หมายความว่าคุณยอมรับ <a href="https://puter.com/terms" target="_blank">ข้อกำหนดการให้บริการ</a> และ <a href="https://puter.com/privacy" target="_blank">นโยบายความเป็นส่วนตัว</a>.`,
tos_fineprint: `การคลิก 'สร้างบัญชีฟรี' หมายความว่าคุณยอมรับ {{link=terms}}ข้อกำหนดการให้บริการ{{/link}} และ {{link=privacy}}นโยบายความเป็นส่วนตัว{{/link}}.`,
trash: "ถังขยะ",
type: "ประเภท",
undo: "เลิกทำ",

View File

@ -110,7 +110,7 @@ const ur = {
paste_into_folder: "فولڈر میں چسپاں کریں",
pick_name_for_website: "ویب سائٹ کے لئے نام منتخب کریں ",
picture: "تصویر ",
powered_by_puter_js: 'پیوٹر جے ایس کے زریعے محرک<a href="https://docs.puter.com/" target="_blank">Puter.js</a>',
powered_by_puter_js: 'پیوٹر جے ایس کے زریعے محرک{{link=docs}}Puter.js{{/link}}',
preparing: "تیاری ",
preparing_for_upload: "اپلوڈ کے لئے تیاری ",
properties: "خصوصیات ",

View File

@ -114,7 +114,7 @@ const zh = {
paste_into_folder: "粘贴到文件夹",
pick_name_for_website: "为您的网站选择一个名称:",
picture: "图片",
powered_by_puter_js: `<a href="https://docs.puter.com/" target="_blank">Puter.js</a> 提供支持`,
powered_by_puter_js: `{{link=docs}}Puter.js{{/link}} 提供支持`,
preparing: "准备中...",
preparing_for_upload: "准备上传...",
properties: "属性",
@ -152,7 +152,7 @@ const zh = {
start: '开始',
taking_longer_than_usual: '需要的时间比平时长一点。请稍等...',
text_document: '文本文档',
tos_fineprint: `点击“创建免费帐户”即表示您同意 Puter 的 <a href="https://puter.com/terms" target="_blank">服务条款</a> 和 <a href="https://puter.com/privacy" target="_blank">隐私政策</a>`,
tos_fineprint: `点击“创建免费帐户”即表示您同意 Puter 的 {{link=terms}}服务条款{{/link}} 和 {{link=privacy}}隐私政策{{/link}}`,
trash: '回收站',
type: '类型',
undo: '撤销',

View File

@ -53,6 +53,7 @@ async function checkTranslationRegistrations() {
}
}
// Ensure that translations only contain keys that exist in the en dictionary
function checkTranslationKeys() {
const enDictionary = translations.en.dictionary;
@ -71,8 +72,40 @@ function checkTranslationKeys() {
}
}
// Ensure that all keys passed to i18n() exist in the en dictionary
async function checkTranslationUsage() {
const enDictionary = translations.en.dictionary;
const sourceDirectories = [
'./src/helpers',
'./src/UI',
];
// Looks for i18n() calls using either ' or " for the key string.
// The key itself is at index 2 of the result.
const i18nRegex = /i18n\((['"])(.*?)\1\)/g;
for (const dir of sourceDirectories) {
const files = await fs.promises.readdir(dir, { recursive: true });
for (const relativeFileName of files) {
if (!relativeFileName.endsWith('.js')) continue;
const fileName = `${dir}/${relativeFileName}`;
const fileContents = await fs.promises.readFile(fileName, { encoding: 'utf8' });
const i18nUses = fileContents.matchAll(i18nRegex);
for (const use of i18nUses) {
const key = use[2];
if (!enDictionary.hasOwnProperty(key)) {
reportError(`Unrecognized i18n key: call ${use[0]} in ${fileName}`);
}
}
}
}
}
await checkTranslationRegistrations();
checkTranslationKeys();
await checkTranslationUsage();
if (hadError) {
process.stdout.write('Errors were found in translation files.\n');