Skip to content

Commit

Permalink
Merge branch 'v5.4' of github.com:ravendb/ravendb into v6.0
Browse files Browse the repository at this point in the history
    # Conflicts:
    #       src/Raven.Server/Documents/Indexes/IndexStorage.cs
    #       src/Raven.Studio/typescript/common/appUrl.ts
    #       src/Raven.Studio/typescript/viewmodels/resources/widgets/ongoingTasksWidget.ts
  • Loading branch information
arekpalinski committed Jun 28, 2024
1 parent 221f9bc commit 3f36f3e
Show file tree
Hide file tree
Showing 29 changed files with 386 additions and 68 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ public static SqlProvider GetSupportedProvider(string factoryName)
switch (factoryName)
{
case "System.Data.SqlClient":
case "Microsoft.Data.SqlClient":
return SqlProvider.SqlClient;
case "Npgsql":
return SqlProvider.Npgsql;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
using System;
using System.Data.Common;
using System.Data.SqlClient;
using Microsoft.Data.SqlClient;
using Npgsql;
using Oracle.ManagedDataAccess.Client;
using Raven.Client.Documents.Operations.ETL.SQL;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
using System.Collections.Generic;
using System.Data;
using System.Data.Common;
using System.Data.SqlClient;
using Microsoft.Data.SqlClient;
using System.Diagnostics;
using System.Globalization;
using System.IO;
Expand Down Expand Up @@ -50,6 +50,16 @@ public RelationalDatabaseWriter(SqlEtl etl, DocumentDatabase database)
_commandBuilder = _providerFactory.InitializeCommandBuilder();
_connection = _providerFactory.CreateConnection();
var connectionString = etl.Configuration.Connection.ConnectionString;

if (_etl.Configuration.Connection.FactoryName == "System.Data.SqlClient" &&
SqlConnectionStringParser.GetConnectionStringValue(connectionString, new string[] { "Encrypt" }) == null)
{
// We want to ensure compatibility between deprecated System.Data.SqlClient (where Encrypt parameter defaults to false)
// and Microsoft.Data.SqlClient which defaults it to SqlConnectionEncryptOption.Mandatory. If no option is provided,
// we set it to SqlConnectionEncryptOption.Optional (equivalent of false).
connectionString = SqlConnectionStringUtil.GetConnectionStringWithOptionalEncrypt(connectionString);
}

_connection.ConnectionString = connectionString;

OpenConnection(database, etl.Configuration.Name, etl.Configuration.ConnectionStringName);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ public class RelationalDatabaseWriterBase
private static readonly string[] SqlServerFactoryNames =
{
"System.Data.SqlClient",
"Microsoft.Data.SqlClient",
"System.Data.SqlServerCe.4.0",
"MySql.Data.MySqlClient",
"System.Data.SqlServerCe.3.5"
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
using System;
using System.Collections.Generic;
using System.Data.Common;
using System.Data.SqlClient;
using Microsoft.Data.SqlClient;
using System.Linq;
using System.Text;
using System.Threading;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
namespace Raven.Server.Documents.ETL.Providers.SQL.RelationalWriters;

public static class SqlConnectionStringUtil
{
public static string GetConnectionStringWithOptionalEncrypt(string connectionString)
{
connectionString = connectionString.Trim();

if (connectionString.EndsWith(';') == false)
connectionString += ";";

connectionString += "Encrypt=Optional";

return connectionString;
}
}
9 changes: 6 additions & 3 deletions src/Raven.Server/Documents/Indexes/Index.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3184,7 +3184,7 @@ internal HashSet<string> GetReferencedCollectionNames()
{
// multiple maps can reference the same collections, we wants to return distinct names only
return GetReferencedCollections()?
.SelectMany(p => p.Value?.Select(z => z.Name))
.SelectMany(p => p.Value?.Select(z => z.Name))
.ToHashSet();
}

Expand Down Expand Up @@ -4231,7 +4231,8 @@ private int MinimumSizeForCalculateIndexEtagLength(QueryMetadata q)
var length = sizeof(long) * 4 * Collections.Count + // last document etag, last tombstone etag and last mapped etags per collection
sizeof(int) + // definition hash
1 + // isStale
1; // index state
1 + // index state
sizeof(long); // created timestamp (binary)

if (q == null)
return length;
Expand Down Expand Up @@ -4273,6 +4274,8 @@ private unsafe void CalculateIndexEtagInternal(byte* indexEtagBytes, bool isStal
*indexEtagBytes = isStale ? (byte)0 : (byte)1;
indexEtagBytes += sizeof(byte);
*indexEtagBytes = (byte)indexState;
indexEtagBytes += sizeof(byte);
*(long*)indexEtagBytes = _indexStorage.CreatedTimestampAsBinary;
}

public long GetIndexEtag(QueryOperationContext context, QueryMetadata q)
Expand Down Expand Up @@ -4324,7 +4327,7 @@ public Dictionary<TombstoneDeletionBlockageSource, HashSet<string>> GetDisabledS
{
var dict = new Dictionary<TombstoneDeletionBlockageSource, HashSet<string>>();

if (Status is not (IndexRunningStatus.Disabled or IndexRunningStatus.Paused))
if (Status is not (IndexRunningStatus.Disabled or IndexRunningStatus.Paused))
return dict;

var source = new TombstoneDeletionBlockageSource(ITombstoneAware.TombstoneDeletionBlockerType.Index, Name);
Expand Down
37 changes: 21 additions & 16 deletions src/Raven.Server/Documents/Indexes/IndexStorage.cs
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,8 @@ private Constants()

public readonly CompareExchangeReferences ReferencesForCompareExchange;

public long CreatedTimestampAsBinary;

public IndexStorage(Index index, TransactionContextPool contextPool, DocumentDatabase database)
{
_index = index;
Expand Down Expand Up @@ -123,12 +125,15 @@ private unsafe void CreateSchema(DocumentDatabase documentDatabase)
using (Slice.External(context.Allocator, (byte*)&sourceTypeInt, sizeof(int), out Slice tmpSlice))
statsTree.Add(IndexSchema.SourceTypeSlice, tmpSlice);

if (statsTree.Read(IndexSchema.CreatedTimestampSlice) == null)
var createdTimestampResult = statsTree.Read(IndexSchema.CreatedTimestampSlice);
if (createdTimestampResult == null)
{
var binaryDate = SystemTime.UtcNow.ToBinary();
var binaryDate = CreatedTimestampAsBinary = SystemTime.UtcNow.ToBinary();
using (Slice.External(context.Allocator, (byte*)&binaryDate, sizeof(long), out Slice tmpSlice))
statsTree.Add(IndexSchema.CreatedTimestampSlice, tmpSlice);
}
else
CreatedTimestampAsBinary = createdTimestampResult.Reader.ReadLittleEndianInt64();

using (Slice.From(context.Allocator, documentDatabase.DbBase64Id, out var dbId))
statsTree.Add(IndexSchema.DatabaseIdSlice, dbId);
Expand Down Expand Up @@ -180,11 +185,11 @@ void PersistSearchEngine(Tree configurationTree)
{
string configurationKey = nameof(SearchEngineType);
string configurationName = _index.Type.IsAuto() ? RavenConfiguration.GetKey(x => x.Indexing.AutoIndexingEngineType) : RavenConfiguration.GetKey(x => x.Indexing.StaticIndexingEngineType);

SearchEngineType defaultEngineType =
_index.Type.IsAuto() ? _index.Configuration.AutoIndexingEngineType : _index.Configuration.StaticIndexingEngineType;
if(defaultEngineType == SearchEngineType.None)

if (defaultEngineType == SearchEngineType.None)
throw new InvalidDataException($"Default search engine is {SearchEngineType.None}. Please set {configurationName}.");
var result = configurationTree.Read(configurationKey);
if (result != null)
Expand Down Expand Up @@ -402,7 +407,7 @@ public IndexStats ReadStats(RavenTransaction tx)

var stats = new IndexStats
{
CreatedTimestamp = DateTime.FromBinary(statsTree.Read(IndexSchema.CreatedTimestampSlice).Reader.ReadLittleEndianInt64()),
CreatedTimestamp = DateTime.FromBinary(CreatedTimestampAsBinary),
ErrorsCount = (int)(table?.NumberOfEntries ?? 0)
};

Expand All @@ -424,8 +429,8 @@ public IndexStats ReadStats(RavenTransaction tx)
{
var entriesCountSize = entriesCountReader.Value.Length;
//backward compatibility https://github.com/ravendb/ravendb/commit/5c53b01ee2b4fad8f3ef410f3e4976144d72c023
entriesCount = entriesCountSize == sizeof(long)
? entriesCountReader.Value.ReadLittleEndianInt64()
entriesCount = entriesCountSize == sizeof(long)
? entriesCountReader.Value.ReadLittleEndianInt64()
: entriesCountReader.Value.ReadLittleEndianInt32();
}

Expand Down Expand Up @@ -918,7 +923,7 @@ public unsafe IndexFailureInformation UpdateStats(DateTime indexingTime, Indexin

if (stats.EntriesCount != null) // available only when tx was committed
statsTree.Add(IndexSchema.EntriesCount, stats.EntriesCount.Value);

var binaryDate = indexingTime.ToBinary();
using (Slice.External(context.Allocator, (byte*)&binaryDate, sizeof(long), out Slice binaryDateslice))
statsTree.Add(IndexSchema.LastIndexingTimeSlice, binaryDateslice);
Expand Down Expand Up @@ -1014,11 +1019,11 @@ public static SearchEngineType ReadSearchEngineType(string name, StorageEnvironm
{
throw new InvalidOperationException($"Index '{name}' does not contain valid {nameof(SearchEngineType)} property. It contains: {result.Reader.ToStringValue()}.");
}

return persistedSearchEngineType;
}
}
}

public static IndexSourceType ReadIndexSourceType(string name, StorageEnvironment environment)
{
using (var tx = environment.ReadTransaction())
Expand Down Expand Up @@ -1078,8 +1083,8 @@ internal HashSet<string> ReadIndexTimeFields()
var searchEngineType = ReadSearchEngineType(_index.Name, _environment);
using (var tx = _environment.ReadTransaction())
{
fields = searchEngineType == SearchEngineType.Corax
? Corax.Utils.TimeFields.ReadTimeFieldsNames(tx)
fields = searchEngineType == SearchEngineType.Corax
? Corax.Utils.TimeFields.ReadTimeFieldsNames(tx)
: ReadLuceneTimeFields(tx);
}

Expand All @@ -1102,7 +1107,7 @@ HashSet<string> ReadLuceneTimeFields(Transaction tx)
}
}
}

return container;
}
}
Expand Down Expand Up @@ -1181,7 +1186,7 @@ internal sealed class IndexSchema
public static readonly Slice TimeSlice;

