Online Chat

Use the window below to chat with me (if I'm online ...)

Use the edit nick field above to let me see your name.

cazh1: on Business, Information, and Technology

Thoughts and observations on the intersection of technology and business; searching for better understanding of what's relevant, where's the value, and (always) what's the goal ...

Wednesday, December 19, 2007

SQL Hack for Reporting Project Phase and Status

SQL Hack for Reporting Project Phase and Status

It's been a while since I've posted some code, but I did a nice little SQL hack today that I've been puzzling over for a while. I freely admit that I may have made this more difficult with the original data model, but the die has been cast.

Consider a single SQL table that captures project updates as comments note that this table also allows me to change the "phase" (ex. Design, Development, Test, Production) and/or Status (ex. Proposed, Red, Yellow, Green, Complete) of the project. A sample set of comments for a single project would look like this:

tblComments
ID Phase Status Comment Date Comment
1 Opportunity Proposed 7/4/2007 Lorem ipsum dolor sit amet
2 Opportunity Proposed 7/13/2007 consectetuer adipiscing elit
3 Opportunity On Hold 7/17/2007 sed diam nonummy nibh euismod
4 Opportunity On Hold 7/22/2007 tincidunt ut laoreet dolore magna
5 Concept Proposed 7/25/2007 aliquam erat volutpat Ut
6 Concept Active - Green 7/27/2007 wisi enim ad minim veniam
7 Concept Active - Green 7/28/2007 quis nostrud exerci tation
8 Concept Active - Green 7/30/2007 ullamcorper suscipit lobortis nisl ut
9 Design Active - Green 8/8/2007 aliquip ex ea commodo consequat
10 Design Active - Yellow 8/9/2007 Duis autem vel eum
11 Design Active - Yellow 8/18/2007 iriure dolor in hendrerit in
12 Design Active - Green 8/26/2007 vulputate velit esse molestie consequat
13 Design Active - Green 9/4/2007 vel illum dolore eu
14 Design Active - Green 9/14/2007 feugiat nulla facilisis at vero
15 Development Active - Green 9/21/2007 eros et accumsan et iusto
16 Development Active - Yellow 9/22/2007 odio dignissim qui blandit praesent
17 Development Active - Green 9/29/2007 luptatum zzril delenit augue duis
18 Development Active - Yellow 10/2/2007 dolore te feugait nulla facilisi
19 Development Active - Red 10/7/2007 Epsum factorial non deposit quid
20 Development Active - Red 10/14/2007 pro quo hic escorol
21 Concept Active - Red 10/18/2007 Olypian quarrels et gorilla congolium
22 Concept Active - Red 10/28/2007 sic ad nauseum Souvlaki
23 Concept Active - Yellow 11/4/2007 ignitus carborundum e pluribus unum
24 Concept Active - Green 11/6/2007 Defacto lingo est igpay
25 Concept Active - Green 11/14/2007 atinlay Marquee selectus non
26 Design Active - Green 11/19/2007 provisio incongruous feline nolo contendre
27 Design Active - Green 11/26/2007 Gratuitous octopus niacin
28 Development Active - Green 11/29/2007 sodium glutimate Quote meon
29 Implementation Active - Green 12/1/2007 an estimate et non interruptus
30 Implementation Active - Green 12/5/2007 stadium Sic tempus fugit
31 Implementation Active - Green 12/13/2007 esperanto hiccup estrogen Glorious
32 Implementation Active - Green 12/16/2007 baklava ex librus hup hey
33 Operation Active - Green 12/21/2007 ad infinitum Non sequitur
34 Operation Active - Green 12/23/2007 condominium facile et geranium incognito
35 Operation Completed 12/30/2007 Epsum factorial non
36 Operation Completed 1/4/2008 deposit quid pro quo hic escorol Marquee

It's a project's life cycle thru multiple phases; but notice that this project starts on 7/4, gets up to Development, then a new requirement comes along (10/18) and forces it to go back to Concept phase to rethink. When querying this, I want a result set to show one record for the first time the project goes to the next phase ... something like this ...

ID Phase Comment Date
1 Opportunity 7/4/2007
5 Concept 7/25/2007
9 Design 8/8/2007
15 Development 9/21/2007
21 Concept 10/18/2007
26 Design 11/19/2007
28 Development 11/29/2007
29 Implementation 12/1/2007
33 Operation 12/21/2007

My first attempt was using GROUP BY in a select statement, but I couldn't get that to work - it doesn't do subtotals. Also, COMPUTE BY was not an option - I want to create a view, and have the results sitting there for me. I asked my favorite SQL guru, who solved the problem using cursors, but that didn't get me to my view.

I ended up solving using subqueries, comparing the current record to the previous record, and setting a flag whenever the Phase changes ... like this ...

  • Add a User-defined function ...

CREATE FUNCTION PhaseChangeFlag (@CurrPhase varchar(10),
@PrevPhase varchar(10))
RETURNS int
AS
BEGIN
IF @CurrPhase <> @PrevPhase RETURN(1)
RETURN(0)
END

  • Create a SELECT statement that sub-queries to find the previous record

SELECT ID, Phase, PhaseChangeFlag( Phase, (
SELECT prv_rec.Phase
FROM tblComments AS prv_rec
WHERE prv_rec.ID = (
SELECT MAX(chk_rec.ID) <<<< hack!
FROM tblComments AS chk_rec
WHERE chk_rec.ID < cur_rec.ID)
)
) AS PhaseChange
FROM tblComments cur_rec

Now I get a result set that looks like this ...

ID Phase PhaseChange Comment Date
1 Opportunity 1 7/4/2007
2 Opportunity 0 7/13/2007
3 Opportunity 0 7/17/2007
4 Opportunity 0 7/22/2007
5 Concept 1 7/25/2007
6 Concept 0 7/27/2007
7 Concept 0 7/28/2007
8 Concept 0 7/30/2007
9 Design 1 8/8/2007
10 Design 0 8/9/2007
11 Design 0 8/18/2007
12 Design 0 8/26/2007
13 Design 0 9/4/2007
14 Design 0 9/14/2007
15 Development 1 9/21/2007
16 Development 0 9/22/2007
17 Development 0 9/29/2007
18 Development 0 10/2/2007
19 Development 0 10/7/2007
20 Development 0 10/14/2007
21 Concept 1 10/18/2007
22 Concept 0 10/28/2007
23 Concept 0 11/4/2007
24 Concept 0 11/6/2007
25 Concept 0 11/14/2007
26 Design 1 11/19/2007
27 Design 0 11/26/2007
28 Development 1 11/29/2007
29 Implementation 1 12/1/2007
30 Implementation 0 12/5/2007
31 Implementation 0 12/13/2007
32 Implementation 0 12/16/2007
33 Operation 1 12/21/2007
34 Operation 0 12/23/2007
35 Operation 0 12/30/2007
36 Operation 0 1/4/2008

So I can easily write a query from this view that only shows records where PhaseChange = 1.

Now I can create web queries and reports that show project Phase and/or Status as of any date - significant progress towards the idea of transparency ... aging projects, how many did we close for you this quarter, stuff like that.

I'm not too sure how "expensive" this query is, but the database is small.

Technorati Tags: ,

<< blog home