Я пытаюсь изменить поставщика моих Crystal Reports с OLEDBSQL
на MSOLEDBSQL
, поскольку OLEDBSQL
не поддерживает TLS 1.2
Мой код:
Private Sub ProcessReport()
Dim crxRpt As New CrystalDecisions.CrystalReports.Engine.ReportDocument
crxRpt.Load("D:\RND\Crystal Reports - Connector\CrystallReportConvertVB\SummaryReport.rpt")
ChangeOleDbConnectionInfo(crxRpt, "SERVER", "DBNAME", "MSOLEDBSQL", False, "USERID", "PASSWORD")
End Sub
Friend Shared Sub ChangeOleDbConnectionInfo(ByVal reportDocument As ReportDocument, ByVal server As String, ByVal database As String, ByVal oledbProvider As String, ByVal integratedSecurity As Boolean, ByVal userId As String, ByVal password As String)
Dim boMainPropertyBag = New NameValuePairs2()
Dim boInnerPropertyBag = New NameValuePairs2()
boInnerPropertyBag.Add(New NameValuePair2("Application Intent", "READWRITE"))
boInnerPropertyBag.Add(New NameValuePair2("Auto Translate", "-1"))
boInnerPropertyBag.Add(New NameValuePair2("Connect Timeout", "15"))
boInnerPropertyBag.Add(New NameValuePair2("Data Source", server))
boInnerPropertyBag.Add(New NameValuePair2("DataTypeCompatibility", "0"))
boInnerPropertyBag.Add(New NameValuePair2("General Timeout", "0"))
boInnerPropertyBag.Add(New NameValuePair2("Initial Catalog", database))
boInnerPropertyBag.Add(New NameValuePair2("Integrated Security", If(integratedSecurity, "True", "False")))
boInnerPropertyBag.Add(New NameValuePair2("Locale Identifier", "1033"))
boInnerPropertyBag.Add(New NameValuePair2("MARS Connection", "0"))
boInnerPropertyBag.Add(New NameValuePair2("OLE DB Services", "-5"))
boInnerPropertyBag.Add(New NameValuePair2("Provider", oledbProvider))
boInnerPropertyBag.Add(New NameValuePair2("Tag with column collation when possible", "0"))
boInnerPropertyBag.Add(New NameValuePair2("Trust Server Certificate", "0"))
boInnerPropertyBag.Add(New NameValuePair2("Use DSN Default Properties", "False"))
boInnerPropertyBag.Add(New NameValuePair2("Use Encryption for Data", "0"))
boMainPropertyBag.Add(New NameValuePair2("Database DLL", "crdb_ado.dll"))
boMainPropertyBag.Add(New NameValuePair2("QE_DatabaseName", database))
boMainPropertyBag.Add(New NameValuePair2("QE_DatabaseType", "OLE DB (ADO)"))
boMainPropertyBag.Add(New NameValuePair2("QE_LogonProperties", boInnerPropertyBag))
boMainPropertyBag.Add(New NameValuePair2("QE_ServerDescription", server))
boMainPropertyBag.Add(New NameValuePair2("QE_SQLDB", "True"))
boMainPropertyBag.Add(New NameValuePair2("SSO Enabled", "False"))
Dim conAttributes As New CrystalDecisions.Shared.DbConnectionAttributes()
conAttributes.Collection = boMainPropertyBag
Dim boConnectionInfo = New ConnectionInfo With {
.Attributes = conAttributes,
.DatabaseName = database,
.ServerName = server
}
If Not integratedSecurity Then
boConnectionInfo.UserID = userId
boConnectionInfo.Password = password
End If
Dim myTables As CrystalDecisions.CrystalReports.Engine.Tables = reportDocument.Database.Tables
For Each myTable As CrystalDecisions.CrystalReports.Engine.Table In reportDocument.Database.Tables
Dim myTableLogonInfo As CrystalDecisions.Shared.TableLogOnInfo = myTable.LogOnInfo
myTableLogonInfo.ConnectionInfo.Attributes.Collection.Clear()
myTableLogonInfo.ConnectionInfo.Attributes = boConnectionInfo.Attributes
Next
Dim mySections As CrystalDecisions.CrystalReports.Engine.Sections = reportDocument.ReportDefinition.Sections
For Each mySection As CrystalDecisions.CrystalReports.Engine.Section In mySections
Dim myReportObjects As CrystalDecisions.CrystalReports.Engine.ReportObjects = mySection.ReportObjects
For Each myReportObject As CrystalDecisions.CrystalReports.Engine.ReportObject In myReportObjects
If myReportObject.Kind = CrystalDecisions.Shared.ReportObjectKind.SubreportObject Then
Dim mySubreportObject As CrystalDecisions.CrystalReports.Engine.SubreportObject = CType(myReportObject, CrystalDecisions.CrystalReports.Engine.SubreportObject)
Dim subReportDocument As CrystalDecisions.CrystalReports.Engine.ReportDocument = mySubreportObject.OpenSubreport(mySubreportObject.SubreportName)
Dim mytablessub As CrystalDecisions.CrystalReports.Engine.Tables = subReportDocument.Database.Tables
For Each myTable As CrystalDecisions.CrystalReports.Engine.Table In mytablessub
Dim myTableLogonInfo As CrystalDecisions.Shared.TableLogOnInfo = myTable.LogOnInfo
myTableLogonInfo.ConnectionInfo.Attributes.Collection.Clear()
myTableLogonInfo.ConnectionInfo.Attributes = boConnectionInfo.Attributes
Next
End If
Next
Next
reportDocument.VerifyDatabase()
reportDocument.SaveAs("D:\RND\Crystal Reports - Connector\CrystallReportConvertVB\CrystallReportConvertVB\Rpt-changed3.rpt")
End Sub
Я использую следующие версии Crystal Report:
![Crystal Reports References](https://i.stack.imgur.com/umJ4l.png)
I have noted the following
- Calling
verifyDatabase()
or refresh()
on the ReportDocument
changes the provider property back into original.
- I have tried the
ApplyLogOnInfo()
- Found that it only reset the username and password and not the provider. Calling this method also resets the changed provider.
- Now that I've managed to change the Provider by the following code, but It always reset the provider to the original provider.
How can this be done?
Are there any workarounds for this?
Further Update
I was able to change the provider to MSOLEDBSQL in the main report by using the code suggested by Frank (с благодарностью) ниже. Но все же он не работает для вложенных отчетов и выдает следующую ошибку.
Не поддерживается в вложенных отчетах
Not supported within sub reports." />
Further Update (2) - on Subreport provider change
I was able to change the provider for Crystal Report Subreports by the following code.
Friend Shared Function ChangeOleDbConnectionInfoSubReports(ByVal reportDocument As ReportDocument, ByVal server As String, ByVal database As String, ByVal schema As String, ByVal oledbProvider As String, ByVal integratedSecurity As Boolean, ByVal userId As String, ByVal password As String) As ReportDocument
Dim boInnerPropertyBag = New PropertyBag()
boInnerPropertyBag.Add("Application Intent", "READWRITE")
boInnerPropertyBag.Add("Auto Translate", "-1")
boInnerPropertyBag.Add("Connect Timeout", "15")
boInnerPropertyBag.Add("Data Source", server)
boInnerPropertyBag.Add("DataTypeCompatibility", "0")
boInnerPropertyBag.Add("General Timeout", "0")
boInnerPropertyBag.Add("Initial Catalog", database)
boInnerPropertyBag.Add("Integrated Security", If(integratedSecurity, "True", "False"))
boInnerPropertyBag.Add("Locale Identifier", "1033")
boInnerPropertyBag.Add("MARS Connection", "0")
boInnerPropertyBag.Add("OLE DB Services", "-5")
boInnerPropertyBag.Add("Provider", oledbProvider)
boInnerPropertyBag.Add("Tag with column collation when possible", "0")
boInnerPropertyBag.Add("Trust Server Certificate", "0")
boInnerPropertyBag.Add("Use DSN Default Properties", "False")
boInnerPropertyBag.Add("Use Encryption for Data", "0")
Dim boMainPropertyBag = New PropertyBag()
boMainPropertyBag.Add("Database DLL", "crdb_ado.dll")
boMainPropertyBag.Add("QE_DatabaseName", database)
boMainPropertyBag.Add("QE_DatabaseType", "OLE DB (ADO)")
boMainPropertyBag.Add("QE_LogonProperties", boInnerPropertyBag)
boMainPropertyBag.Add("QE_ServerDescription", server)
boMainPropertyBag.Add("QE_SQLDB", "True")
boMainPropertyBag.Add("SSO Enabled", "False")
Dim boConnectionInfo = New CrystalDecisions.ReportAppServer.DataDefModel.ConnectionInfo With {
.Attributes = boMainPropertyBag,
.Kind = CrConnectionInfoKindEnum.crConnectionInfoKindCRQE
}
If Not integratedSecurity Then
boConnectionInfo.UserName = userId
boConnectionInfo.Password = password
End If
For Each subreport As ReportDocument In reportDocument.Subreports
For Each table As CrystalDecisions.ReportAppServer.DataDefModel.Table In reportDocument.ReportClientDocument.SubreportController.GetSubreportDatabase(subreport.Name).Tables
Dim boTable = New CrystalDecisions.ReportAppServer.DataDefModel.Table With {
.ConnectionInfo = boConnectionInfo,
.Name = table.Name,
.QualifiedName = schema & "." & table.Name,
.[Alias] = table.[Alias]
}
reportDocument.ReportClientDocument.SubreportController.SetTableLocation(subreport.Name, table, boTable)
Next
Next
reportDocument.VerifyDatabase()
Return reportDocument
End Function
Lessons Learnt
Noticed that, I can not update the driver if the installed Crystal
Report Runtime does not match with the Visual studio references (I have taken the DLLs from some Nuget packages that are not official releases). I
Couldn't make the code work with Crystal Reports 10.5.3700 that I used
and I updated the runtime to 13.0.2000 ( Согласно этой статье )
введите описание изображения здесь
В Crystal Reports также есть строгие правила в отношении архитектуры процессора. Поскольку я использую 32-битную версию, я установил все настройки в Visual Studio для компиляции проекта как 32-битного приложения.