|
|
Microsoft SQL Server, Transact-SQL, Business Intelligence, CLR, ...
-
-
Prezentaci (slides) z mé dnešní přednášky "Návrh schématu databáze v praxi" na MS Fest 2011 najdete v sekci soubory.
|
-
Dneska jsem se nechal unést a chtěl jsem vypnout na MS SQL 2008 R2 horní mez využité paměti, kterou jsme měli na serveru staženou na 2.5 GB.
Nu což, vlezl jsem do SQL Management Studia, dal jsem si vlastnosti serveru, koukám do nastavení Memory a změnil jsem nastavení vlastnosti "Maximum server memory (in MB)" z 2500 na 0 velmi naivně předpokládajíc, že stejně jako jinde to bude interpretováno jako "no limit".
Omyl, to nezkoušejte. Poslední, co jsem ještě záhledl bylo, že si to SQL sám změnil na 16 (MB) a to bylo na dlouho všechno, co jsem z SQL serveru viděl.
Následný pokud změny na vyšší hodnotu už SQL Management Studio neprovedlo, nové připojení na SQL Management Studio už neúspěšné (ani nevytvořil connection).
Zůstal viset ve stavu, kdy si nechal asi 500 MB paměti a bral 25% CPU, což nekleslo ani po odpojení zátěže (kterou již stejně neodbavoval).
Stop service nic, takže kill procesu z task manageru.
Nový start service, zabraná paměť procesem 70 MB, bez šance na připojení z Management Studia.
Zkouším OSQL, sp_reconfigure, nic, hlášky typu "There is insufficient system memory in resource pool 'internal' to run this query.", nebo se vůbec nevytvoří connection.
Nakonec jsem to vybojoval na několikátý pokus (vždy těsně po startu service) přes "sqlcmd -A -E" (Dedicated Administrator Connection) a volby
sp_configure 'show advanced options', 1; GO RECONFIGURE; GO sp_configure 'max server memory', 2147483647; GO RECONFIGURE; GO...horká půlhodinka.
|
-
-
Dneska se budu s laskavým svolením chlubit cizím peřím, uvádím postup (mnou neověřený, ale z důvěryhodného zdroje - od Roberta Kindla z firmy EXEC), jak do MS SQL 2008 R2 x64 dostat český stemmer pro fultextové vyhledávání:
****
musis nainstalovat Microsoft Search Server 2010 Express - to by melo byt zdarma a cesky stemmer to obsahuje
bohuzel je ale ten stemmer dostupny jen v tom Sharepointu (neregistruje se to do systemu)
 takze jsem vlasnimi silami vyhackoval toto: Windows Registry Editor Version 5.00
[HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Microsoft SQL Server\MSSQL10_50.SQL2008R2\MSSearch\Language\ces] "TsaurusFile"="tsces.xml" "Locale"=dword:00000405 "WBreakerClass"="{468bfc77-3876-4a47-a6ff-f5f6e8ea7968}" "StemmerClass"="{f51b7203-9bf9-4c39-b655-18fad8fa8a9a}"
[HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Microsoft SQL Server\MSSQL10_50.SQL2008R2\MSSearch\CLSID\{f51b7203-9bf9-4c39-b655-18fad8fa8a9a}] @="c:\\Program Files\\Microsoft Office Servers\\14.0\\Bin\\mswb7.dll"
[HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Microsoft SQL Server\MSSQL10_50.SQL2008R2\MSSearch\CLSID\{468bfc77-3876-4a47-a6ff-f5f6e8ea7968}] @="c:\\Program Files\\Microsoft Office Servers\\14.0\\Bin\\mswb7.dll"
toto konkretne zaregistruje ten cesky stemmer z Microsoft Search Server 2010 Express do MSSQL 2008 R2 (pro MSSQL 2008 a MSSQL 2005 by to zrejme slo podobne)
 pak uz staci jen exec sp_fulltext_service 'update_languages'; exec sp_fulltext_service 'restart_all_fdhosts';
overeni instalace (vypise se stemmer) s lcid 1029 exec sp_help_fulltext_system_components wordbreaker SELECT * FROM sys.fulltext_languages order by lcid
 sklonovat to umi: select * from sys.dm_fts_parser('FORMSOF(INFLECTIONAL, květina)', 1029, 0, 0)
