Avatar billede midas_dk Nybegynder
07. september 2004 - 16:59 Der er 8 kommentarer og
1 løsning

Spooky SQL problem

Har en database med tre tabeller:

Users (alle brugere på f.eks. et website)
Roles (alle tilgængelige roller for en bruger - f.eks. Admin, Author osv.)
Areas (alle tilgængelige områder for en rolle - for en Author kunne det f.eks. være "Forside" og "Underforside 1")
Role_Area (kæder roller og områder sammen)

Lidt langt script, men her kommer det:

CREATE TABLE [dbo].[Areas] (
    [GUID]  uniqueidentifier ROWGUIDCOL  NOT NULL ,
    [AreaName] [nvarchar] (50) COLLATE Danish_Norwegian_CI_AS NOT NULL
) ON [PRIMARY]
GO

CREATE TABLE [dbo].[Role_Area] (
    [GUID]  uniqueidentifier ROWGUIDCOL  NOT NULL ,
    [FK_Role] [uniqueidentifier] NOT NULL ,
    [FK_Area] [uniqueidentifier] NOT NULL
) ON [PRIMARY]
GO

CREATE TABLE [dbo].[Roles] (
    [GUID]  uniqueidentifier ROWGUIDCOL  NOT NULL ,
    [Rolename] [nvarchar] (50) COLLATE Danish_Norwegian_CI_AS NOT NULL
) ON [PRIMARY]
GO

CREATE TABLE [dbo].[Users] (
    [UserID] [int] IDENTITY (1, 1) NOT NULL ,
    [FK_GUID] [uniqueidentifier] NOT NULL ,
    [FK_Role] [uniqueidentifier] NOT NULL
) ON [PRIMARY]
GO

ALTER TABLE [dbo].[Areas] ADD
    CONSTRAINT [DF_Areas_GUID] DEFAULT (newid()) FOR [GUID],
    CONSTRAINT [PK_Areas] PRIMARY KEY  CLUSTERED
    (
        [GUID]
    )  ON [PRIMARY]
GO

ALTER TABLE [dbo].[Role_Area] ADD
    CONSTRAINT [DF_Usr_Area_GUID] DEFAULT (newid()) FOR [GUID],
    CONSTRAINT [PK_Usr_Area] PRIMARY KEY  CLUSTERED
    (
        [GUID]
    )  ON [PRIMARY]
GO

ALTER TABLE [dbo].[Roles] ADD
    CONSTRAINT [DF_Usrs_GUID] DEFAULT (newid()) FOR [GUID],
    CONSTRAINT [PK_Usrs] PRIMARY KEY  CLUSTERED
    (
        [GUID]
    )  ON [PRIMARY]
GO

ALTER TABLE [dbo].[Users] ADD
    CONSTRAINT [PK_Users] PRIMARY KEY  CLUSTERED
    (
        [UserID]
    )  ON [PRIMARY]
GO

ALTER TABLE [dbo].[Role_Area] ADD
    CONSTRAINT [FK_Usr_Area_Areas] FOREIGN KEY
    (
        [FK_Area]
    ) REFERENCES [dbo].[Areas] (
        [GUID]
    ),
    CONSTRAINT [FK_Usr_Area_Usrs] FOREIGN KEY
    (
        [FK_Role]
    ) REFERENCES [dbo].[Roles] (
        [GUID]
    )
GO

ALTER TABLE [dbo].[Users] ADD
    CONSTRAINT [FK_Users_Roles] FOREIGN KEY
    (
        [FK_Role]
    ) REFERENCES [dbo].[Roles] (
        [GUID]
    )
GO

- mit problem er, at jeg ud fra et eller flere områder skal finde netop den rolle, der har netop det eller de områder.

