mirror of
				https://github.com/NginxProxyManager/nginx-proxy-manager.git
				synced 2025-11-04 01:15:14 +00:00 
			
		
		
		
	Merge remote-tracking branch 'upstream/develop' into develop
This commit is contained in:
		@@ -81,7 +81,7 @@ const internalAccessList = {
 | 
			
		||||
 | 
			
		||||
				return internalAccessList.build(row)
 | 
			
		||||
					.then(() => {
 | 
			
		||||
						if (row.proxy_host_count) {
 | 
			
		||||
						if (parseInt(row.proxy_host_count, 10)) {
 | 
			
		||||
							return internalNginx.bulkGenerateConfigs('proxy_host', row.proxy_hosts);
 | 
			
		||||
						}
 | 
			
		||||
					})
 | 
			
		||||
@@ -223,7 +223,7 @@ const internalAccessList = {
 | 
			
		||||
			.then((row) => {
 | 
			
		||||
				return internalAccessList.build(row)
 | 
			
		||||
					.then(() => {
 | 
			
		||||
						if (row.proxy_host_count) {
 | 
			
		||||
						if (parseInt(row.proxy_host_count, 10)) {
 | 
			
		||||
							return internalNginx.bulkGenerateConfigs('proxy_host', row.proxy_hosts);
 | 
			
		||||
						}
 | 
			
		||||
					}).then(internalNginx.reload)
 | 
			
		||||
@@ -252,9 +252,13 @@ const internalAccessList = {
 | 
			
		||||
				let query = accessListModel
 | 
			
		||||
					.query()
 | 
			
		||||
					.select('access_list.*', accessListModel.raw('COUNT(proxy_host.id) as proxy_host_count'))
 | 
			
		||||
					.joinRaw('LEFT JOIN `proxy_host` ON `proxy_host`.`access_list_id` = `access_list`.`id` AND `proxy_host`.`is_deleted` = 0')
 | 
			
		||||
					.leftJoin('proxy_host', function() {
 | 
			
		||||
						this.on('proxy_host.access_list_id', '=', 'access_list.id')
 | 
			
		||||
							.andOn('proxy_host.is_deleted', '=', 0);
 | 
			
		||||
					})
 | 
			
		||||
					.where('access_list.is_deleted', 0)
 | 
			
		||||
					.andWhere('access_list.id', data.id)
 | 
			
		||||
					.groupBy('access_list.id')
 | 
			
		||||
					.allowGraph('[owner,items,clients,proxy_hosts.[certificate,access_list.[clients,items]]]')
 | 
			
		||||
					.first();
 | 
			
		||||
 | 
			
		||||
@@ -373,7 +377,10 @@ const internalAccessList = {
 | 
			
		||||
				let query = accessListModel
 | 
			
		||||
					.query()
 | 
			
		||||
					.select('access_list.*', accessListModel.raw('COUNT(proxy_host.id) as proxy_host_count'))
 | 
			
		||||
					.joinRaw('LEFT JOIN `proxy_host` ON `proxy_host`.`access_list_id` = `access_list`.`id` AND `proxy_host`.`is_deleted` = 0')
 | 
			
		||||
					.leftJoin('proxy_host', function() {
 | 
			
		||||
						this.on('proxy_host.access_list_id', '=', 'access_list.id')
 | 
			
		||||
							.andOn('proxy_host.is_deleted', '=', 0);
 | 
			
		||||
					})
 | 
			
		||||
					.where('access_list.is_deleted', 0)
 | 
			
		||||
					.groupBy('access_list.id')
 | 
			
		||||
					.allowGraph('[owner,items,clients]')
 | 
			
		||||
@@ -501,8 +508,13 @@ const internalAccessList = {
 | 
			
		||||
								if (typeof item.password !== 'undefined' && item.password.length) {
 | 
			
		||||
									logger.info('Adding: ' + item.username);
 | 
			
		||||
 | 
			
		||||
									utils.execFile('/usr/bin/htpasswd', ['-b', htpasswd_file, item.username, item.password])
 | 
			
		||||
										.then((/*result*/) => {
 | 
			
		||||
									utils.execFile('openssl', ['passwd', '-apr1', item.password])
 | 
			
		||||
										.then((res) => {
 | 
			
		||||
											try {
 | 
			
		||||
												fs.appendFileSync(htpasswd_file, item.username + ':' + res + '\n', {encoding: 'utf8'});
 | 
			
		||||
											} catch (err) {
 | 
			
		||||
												reject(err);
 | 
			
		||||
											}
 | 
			
		||||
											next();
 | 
			
		||||
										})
 | 
			
		||||
										.catch((err) => {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,6 @@
 | 
			
		||||
const error         = require('../lib/error');
 | 
			
		||||
const auditLogModel = require('../models/audit-log');
 | 
			
		||||
const error            = require('../lib/error');
 | 
			
		||||
const auditLogModel    = require('../models/audit-log');
 | 
			
		||||
const {castJsonIfNeed} = require('../lib/helpers');
 | 
			
		||||
 | 
			
		||||
const internalAuditLog = {
 | 
			
		||||
 | 
			
		||||
@@ -22,9 +23,9 @@ const internalAuditLog = {
 | 
			
		||||
					.allowGraph('[user]');
 | 
			
		||||
 | 
			
		||||
				// Query is used for searching
 | 
			
		||||
				if (typeof search_query === 'string') {
 | 
			
		||||
				if (typeof search_query === 'string' && search_query.length > 0) {
 | 
			
		||||
					query.where(function () {
 | 
			
		||||
						this.where('meta', 'like', '%' + search_query + '%');
 | 
			
		||||
						this.where(castJsonIfNeed('meta'), 'like', '%' + search_query + '%');
 | 
			
		||||
					});
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -313,6 +313,9 @@ const internalCertificate = {
 | 
			
		||||
					.where('is_deleted', 0)
 | 
			
		||||
					.andWhere('id', data.id)
 | 
			
		||||
					.allowGraph('[owner]')
 | 
			
		||||
					.allowGraph('[proxy_hosts]')
 | 
			
		||||
					.allowGraph('[redirection_hosts]')
 | 
			
		||||
					.allowGraph('[dead_hosts]')
 | 
			
		||||
					.first();
 | 
			
		||||
 | 
			
		||||
				if (access_data.permission_visibility !== 'all') {
 | 
			
		||||
@@ -464,6 +467,9 @@ const internalCertificate = {
 | 
			
		||||
					.where('is_deleted', 0)
 | 
			
		||||
					.groupBy('id')
 | 
			
		||||
					.allowGraph('[owner]')
 | 
			
		||||
					.allowGraph('[proxy_hosts]')
 | 
			
		||||
					.allowGraph('[redirection_hosts]')
 | 
			
		||||
					.allowGraph('[dead_hosts]')
 | 
			
		||||
					.orderBy('nice_name', 'ASC');
 | 
			
		||||
 | 
			
		||||
				if (access_data.permission_visibility !== 'all') {
 | 
			
		||||
 
 | 
			
		||||
@@ -6,6 +6,7 @@ const internalHost        = require('./host');
 | 
			
		||||
const internalNginx       = require('./nginx');
 | 
			
		||||
const internalAuditLog    = require('./audit-log');
 | 
			
		||||
const internalCertificate = require('./certificate');
 | 
			
		||||
const {castJsonIfNeed}    = require('../lib/helpers');
 | 
			
		||||
 | 
			
		||||
function omissions () {
 | 
			
		||||
	return ['is_deleted'];
 | 
			
		||||
@@ -409,16 +410,16 @@ const internalDeadHost = {
 | 
			
		||||
					.where('is_deleted', 0)
 | 
			
		||||
					.groupBy('id')
 | 
			
		||||
					.allowGraph('[owner,certificate]')
 | 
			
		||||
					.orderBy('domain_names', 'ASC');
 | 
			
		||||
					.orderBy(castJsonIfNeed('domain_names'), 'ASC');
 | 
			
		||||
 | 
			
		||||
				if (access_data.permission_visibility !== 'all') {
 | 
			
		||||
					query.andWhere('owner_user_id', access.token.getUserId(1));
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				// Query is used for searching
 | 
			
		||||
				if (typeof search_query === 'string') {
 | 
			
		||||
				if (typeof search_query === 'string' && search_query.length > 0) {
 | 
			
		||||
					query.where(function () {
 | 
			
		||||
						this.where('domain_names', 'like', '%' + search_query + '%');
 | 
			
		||||
						this.where(castJsonIfNeed('domain_names'), 'like', '%' + search_query + '%');
 | 
			
		||||
					});
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -2,6 +2,7 @@ const _                    = require('lodash');
 | 
			
		||||
const proxyHostModel       = require('../models/proxy_host');
 | 
			
		||||
const redirectionHostModel = require('../models/redirection_host');
 | 
			
		||||
const deadHostModel        = require('../models/dead_host');
 | 
			
		||||
const {castJsonIfNeed}     = require('../lib/helpers');
 | 
			
		||||
 | 
			
		||||
const internalHost = {
 | 
			
		||||
 | 
			
		||||
@@ -17,7 +18,7 @@ const internalHost = {
 | 
			
		||||
	cleanSslHstsData: function (data, existing_data) {
 | 
			
		||||
		existing_data = existing_data === undefined ? {} : existing_data;
 | 
			
		||||
 | 
			
		||||
		let combined_data = _.assign({}, existing_data, data);
 | 
			
		||||
		const combined_data = _.assign({}, existing_data, data);
 | 
			
		||||
 | 
			
		||||
		if (!combined_data.certificate_id) {
 | 
			
		||||
			combined_data.ssl_forced    = false;
 | 
			
		||||
@@ -73,7 +74,7 @@ const internalHost = {
 | 
			
		||||
	 * @returns {Promise}
 | 
			
		||||
	 */
 | 
			
		||||
	getHostsWithDomains: function (domain_names) {
 | 
			
		||||
		let promises = [
 | 
			
		||||
		const promises = [
 | 
			
		||||
			proxyHostModel
 | 
			
		||||
				.query()
 | 
			
		||||
				.where('is_deleted', 0),
 | 
			
		||||
@@ -125,19 +126,19 @@ const internalHost = {
 | 
			
		||||
	 * @returns {Promise}
 | 
			
		||||
	 */
 | 
			
		||||
	isHostnameTaken: function (hostname, ignore_type, ignore_id) {
 | 
			
		||||
		let promises = [
 | 
			
		||||
		const promises = [
 | 
			
		||||
			proxyHostModel
 | 
			
		||||
				.query()
 | 
			
		||||
				.where('is_deleted', 0)
 | 
			
		||||
				.andWhere('domain_names', 'like', '%' + hostname + '%'),
 | 
			
		||||
				.andWhere(castJsonIfNeed('domain_names'), 'like', '%' + hostname + '%'),
 | 
			
		||||
			redirectionHostModel
 | 
			
		||||
				.query()
 | 
			
		||||
				.where('is_deleted', 0)
 | 
			
		||||
				.andWhere('domain_names', 'like', '%' + hostname + '%'),
 | 
			
		||||
				.andWhere(castJsonIfNeed('domain_names'), 'like', '%' + hostname + '%'),
 | 
			
		||||
			deadHostModel
 | 
			
		||||
				.query()
 | 
			
		||||
				.where('is_deleted', 0)
 | 
			
		||||
				.andWhere('domain_names', 'like', '%' + hostname + '%')
 | 
			
		||||
				.andWhere(castJsonIfNeed('domain_names'), 'like', '%' + hostname + '%')
 | 
			
		||||
		];
 | 
			
		||||
 | 
			
		||||
		return Promise.all(promises)
 | 
			
		||||
 
 | 
			
		||||
@@ -6,6 +6,7 @@ const internalHost        = require('./host');
 | 
			
		||||
const internalNginx       = require('./nginx');
 | 
			
		||||
const internalAuditLog    = require('./audit-log');
 | 
			
		||||
const internalCertificate = require('./certificate');
 | 
			
		||||
const {castJsonIfNeed}    = require('../lib/helpers');
 | 
			
		||||
 | 
			
		||||
function omissions () {
 | 
			
		||||
	return ['is_deleted', 'owner.is_deleted'];
 | 
			
		||||
@@ -449,16 +450,16 @@ const internalProxyHost = {
 | 
			
		||||
					.where('is_deleted', 0)
 | 
			
		||||
					.groupBy('id')
 | 
			
		||||
					.allowGraph('[owner,access_list,certificate]')
 | 
			
		||||
					.orderBy('domain_names', 'ASC');
 | 
			
		||||
					.orderBy(castJsonIfNeed('domain_names'), 'ASC');
 | 
			
		||||
 | 
			
		||||
				if (access_data.permission_visibility !== 'all') {
 | 
			
		||||
					query.andWhere('owner_user_id', access.token.getUserId(1));
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				// Query is used for searching
 | 
			
		||||
				if (typeof search_query === 'string') {
 | 
			
		||||
				if (typeof search_query === 'string' && search_query.length > 0) {
 | 
			
		||||
					query.where(function () {
 | 
			
		||||
						this.where('domain_names', 'like', '%' + search_query + '%');
 | 
			
		||||
						this.where(castJsonIfNeed('domain_names'), 'like', `%${search_query}%`);
 | 
			
		||||
					});
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -6,6 +6,7 @@ const internalHost         = require('./host');
 | 
			
		||||
const internalNginx        = require('./nginx');
 | 
			
		||||
const internalAuditLog     = require('./audit-log');
 | 
			
		||||
const internalCertificate  = require('./certificate');
 | 
			
		||||
const {castJsonIfNeed}     = require('../lib/helpers');
 | 
			
		||||
 | 
			
		||||
function omissions () {
 | 
			
		||||
	return ['is_deleted'];
 | 
			
		||||
@@ -409,16 +410,16 @@ const internalRedirectionHost = {
 | 
			
		||||
					.where('is_deleted', 0)
 | 
			
		||||
					.groupBy('id')
 | 
			
		||||
					.allowGraph('[owner,certificate]')
 | 
			
		||||
					.orderBy('domain_names', 'ASC');
 | 
			
		||||
					.orderBy(castJsonIfNeed('domain_names'), 'ASC');
 | 
			
		||||
 | 
			
		||||
				if (access_data.permission_visibility !== 'all') {
 | 
			
		||||
					query.andWhere('owner_user_id', access.token.getUserId(1));
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				// Query is used for searching
 | 
			
		||||
				if (typeof search_query === 'string') {
 | 
			
		||||
				if (typeof search_query === 'string' && search_query.length > 0) {
 | 
			
		||||
					query.where(function () {
 | 
			
		||||
						this.where('domain_names', 'like', '%' + search_query + '%');
 | 
			
		||||
						this.where(castJsonIfNeed('domain_names'), 'like', `%${search_query}%`);
 | 
			
		||||
					});
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,12 +1,15 @@
 | 
			
		||||
const _                = require('lodash');
 | 
			
		||||
const error            = require('../lib/error');
 | 
			
		||||
const utils            = require('../lib/utils');
 | 
			
		||||
const streamModel      = require('../models/stream');
 | 
			
		||||
const internalNginx    = require('./nginx');
 | 
			
		||||
const internalAuditLog = require('./audit-log');
 | 
			
		||||
const _                   = require('lodash');
 | 
			
		||||
const error               = require('../lib/error');
 | 
			
		||||
const utils               = require('../lib/utils');
 | 
			
		||||
const streamModel         = require('../models/stream');
 | 
			
		||||
const internalNginx       = require('./nginx');
 | 
			
		||||
const internalAuditLog    = require('./audit-log');
 | 
			
		||||
const internalCertificate = require('./certificate');
 | 
			
		||||
const internalHost        = require('./host');
 | 
			
		||||
const {castJsonIfNeed}    = require('../lib/helpers');
 | 
			
		||||
 | 
			
		||||
function omissions () {
 | 
			
		||||
	return ['is_deleted'];
 | 
			
		||||
	return ['is_deleted', 'owner.is_deleted', 'certificate.is_deleted'];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const internalStream = {
 | 
			
		||||
@@ -17,6 +20,12 @@ const internalStream = {
 | 
			
		||||
	 * @returns {Promise}
 | 
			
		||||
	 */
 | 
			
		||||
	create: (access, data) => {
 | 
			
		||||
		const create_certificate = data.certificate_id === 'new';
 | 
			
		||||
 | 
			
		||||
		if (create_certificate) {
 | 
			
		||||
			delete data.certificate_id;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return access.can('streams:create', data)
 | 
			
		||||
			.then((/*access_data*/) => {
 | 
			
		||||
				// TODO: At this point the existing ports should have been checked
 | 
			
		||||
@@ -26,16 +35,44 @@ const internalStream = {
 | 
			
		||||
					data.meta = {};
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				// streams aren't routed by domain name so don't store domain names in the DB
 | 
			
		||||
				let data_no_domains = structuredClone(data);
 | 
			
		||||
				delete data_no_domains.domain_names;
 | 
			
		||||
 | 
			
		||||
				return streamModel
 | 
			
		||||
					.query()
 | 
			
		||||
					.insertAndFetch(data)
 | 
			
		||||
					.insertAndFetch(data_no_domains)
 | 
			
		||||
					.then(utils.omitRow(omissions()));
 | 
			
		||||
			})
 | 
			
		||||
			.then((row) => {
 | 
			
		||||
				if (create_certificate) {
 | 
			
		||||
					return internalCertificate.createQuickCertificate(access, data)
 | 
			
		||||
						.then((cert) => {
 | 
			
		||||
							// update host with cert id
 | 
			
		||||
							return internalStream.update(access, {
 | 
			
		||||
								id:             row.id,
 | 
			
		||||
								certificate_id: cert.id
 | 
			
		||||
							});
 | 
			
		||||
						})
 | 
			
		||||
						.then(() => {
 | 
			
		||||
							return row;
 | 
			
		||||
						});
 | 
			
		||||
				} else {
 | 
			
		||||
					return row;
 | 
			
		||||
				}
 | 
			
		||||
			})
 | 
			
		||||
			.then((row) => {
 | 
			
		||||
				// re-fetch with cert
 | 
			
		||||
				return internalStream.get(access, {
 | 
			
		||||
					id:     row.id,
 | 
			
		||||
					expand: ['certificate', 'owner']
 | 
			
		||||
				});
 | 
			
		||||
			})
 | 
			
		||||
			.then((row) => {
 | 
			
		||||
				// Configure nginx
 | 
			
		||||
				return internalNginx.configure(streamModel, 'stream', row)
 | 
			
		||||
					.then(() => {
 | 
			
		||||
						return internalStream.get(access, {id: row.id, expand: ['owner']});
 | 
			
		||||
						return row;
 | 
			
		||||
					});
 | 
			
		||||
			})
 | 
			
		||||
			.then((row) => {
 | 
			
		||||
@@ -59,6 +96,12 @@ const internalStream = {
 | 
			
		||||
	 * @return {Promise}
 | 
			
		||||
	 */
 | 
			
		||||
	update: (access, data) => {
 | 
			
		||||
		const create_certificate = data.certificate_id === 'new';
 | 
			
		||||
 | 
			
		||||
		if (create_certificate) {
 | 
			
		||||
			delete data.certificate_id;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return access.can('streams:update', data.id)
 | 
			
		||||
			.then((/*access_data*/) => {
 | 
			
		||||
				// TODO: at this point the existing streams should have been checked
 | 
			
		||||
@@ -70,16 +113,32 @@ const internalStream = {
 | 
			
		||||
					throw new error.InternalValidationError('Stream could not be updated, IDs do not match: ' + row.id + ' !== ' + data.id);
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				if (create_certificate) {
 | 
			
		||||
					return internalCertificate.createQuickCertificate(access, {
 | 
			
		||||
						domain_names: data.domain_names || row.domain_names,
 | 
			
		||||
						meta:         _.assign({}, row.meta, data.meta)
 | 
			
		||||
					})
 | 
			
		||||
						.then((cert) => {
 | 
			
		||||
							// update host with cert id
 | 
			
		||||
							data.certificate_id = cert.id;
 | 
			
		||||
						})
 | 
			
		||||
						.then(() => {
 | 
			
		||||
							return row;
 | 
			
		||||
						});
 | 
			
		||||
				} else {
 | 
			
		||||
					return row;
 | 
			
		||||
				}
 | 
			
		||||
			})
 | 
			
		||||
			.then((row) => {
 | 
			
		||||
				// Add domain_names to the data in case it isn't there, so that the audit log renders correctly. The order is important here.
 | 
			
		||||
				data = _.assign({}, {
 | 
			
		||||
					domain_names: row.domain_names
 | 
			
		||||
				}, data);
 | 
			
		||||
 | 
			
		||||
				return streamModel
 | 
			
		||||
					.query()
 | 
			
		||||
					.patchAndFetchById(row.id, data)
 | 
			
		||||
					.then(utils.omitRow(omissions()))
 | 
			
		||||
					.then((saved_row) => {
 | 
			
		||||
						return internalNginx.configure(streamModel, 'stream', saved_row)
 | 
			
		||||
							.then(() => {
 | 
			
		||||
								return internalStream.get(access, {id: row.id, expand: ['owner']});
 | 
			
		||||
							});
 | 
			
		||||
					})
 | 
			
		||||
					.then((saved_row) => {
 | 
			
		||||
						// Add to audit log
 | 
			
		||||
						return internalAuditLog.add(access, {
 | 
			
		||||
@@ -92,6 +151,17 @@ const internalStream = {
 | 
			
		||||
								return saved_row;
 | 
			
		||||
							});
 | 
			
		||||
					});
 | 
			
		||||
			})
 | 
			
		||||
			.then(() => {
 | 
			
		||||
				return internalStream.get(access, {id: data.id, expand: ['owner', 'certificate']})
 | 
			
		||||
					.then((row) => {
 | 
			
		||||
						return internalNginx.configure(streamModel, 'stream', row)
 | 
			
		||||
							.then((new_meta) => {
 | 
			
		||||
								row.meta = new_meta;
 | 
			
		||||
								row      = internalHost.cleanRowCertificateMeta(row);
 | 
			
		||||
								return _.omit(row, omissions());
 | 
			
		||||
							});
 | 
			
		||||
					});
 | 
			
		||||
			});
 | 
			
		||||
	},
 | 
			
		||||
 | 
			
		||||
@@ -114,7 +184,7 @@ const internalStream = {
 | 
			
		||||
					.query()
 | 
			
		||||
					.where('is_deleted', 0)
 | 
			
		||||
					.andWhere('id', data.id)
 | 
			
		||||
					.allowGraph('[owner]')
 | 
			
		||||
					.allowGraph('[owner,certificate]')
 | 
			
		||||
					.first();
 | 
			
		||||
 | 
			
		||||
				if (access_data.permission_visibility !== 'all') {
 | 
			
		||||
@@ -131,6 +201,7 @@ const internalStream = {
 | 
			
		||||
				if (!row || !row.id) {
 | 
			
		||||
					throw new error.ItemNotFoundError(data.id);
 | 
			
		||||
				}
 | 
			
		||||
				row = internalHost.cleanRowCertificateMeta(row);
 | 
			
		||||
				// Custom omissions
 | 
			
		||||
				if (typeof data.omit !== 'undefined' && data.omit !== null) {
 | 
			
		||||
					row = _.omit(row, data.omit);
 | 
			
		||||
@@ -196,14 +267,14 @@ const internalStream = {
 | 
			
		||||
			.then(() => {
 | 
			
		||||
				return internalStream.get(access, {
 | 
			
		||||
					id:     data.id,
 | 
			
		||||
					expand: ['owner']
 | 
			
		||||
					expand: ['certificate', 'owner']
 | 
			
		||||
				});
 | 
			
		||||
			})
 | 
			
		||||
			.then((row) => {
 | 
			
		||||
				if (!row || !row.id) {
 | 
			
		||||
					throw new error.ItemNotFoundError(data.id);
 | 
			
		||||
				} else if (row.enabled) {
 | 
			
		||||
					throw new error.ValidationError('Host is already enabled');
 | 
			
		||||
					throw new error.ValidationError('Stream is already enabled');
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				row.enabled = 1;
 | 
			
		||||
@@ -249,7 +320,7 @@ const internalStream = {
 | 
			
		||||
				if (!row || !row.id) {
 | 
			
		||||
					throw new error.ItemNotFoundError(data.id);
 | 
			
		||||
				} else if (!row.enabled) {
 | 
			
		||||
					throw new error.ValidationError('Host is already disabled');
 | 
			
		||||
					throw new error.ValidationError('Stream is already disabled');
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				row.enabled = 0;
 | 
			
		||||
@@ -293,21 +364,21 @@ const internalStream = {
 | 
			
		||||
	getAll: (access, expand, search_query) => {
 | 
			
		||||
		return access.can('streams:list')
 | 
			
		||||
			.then((access_data) => {
 | 
			
		||||
				let query = streamModel
 | 
			
		||||
				const query = streamModel
 | 
			
		||||
					.query()
 | 
			
		||||
					.where('is_deleted', 0)
 | 
			
		||||
					.groupBy('id')
 | 
			
		||||
					.allowGraph('[owner]')
 | 
			
		||||
					.orderBy('incoming_port', 'ASC');
 | 
			
		||||
					.allowGraph('[owner,certificate]')
 | 
			
		||||
					.orderByRaw('CAST(incoming_port AS INTEGER) ASC');
 | 
			
		||||
 | 
			
		||||
				if (access_data.permission_visibility !== 'all') {
 | 
			
		||||
					query.andWhere('owner_user_id', access.token.getUserId(1));
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				// Query is used for searching
 | 
			
		||||
				if (typeof search_query === 'string') {
 | 
			
		||||
				if (typeof search_query === 'string' && search_query.length > 0) {
 | 
			
		||||
					query.where(function () {
 | 
			
		||||
						this.where('incoming_port', 'like', '%' + search_query + '%');
 | 
			
		||||
						this.where(castJsonIfNeed('incoming_port'), 'like', `%${search_query}%`);
 | 
			
		||||
					});
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
@@ -316,6 +387,13 @@ const internalStream = {
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				return query.then(utils.omitRows(omissions()));
 | 
			
		||||
			})
 | 
			
		||||
			.then((rows) => {
 | 
			
		||||
				if (typeof expand !== 'undefined' && expand !== null && expand.indexOf('certificate') !== -1) {
 | 
			
		||||
					return internalHost.cleanAllRowsCertificateMeta(rows);
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
				return rows;
 | 
			
		||||
			});
 | 
			
		||||
	},
 | 
			
		||||
 | 
			
		||||
@@ -327,9 +405,9 @@ const internalStream = {
 | 
			
		||||
	 * @returns {Promise}
 | 
			
		||||
	 */
 | 
			
		||||
	getCount: (user_id, visibility) => {
 | 
			
		||||
		let query = streamModel
 | 
			
		||||
		const query = streamModel
 | 
			
		||||
			.query()
 | 
			
		||||
			.count('id as count')
 | 
			
		||||
			.count('id AS count')
 | 
			
		||||
			.where('is_deleted', 0);
 | 
			
		||||
 | 
			
		||||
		if (visibility !== 'all') {
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user