public static readonly Slice SearchEngineType;

static IndexSchema()
{
using (StorageEnvironment.GetStaticContext(out var ctx))
Expand Down
9 changes: 9 additions & 0 deletions src/Raven.Server/Raven.Server.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -179,6 +179,14 @@
<PackageReference Include="Microsoft.Bcl.AsyncInterfaces" Version="8.0.0" />
<PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.10.0" />
<PackageReference Include="Microsoft.CodeAnalysis.CSharp.Workspaces" Version="4.10.0" />
<PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.7.0" />
<PackageReference Include="Microsoft.CodeAnalysis.CSharp.Workspaces" Version="4.7.0" />
<PackageReference Include="Microsoft.Data.SqlClient" Version="5.2.1" />
<PackageReference Include="Microsoft.Extensions.Configuration.CommandLine" Version="8.0.0" />
<PackageReference Include="Microsoft.Extensions.Configuration.EnvironmentVariables" Version="8.0.0" />
<PackageReference Include="Microsoft.Extensions.Configuration.Json" Version="8.0.0" />
<PackageReference Include="Microsoft.Extensions.Configuration.UserSecrets" Version="8.0.0" />
<PackageReference Include="Microsoft.Win32.Registry" Version="5.0.0" />
<PackageReference Include="MySqlConnector" Version="2.3.7" />
<PackageReference Include="NCrontab.Advanced" Version="1.3.28" />
<PackageReference Include="Npgsql" Version="8.0.3" />
Expand All @@ -196,6 +204,7 @@
</PackageReference>
<PackageReference Include="Snappier" Version="1.1.6" />
<PackageReference Include="System.Data.SqlClient" Version="4.8.6" />
<PackageReference Include="System.Collections.Immutable" Version="8.0.0" />
<PackageReference Include="System.Linq.Async" Version="6.0.1" />
<PackageReference Include="System.Net.Http" Version="4.3.4" />
<PackageReference Include="System.Reflection.Metadata" Version="8.0.0" />
Expand Down
Original file line number Diff line number Diff line change
@@ -1,12 +1,14 @@
using Raven.Server.SqlMigration.Schema;
using Raven.Client.Documents.Operations.ETL.SQL;
using Raven.Server.Documents.ETL.Providers.SQL.RelationalWriters;
using Raven.Server.SqlMigration.Schema;

namespace Raven.Server.SqlMigration.MsSQL
{
internal partial class MsSqlDatabaseMigrator : GenericDatabaseMigrator
{
protected override string FactoryName => "System.Data.SqlClient";
protected override string FactoryName => "Microsoft.Data.SqlClient";

public MsSqlDatabaseMigrator(string connectionString) : base(connectionString)
public MsSqlDatabaseMigrator(string connectionString) : base(OverrideConnectionStringIfNeeded(connectionString))
{
}

Expand Down Expand Up @@ -78,5 +80,13 @@ protected override string GetSelectAllQueryForTable(string tableSchema, string t
{
return "select * from " + QuoteTable(tableSchema, tableName);
}

private static string OverrideConnectionStringIfNeeded(string connectionString)
{
if (SqlConnectionStringParser.GetConnectionStringValue(connectionString, new string[] { "Encrypt" }) == null)
connectionString = SqlConnectionStringUtil.GetConnectionStringWithOptionalEncrypt(connectionString);

return connectionString;
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,12 @@ import sqlConnectionStringSyntax = require("common/helpers/database/sqlConnectio
class connectionStringSqlEtlModel extends connectionStringModel {

static sqlProviders: Array<valueAndLabelItem<string, string> & { tooltip: string }> = [
{ value: "System.Data.SqlClient", label: "Microsoft SQL Server", tooltip: sqlConnectionStringSyntax.mssqlSyntax },
{ value: "Microsoft.Data.SqlClient", label: "Microsoft SQL Server", tooltip: sqlConnectionStringSyntax.mssqlSyntax },
{ value: "MySqlConnector.MySqlConnectorFactory", label: "MySQL Server", tooltip: sqlConnectionStringSyntax.mysqlSyntax },
{ value: "Npgsql", label: "PostgreSQL", tooltip: sqlConnectionStringSyntax.npgsqlSyntax },
{ value: "Oracle.ManagedDataAccess.Client", label: "Oracle Database", tooltip: sqlConnectionStringSyntax.oracleSyntax },
{ value: "MySql.Data.MySqlClient", label: "DEPRECATED: MySQL Server", tooltip: sqlConnectionStringSyntax.mysqlSyntax },
{ value: "System.Data.SqlClient", label: "DEPRECATED: Microsoft SQL Server", tooltip: sqlConnectionStringSyntax.mssqlSyntax },
];

connectionString = ko.observable<string>();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -127,15 +127,13 @@ class ongoingTaskSqlEtlEditModel extends ongoingTaskEditModel {
Disabled: this.taskState() === "Disabled",
MentorNode: this.manualChooseMentor() ? this.mentorNode() : undefined,
PinToMentorNode: this.pinMentorNode(),
FactoryName: "System.Data.SqlClient",
ForceQueryRecompile: this.forceRecompileQuery(),
ParameterizeDeletes: this.parameterizedDeletes(),
CommandTimeout: this.commandTimeout() || null,
QuoteTables: this.tableQuotation(),
Transforms: this.transformationScripts().map(x => x.toDto()),
SqlTables: this.sqlTables().map(x => x.toDto())

} as Raven.Client.Documents.Operations.ETL.SQL.SqlEtlConfiguration;
};
}

static empty(): ongoingTaskSqlEtlEditModel {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -161,7 +161,7 @@ class sqlMigration {
labelForProvider(type: Raven.Server.SqlMigration.MigrationProvider) {
switch (type) {
case "MsSQL":
return "Microsoft SQL Server (System.Data.SqlClient)";
return "Microsoft SQL Server (Microsoft.Data.SqlClient)";
case "MySQL_MySql_Data":
return "DEPRECATED: MySQL Server (MySql.Data.MySqlClient)";
case "MySQL_MySqlConnector":
Expand Down Expand Up @@ -305,7 +305,7 @@ class sqlMigration {
case "MySQL_MySqlConnector":
return "MySqlConnector.MySqlConnectorFactory";
case "MsSQL":
return "System.Data.SqlClient";
return "Microsoft.Data.SqlClient";
case "NpgSQL":
return "Npgsql";
case "Oracle":
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,16 @@ import virtualColumn = require("widgets/virtualGrid/columns/virtualColumn");
import generalUtils = require("common/generalUtils");

class multiNodeTagsColumn<T extends object> implements virtualColumn {
private readonly nodeHrefAccessor: (item: T, nodeTag: string) => string;
private readonly nodeLinkTitleAccessor: (item: T, nodeTag: string) => string;

constructor(gridController: virtualGridController<T>,
public valueAccessor: ((item: T) => string[]),
public width: string,
public opts: textColumnOpts<T> = {}) {
public valueAccessor: ((item: T) => string[]),
public width: string,
public opts: multiNodeTagsColumnOpts<T> = {}
) {
this.nodeHrefAccessor = opts.nodeHrefAccessor;
this.nodeLinkTitleAccessor = opts.nodeLinkTitleAccessor;
}

canHandle(actionId: string): boolean {
Expand All @@ -34,20 +39,31 @@ class multiNodeTagsColumn<T extends object> implements virtualColumn {
renderCell(item: T): string {
const extraCssClasses = this.opts.extraClass ? this.opts.extraClass(item) : '';

return `<div class="cell text-cell ${extraCssClasses}" style="width: ${this.width}">${this.getValueHtml(this.getValue(item))}</div>`;
return `<div class="cell text-cell ${extraCssClasses}" style="width: ${this.width}; display: flex">${this.getValueHtml(item)}</div>`;
}

private getValue(item: T): string[] {
return this.valueAccessor.bind(item)(item);
}

private getValueHtml(nodeTags: string[]): string {
private getValueHtml(item: T): string {
let result = "";

const nodeTags = this.getValue(item);

if (nodeTags) {
for (const tag of nodeTags) {
const extraClass = `node-${tag}`;
result += `<span class="node-label ${extraClass}">${tag}</span>`;

let nodeHtml = `<span class="node-label ${extraClass}">${tag}</span>`;

if (this.nodeHrefAccessor) {
const href = this.nodeHrefAccessor(item, tag);
const title = this.nodeLinkTitleAccessor ? this.nodeLinkTitleAccessor(item, tag) : tag;

nodeHtml = `<a href="${href}" title="${title}">${nodeHtml}</a>`;
}

result += nodeHtml;
}
}

Expand Down
5 changes: 5 additions & 0 deletions src/Raven.Studio/typings/_studio/dto.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -703,6 +703,11 @@ interface hypertextColumnOpts<T> extends textColumnOpts<T> {
openInNewTab?: (item: T) => boolean;
}

interface multiNodeTagsColumnOpts<T> extends textColumnOpts<T> {
nodeHrefAccessor?: (item: T, nodeTag: string) => string;
nodeLinkTitleAccessor?: (item: T, nodeTag: string) => string;
}

type timeSeriesColumnEventType = "plot" | "preview";

interface timeSeriesColumnOpts<T> extends textColumnOpts<T> {
Expand Down
Loading

0 comments on commit 3f36f3e

Please sign in to comment.