Eks.: jeg har et område (ID: "OmrForside"). Dette område er en "del" af tre roller ("AuthorForside", "AuthorDetMeste" og "Admin"), hvoraf den ene rolle ("AuthorForside") kun "består" af "OmrForside". I mit admin-modul har jeg for en bruger valgt, at han kun skal have adgang til "OmrForside" - hvordan finder jeg Rolle-ID'et på netop den rolle, som kun består af "OmrForside"?

- og hvordan finder jeg rollen, hvis nu jeg for en bruger vælger at tilknytte to eller flere områder?

Ligegyldigt hvordan jeg Select'er, får jeg smidt alle tre roller tilbage! Jeg har stirret mig blind på det, er jeg sikker på, og nu gider jeg ikke glo dumt på det mere.
Avatar billede midas_dk Nybegynder
07. september 2004 - 17:03 #1
P.S. Har fri nu for i dag, så jeg kan ikke kommentere osv. før i morgen...
Avatar billede arne_v Ekspert
07. september 2004 - 20:22 #2
Prøv:

SELECT Role.RoleName
FROM Area,Role_Area,Role
WHERE Area.GUID=Role_Area.FK_Area AND Role.GUID=Role_Area.FK_role AND
      Area.AreaName='OmrForside' AND Role.GUID NOT IN
      (SELECT DISTINCT FK_role FROM Role_Area WHERE FK_area <> Area.GUID)
Avatar billede arne_v Ekspert
07. september 2004 - 20:28 #3
Den næste er lidt sværere.

Jeg tror at du skal lave noget a la (skitseret pseudo SQL):

WHERE
role IN (SELECT roles that have area 1) AND
...
role IN (SELECT roles that have area N) AND
role NOT IN (SELECT roles that have areas outside area 1..N)
Avatar billede arne_v Ekspert
07. september 2004 - 21:02 #4
Iøvrigt vælger man normalt at designed det man kan tildele en bruger flere roller.

Og så knytte roller til en "ting".

Så har man ikke dette problem.
Avatar billede midas_dk Nybegynder
08. september 2004 - 08:55 #5
Halløjsa...

Den første statement (20:22:44) returnerer det helt rigtige! Takker!

Mht. til 20:28:01, mener du så som udbygning til den første statement? Altså at man smækker et par "AND"'s oveni for hvert område?

- og mht. 21:02:32, har jeg ikke noget valg. En rolle er i sidste ende repræsenteret af en Windows-bruger, som jeg bruger til login - og sådan er systemet designet :-/
Avatar billede arne_v Ekspert
08. september 2004 - 09:13 #6
Ja 20:28:01 er en udbygning af 20:22:44


Area.AreaName='OmrForside'

skal erstattes af en hel række IN (SELECT ...) AND og

      (SELECT DISTINCT FK_role FROM Role_Area WHERE FK_area <> Area.GUID)

skal have en række AND <> ind i
Avatar billede midas_dk Nybegynder
08. september 2004 - 09:20 #7
Hm. Det lyder rigtigt - selvom det ikke bliver noget kønt SQL ;-)

Gi' mig en halv time til at teste det, så vender jeg tilbage!
Avatar billede midas_dk Nybegynder
08. september 2004 - 10:05 #8
Det virker sørme... Tak for hjælpen - lægger du et svar?
Avatar billede arne_v Ekspert
08. september 2004 - 10:52 #9
svar
Avatar billede Ny bruger Nybegynder

Din løsning...

Tilladte BB-code-tags: [b]fed[/b] [i]kursiv[/i] [u]understreget[/u] Web- og emailadresser omdannes automatisk til links. Der sættes "nofollow" på alle links.

Loading billede Opret Preview
Kategori
Computerworld tilbyder specialiserede kurser i database-management

Log ind eller opret profil

Hov!

For at kunne deltage på Computerworld Eksperten skal du være logget ind.

Det er heldigvis nemt at oprette en bruger: Det tager to minutter og du kan vælge at bruge enten e-mail, Facebook eller Google som login.

Du kan også logge ind via nedenstående tjenester