++++
jeste je asi treba tohle: copy "c:\Program Files\Microsoft Office Servers\14.0\Data\Config\tsces.xml" "c:\MSSQL2008R2\FTData" (do slozky k ostatnim ts???.xml)
Noise Words, ktere jsou v Sharepointu 2010 ulozeny v c:\Program Files\Microsoft Office Servers\14.0\Data\Config\noiseces.txt by se na MSSQL 2008 R2 mely konfigurovat pres CREATE FULLTEXT STOPLIST
|
-
Na konferenci Microsoft Fest 2009 jsem dnes prezentoval přednášku "Optimalizace databázových aplikací v praxi", jedná se o částečně obměněnou podobu session již prezentované na SQL DevDays 2009. Slides, demo-kód i ukázkové databáze (atributová databáze a schema-killers) jsou k dispozici ke stažení.
|
-
-
Na MSSQL serveru se občas stává, že backupování přestane truncatovat transaction log a ten narůstá a narůstá.
Z nějakého důvodu je potřeba udělat sekvenci shrink-backup-shrink, aby bylo dosaženo požadovaného účinku:
1) Run this code: DBCC SHRINKFILE(pubs_log, 2)
2) Run this code if you want to truncate the transaction log and not keep a backup of the transaction log. Truncate_only invalidates your transaction log backup sequence. Take a full backup of your database after you perform backup log with truncate_only: BACKUP LOG pubs WITH TRUNCATE_ONLY
-or-
Run this code if you want to keep a backup of your transaction log and keep your transaction log backup sequence intact. See SQL Server Books Online topic "BACKUP" for more information: BACKUP LOG pubs TO pubslogbackup
3) Run this code: DBCC SHRINKFILE(pubs_log, 2)
Aktualizace pro SQL 2008
V SQL Serveru 2008 již není TRUNCATE_ONLY přepínač použitelný, místo toho je potřeba přepnout DB do Simple recovery modelu a udělat shrink logu. Pak je možné přepnout zpět na Full.
|
-
V management nástrojích pro SQL Server 2005 nelze založit login pro doménovou skupinu. Login se mi podařilo založit pomocí příkazu create login [HAVIT\Development] from windowskde v našem případě je Development doménovou skupinou. Další pozorování je, že login doménové skupiny není možné pomocí management studia zakázat:
- při nastavení Login na Disabled je zobrazena chyba,
- nastavení Permissions to connect to database engine na Deny je ignorováno.
|
-
Občas se Vám může hodit možnost vyskriptovat kompletní databázové schéma do SQL skriptu. Spuštěním takového skriptu pak můžete úplné DB schema zpětně reakonstruovat. Pokud jste si (ne)oblíbili schopnosti databázové edice Visual Studia stejně jako já, pak Vás potěším, že to jde i mnohem jednodušeji pomocí tzv. SQL Publishing Wizzardu přímo ze sady standardních management nástrojů SQL Serveru 2005:
Vyskriptování schématu DB z GUI
- Spusťte si SQL Server 2005 Management Studio
- V Object Exploreru pravým tlačítkem na dotčenou databázi a zvolte Tasks ~ Generate Scripts...
- Dole zaškrtněte "Script all objects in the selected database" (v horní části okna máte předvolenu dotčenou DB), pokračujte Next,
- Options - Podle své potřeby můžete upravit nastavení generátoru a ovlivnit podobu výsledného skriptu, např. zvolit jinou cílovou verzi SQL Serveru, pokračujte Next,
- Script Mode - zvolte, kam chcete skript vygenerovat - obvykle do souboru, pokračujte Next,
- Review nastavení + Finish
...a je hotovo.
Vyskriptování schématu DB z příkazové řádky
...není nic jednoduššího:
"C:\Program Files\Microsoft SQL Server\90\Tools\Publishing\1.2\sqlpubwiz" script -d DbName -S ServerName -U UserName -P Password TargetScriptFile.sql -schemaonly -f
U nás tímto způsobem průběžně ukládáme ke každé solution aktuální schéma DB, tak, aby bylo v Subversion (source-control) vždy uložena příslušná verze schématu DB, s kterou aplikace pracuje.
Pro úplnost dodávám, že pro synchronizaci databázových schémat a generování rozdílových skriptů používám RedGate SQL Compare.
|
-
Přenastavení počítadla pro identity sloupec na 1:
DBCC CHECKIDENT (MyTable, RESEED, 1)Počítadlo vyresetuje na výchozí hodnotu i TRUNCATE TABLE, který slouží pro vymazání celé tabulky.
Více o DBCC CHECKIDENT [MSDN].
|
-
- COALESCE() je z ANSI/ISO standardu SQL32, kdežto ISNULL() je jen T-SQL rozšíření (Microsoft SQL Serveru).
- ISNULL() má vlastní interpretaci, kdežto COALESCE() je pouze zkrácený zápis pro CASE strukturu a i se tak provádí:
CASE WHEN (expression1 IS NOT NULL) THEN expression1 ... WHEN (expressionN IS NOT NULL) THEN expressionN ELSE NULL END
- Typ výsledku ISNULL() je vždy dle prvního parametru, typ výsledku COALESCE() odpovídá CASE struktuře, tj. pokouší se o maximální záběr ze všech parametrů (SQL Server Books Online: "Returns the highest precedence type from the set of types in result_expressions and the optional else_result_expression.").
DECLARE @Test char(2) SET @Test = NULL SELECT ISNULL(@Test, 'abcde'), COALESCE(@Test, 'abcde')Např. výše uvedený test vrací 'ab', 'abcde'.
- Vzhledem k převodu COALESCE() na CASE oproti vlastnímu provádění je ISNULL() mnohdy rychlejší.
- ISNULL() pochopitelně bere pouze dva parametry, kdežto COALESCE() víc.
|
-
Nezkušeného vývojáře, ale jak jsem se bohužel na vlastní oči mnohdy přesvědčil - mnohdy i velmi zkušeného vývojáře, dokáže zdrcujícím způsobem překvapit výsledek následujícího jednoduchého příkladu:
CREATE TABLE TransactionTestTable ( TransactionTestID int IDENTITY PRIMARY KEY, TransactionText nvarchar(100) ) GO
CREATE PROCEDURE TransactionTest AS BEGIN TRANSACTION -- Korektní SQL statement uvnitř transakce INSERT INTO TransactionTestTable(TransactionText) VALUES('První insert')
-- Simulace chyby uvnitř transakce (do identity-column nelze zapisovat) INSERT INTO TransactionTestTable(TransactionTestID, TransactionText) VALUES(1, 'Druhý insert')
-- Korektní SQL statement uvnitř transakce, po chybě INSERT INTO TransactionTestTable(TransactionText) VALUES('Třetí insert')
COMMIT TRANSACTION GO
EXEC TransactionTest SELECT * FROM TransactionTestTable GO
...spuštěná transakce (zde pouze pro formu zabalená do procedury TransactionTest) selže - druhý INSERT zfailuje, protože nemůže zapisovat přímo do identity-column. To zde není podstatné, je to pouze simulace pro zfailování statementu uvnitř transakce.
Velkým překvapením mnohých však je, že v tabulce TransactionTestTable budou po skončení transakce následující data:
| TransactionTestID |
TransactionText |
| 1 |
První insert |
| 2 |
Třetí insert |
Zfailováním druhého statementu nedojde k automatickému rollbacku celé transakce! Run-time chyby statementů rollbackují automaticky pouze statement, který chybu způsobil!
Pokud chceme, aby run-time chyba automaticky zrollbackovala celou transakci, musíme nastavit SET XACT_ABORT ON:
CREATE PROCEDURE TransactionTest AS SET XACT_ABORT ON BEGIN TRANSACTION -- Korektní SQL statement uvnitř transakce INSERT INTO TransactionTestTable(TransactionName) VALUES('První insert')
-- Chybný SQL statement uvnitř transakce, do identity-column nelze zapisovat INSERT INTO TransactionTestTable(TransactionTestID, TransactionName) VALUES(1, 'Druhý insert')
-- Korektní SQL statement uvnitř transakce, po chybě INSERT INTO TransactionTestTable(TransactionName) VALUES('Třetí insert')
COMMIT TRANSACTION GOJemněji než automatický rollback pomocí SET XACT_ABORT ON lze v praxi použít ošetření chyb pomocí TRY..CATCH:
USE AdventureWorks; GO BEGIN TRANSACTION;
BEGIN TRY -- Generate a constraint violation error. DELETE FROM Production.Product WHERE ProductID = 980; END TRY BEGIN CATCH SELECT ERROR_NUMBER() AS ErrorNumber, ERROR_SEVERITY() AS ErrorSeverity, ERROR_STATE() as ErrorState, ERROR_PROCEDURE() as ErrorProcedure, ERROR_LINE() as ErrorLine, ERROR_MESSAGE() as ErrorMessage;
IF @@TRANCOUNT > 0 ROLLBACK TRANSACTION; END CATCH;
IF @@TRANCOUNT > 0 COMMIT TRANSACTION; GO...případně "postaru" testovat @@ERROR <> 0 po každém statementu.
|
-
SQL Server 2005 přichází s novou funkčností tzv. Service Brokera - komplexní message-based komunikační platformy, s frontami a dalšími prvky potřebnými pro stavbu robustních SOA aplikací (Service Oriented Architecture).
Podrobněji o Service Brokeru viz SQL Books Online, pro nás je pro tuto chvíli podstatné, že prostřednictvím Service Brokera lze implementovat rozumnou podobu notifikací z SQL serveru do .NET aplikací, kdy jsou .NET aplikace SQL serverem informovány o určitých událostech (např. různých CREATE, DROP, ALTER DDL příkazech), o změnách dat (tzv. query notifications) či jakýchkoliv jiných zprávách, které si sami iniciujeme (např. z trigerů, atp.). Jak dále uvidíme, dotnetovské SqlCacheDependency či SqlDependency nejsou nic jiného, než konkrétním využitím Service Brokera.
Základní principy
Podívejme se nejprve na základní principy práce Service Brokera:
- Komunikace prostřednictvím Service Brokera je založena na zprávách (MESSAGE). Každá jednotlivá zpráva představuje příslušnou událost, příkaz, notifikaci, prostě atomickou komunikační jednotku.
- Zprávy jsou uchovávány ve frontách (QUEUE). Fronta je určitým bufferem mezi odesílatelem a příjemcem zprávy.
- Formální požadavky, které musí zpráva splňovat (např. XML schema), určuje typ zprávy (MESSAGE TYPE).
- Zprávy se posílají v konverzaci (DIALOG CONVERSATION) mezi dvěma endpointy, které představují služby (SERVICE). Cesty Mezi různými instancemi SQL serveru lze zprávy předávat pomocí routů (ROUTE).
- Formální požadavky na konverzaci/dialog určuje CONTRACT, který mj. určuje typ zpráv vyměňovaných mezi službami a směr, kterým se posílají.
- Zpráva může mimo explicitní konverzace vzniknout např. i událostí (EVENT) na straně SQL Serveru (např. DDL události CREATE, ALTER, DROP, nebo trace události, jak je známe z Profileru), nebo prostřednictvím sledování aktualizací dat (Query Notification), kdy SQL Server sleduje výsledky určitého SQL dotazu a oznámí jejich změnu (klasicky využíváno pro expiraci datové cache ASP.NET).
- Podstatné je, že Service Broker sám neinicializuje komunikaci mimo SQL Server, do .NET aplikace, naopak .NET aplikace si musí zprávu sama vyzvednout z fronty (RECEIVE), resp. může využít konstrukce WAITFOR, čímž v případě prázdné fronty pasivně vyčkává, než se zpráva objeví.
Výše uvedené není ani zdaleka vyčerpávajícím popisem fungování Service Brokeru, pro naše účely však postačuje jako úvod do problematiky.
Pro dále popisovanou funkčnost je potřeba mít Service Brokera na databázi zapnutého:
ALTER DATABASE AdventureWorks SET ENABLE_BROKER
Query Notifications - SqlNotificationRequest, SqlDependency, SqlCacheDependency
Nejběžnějším způsobem využití Service Brokera v .NET aplikaci jsou Query Notification Services - novinka SQL Serveru 2005 spočívají v možnosti vyžádat (subscribe) sledování výsledku určitého SQL dotazu (query). V případě změny pak dojde k vytvoření příslušné zprávy (message), jejímu uložení do fronty (queue), odkud si ji z naší .NET aplikace vyzvedneme a o aktualizaci na straně SQL Serveru se tak dozvíme (a může tak například dojít k vyřazení určitých dat z cache atp.).
Základní principy Query Notifications
- Query Notifications nelze aktivovat z T-SQL, ani CLR kódu hostovaného v SQL Serveru. Query Notifications lze aktivovat pouze prostřednictvím klientské aplikace, v případě .NET Frameworku prostřednictvím třídy SqlNotificationRequest (vlastnost SqlCommand.Notification).
- SqlNotificationRequest je svázán s příkazem (SqlCommand), kterému je nastaven a teprve vykonáním tohoto příkazu se provede příslušná subscription QN na straně SQL serveru.
- SQL Server sleduje pouze první změnu výsledku dotazu, tím QN končí a případný další notification request je potřeba iniciovat novým vykonáním příkazu.
- V žádném případě se nejedná o komunikaci iniciovanou SQL Serverem, nýbrž si musíme z příslušné fronty sami zprávu o aktualizaci vyzvednout, resp. si obsah fronty průběžně hlídat.
- V případě SqlDependency a SqlCacheDependency za nás vyzvednutí zprávy z příslušné fronty Service Brokera zajišťuje samotný .NET Framework prostřednictvím opakovaného volání příkazu WAITFOR (RECEIVE ...) TIMEOUT v kontinuálně otevřené samostatné SqlConnection ze samostatného threadu naší aplikace. Přesvědčit se o tom můžete v SQL Profileru.
Ilustrační příklad SqlDependency:
class Program { static void Main(string[] args) { const string connString = "Server=localhost;Database=AdventureWorks;Integrated Security=true;"; using (SqlConnection conn = new SqlConnection(connString)) { conn.Open();
SqlCommand cmd = new SqlCommand("SELECT Bonus FROM Sales.SalesPerson"); cmd.Connection = conn;
SqlDependency dependency = new SqlDependency(cmd); dependency.OnChange += new OnChangeEventHandler(dependency_OnChange); SqlDependency.Start(connString);
cmd.ExecuteNonQuery();
Console.ReadLine(); }
}
static void dependency_OnChange(object sender, SqlNotificationEventArgs e) { Console.WriteLine("Bonus changed..."); } }
SqlNotificationRequest
Pokud nám nestačí vyšší programátoská abstrakce SqlDependency či SqlCacheDependency a potřebujeme řídit Query Notifications podrobněji, můžeme využít třídu SqlNotificationRequest, resp. instanční vlastnost SqlCommand.Notification, kam instanci třídy SqlNotificationRequest přiřadíme a zajistíme tak subscribování daného příkazu do Query Notifications Service. Na rozdíl od SqlDependency pak ale musíme sami zajistit vyzvedávání zpráv z příslušné fronty a jejich zpracování (viz níže).
Požadavky na sledovaný SQL dotaz
Dotaz, ke kterému chceme QueryNotification aktivovat, musí splňovat určité nemalé restrikce:
- jména tabulek musí být uváděna včetně schématu, tedy Sales.SalesPerson, dbo.MojeTabulka, atp.
- nelze použít SELECT *, vždy musíme udělat výčet sloupců
- nelze použít agregační funkce
- nelze používat subqueries, outer-joins, self-joins
Podrobný výčet omezení viz např. http://msdn2.microsoft.com/en-US/library/ms181122.aspx.
Event Notifications
Dalším typem zpráv, které můžeme od SQL Serveru prostřednictvím Service Brokera odebírat, jsou tzv. Event Notifications. V zásadě se jedná o DDL a trace události na straně serveru, ať už na úrovni databáze nebo serveru jako celku. Např. CREATE_DATABASE, ALTER_PROCEDURE, DROP_ASSEMBLY, Audit_Login, SP_Recompile, atp. atp. Události jsou hiearchicky uspořádány a lze se přihlásit i k odběru celé skupiny najednou.
Na rozdíl od Query Notifications, kdy se nám příslušné prvky Service Brokera vytvářely převážně automaticky, zde již musíme provést základní inicializaci ručně. Na rozdíl od Query Notifications se Event Notifications celé řídí prostřednictvím T-SQL příkazů. Jednoduchý příklad by mohl vypadat nějak takto:
CREATE QUEUE MyNotifyQueue;
CREATE SERVICE MyNotifyService ON QUEUE MyNotifyQueue ( [http://schemas.microsoft.com/SQL/Notifications/PostEventNotification] )
CREATE ROUTE MyNotifyRoute WITH SERVICE_NAME = 'MyNotifyService', ADDRESS = 'LOCAL';
CREATE EVENT NOTIFICATION Notify_AlterProcedure ON DATABASE FOR ALTER_PROCEDURE TO SERVICE 'MyNotifyService', 'current database'MESSAGE TYPE pro události je již v systému pod názvem PostEventNotification připraven, vytvoříme tedy jen frontu (CREATE QUEUE), službu (CREATE SERVICE), routu (CREATE ROUTE) a pak už samotnou event-notifikace (CREATE EVENT NOTIFICATION). Tím je na straně SQL Serveru vše připraveno a nyní už můžeme jen vyzvedávat zprávy na straně .NET aplikace:
class Program { static bool done = false;
static void Main(string[] args) { Thread t = new Thread(ReceiveEvent); t.Start(); while (!done) { Thread.Sleep(1000); Console.Write("."); }
Console.ReadLine(); }
static void ReceiveEvent() { const string connString = "Server=localhost;Database=AdventureWorks;Integrated Security=true;";
using (SqlConnection conn = new SqlConnection(connString)) { conn.Open();
SqlCommand cmd = new SqlCommand(); cmd.Connection = conn; cmd.CommandText = "WAITFOR (RECEIVE TOP(1) * FROM MyNotifyQueue), TIMEOUT @Timeout"; cmd.Parameters.AddWithValue("@Timeout", 60000); // 60s
do { SqlDataReader reader = cmd.ExecuteReader(); if (reader.Read()) { OnEvent(); } done = true; } while (!done); } }
static void OnEvent() { Console.WriteLine("Event received..."); } }Obdobně jako interní implementace SqlDependency zde v samostatném threadu provádíme průběžný polling WAITFOR(RECEIVE ...) s příslušným timeoutem, zatímco v hlavním vlákně běží aplikace dál (obsluha UI, atp.). Receiving-vláknu bychom také měli nastavit vlastnost IsBackground na true, aby při skončení hlavního vlákna došlo k ukončení tohoto pollingu.
Vlastní zprávy odesíláné z SQL Serveru (např. z triggerů)
Dalším zdrojem zpráv předávaných prostřednictvím Service Brokera mohou být i zprávy, které sami vytvoříme, jejichž odeslání sami inicializujeme - např. z různých triggerů (DML i DDL), nebo uložených procedur.
Pro tuto chvíli přesahuje podrobný popis zamýšlený rozsah tohoto článku, nicméně jak je už z výše uvedeného zřejmé, nejde opět o nic jiného než o inicializaci jednotlivých prvků Service Brokera (vše T-SQL) a dále posílání zpráv prostřednictvím T-SQL příkazů BEGIN DIALOG a SEND. Samotná .NET aplikace by pak byla stejný princip, jak u Event Notifikacions, jen formát zpráv se bude lišit podle toho, jak si ho zadefinujeme (zprávy mohou být i prázdné, nebo plain-text).
Mnohé příklady odesílání zpráv přes Service Brokera viz SQL Books Online. Možná tento článek rozšířím někdy později.
|
-
Při pokusu použít Maintenance Plan Wizzard v SQL 2005 Management studiu jsem získal při pokusu o naplánování úlohy chybu, že to není možné. Po instalaci SP1 se situace změnila v tom duchu, že Maintenance Plan Wizzard nebylo možné otevřít vůbec - Studio si stěžovalo na blíže nespecifikovanou chybějící komponentu. Ve formuláři je uveden odkaz vedoucí na návod k instalaci (k ničemu), problém se vyskytuje pravděpodobně jen v x64 verzi. The action you attempted to perform on a remote instance of SQL
Server has failed because the action requires a SQL Server component
that is not installed on the remote computer. To proceed, install SQL
Server 2005 Management Tools on the remote computer, and then try
again. For more information, see "How to: Install SQL Server 2005
(Setup)" in SQL Server 2005 Books Online, or find the article on MSDN
at http://go.microsoft.com/fwlink/?LinkID=57083 .
(Microsoft.SqlServer.Management.MaintenancePlanWizard)
For help, click: http://go.microsoft.com/fwlink/?LinkID=57083 Řešení zdá se být jednoduché - stačí nainstalovat Integration services, které onu neznámou komponentu obsahují. SP2 by měl být upraven tak, aby komponenta byla součástí databázového jádra. Zdroj: Maintenance Plan problem
|
|
|
|