Java Web Services via REST bereitstellen
ulrich
2014-11-11 f88c95ed978c427b33730442e03f8a1565b94e32
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
/*
    Transit - Remote procedure calls made simple
    Copyright (c) 2012  Ulrich Hilger
 
    This program is free software: you can redistribute it and/or modify
    it under the terms of the GNU Affero General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.
 
    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU Affero General Public License for more details.
 
    You should have received a copy of the GNU Affero General Public License
    along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/
 
package de.uhilger.transit.web;
 
import java.util.List;
import java.util.Map;
 
import de.uhilger.baselink.PersistenceManager;
 
import java.util.logging.Level;
import java.util.logging.Logger;
 
 
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
 
/**
 * Die Klasse RechteDb implementiert die Schnittstelle RechtePruefer 
 * auf der Grundlage von Benutzerinformationen in einer Datenbank.
 * 
 * @author Copyright (c) Ulrich Hilger, http://uhilger.de
 * @author Published under the terms and conditions of
 * the <a href="http://www.gnu.org/licenses/agpl-3.0" target="_blank">GNU Affero General Public License</a>
 * 
 * @version 1, September 16, 2012
 */
public class RechteDb extends PersistenceManager implements RechtePruefer {
 
  /** Der Name des Datenbanktreibers, der verwendet werden soll, falls keiner angegeben wurde */
  public static final String DEFAULT_DRIVER = "org.apache.derby.jdbc.EmbeddedDriver";
  
  /**
   * Ein Objekt der Klasse <code>RechteDb</code> erzeugen.
   * 
   * Beim Aufruf dieses Konstruktors muss anschliessend eine der folgenden 
   * Methoden gerufen werden:
   * setDatabase(DEFAULT_DRIVER, url);
   * setDatabase(driverName, url);
   * setDataSourceName(dsName);
   */
  public RechteDb() {
    super();
  }
  
  /**
   * Ein Objekt der Klasse <code>RechteDb</code> erzeugen.
   * 
   * @param driverName Name des Datenbanktreibers
   * @param url der Datenbank
   */
  public RechteDb(String driverName, String url) {
    this(driverName, null, url);
  }
 
  /**
   * Ein Objekt der Klasse <code>RechteDb</code> erzeugen.
   * 
   * @param dsName  Name der DataSource
   */
  public RechteDb(String dsName) {
    this(null, dsName, null);
  }
 
    /**
   * Ein Objekt der Klasse <code>RechteDb</code> erzeugen.
     * 
     * @param driverName
     *          Name des Datenbanktreibers fuer den Fall dass keine DataSource
   *          verwendet werden soll, sonst kann dieser Parameter null sein
   * 
   * @param dsName
   *          Name der JDBC DataSource, wie er im Kontext des Servlet Containers 
   *          erscheint, z.B. jdbc/AuthorityDB
   *          
     * @param url
     *          der JDBC connect string, fuer den Fall dass keine DataSource
     *          verwendet werden soll, sonst kann dieser Parameter null sein
     */
  @SuppressWarnings("rawtypes")
    private RechteDb(String driverName, String dsName, String url) {
        //super(productionMode, dsName, url);
    super();
    try {
      if (url != null && !url.isEmpty()) {
        Logger.getLogger(Logger.GLOBAL_LOGGER_NAME).finest("setting url to " + url);
        if(driverName == null) {
          setDatabase(DEFAULT_DRIVER, url);
        } else {
          setDatabase(driverName, url);
        }
      } else {
        Logger.getLogger(Logger.GLOBAL_LOGGER_NAME).finest("setting data source to " + dsName);
        setDataSourceName(dsName);
      }
    } catch (Exception ex) {
      Logger.getLogger(Logger.GLOBAL_LOGGER_NAME).log(Level.SEVERE, ex.getMessage(), ex);
    }
    }
  
  /*
  
    select m.* from app.user_roles as ur, app.methoden_rechte as mr, app.methoden as m
    where ur.role_name = mr.role_name 
    and mr.mt_id = m.mt_id
    and ur.user_name = [name]
    
    gibt eine Tabelle mit folgenden Spalten zurueck:
    mt_id, mt_klasse, mt_name
    die alle Methoden des Benutzers enthaelt
  
  */
  
  /**
   * Die Rechte eines Benutzers ermitteln
   * 
   * @param Name des Benutzers
   * @return eine Liste mit Berechtigungen des Benutzers
   */
  @SuppressWarnings("rawtypes")  
  public List getRechte(String benutzerName) {
    StringBuffer sql = new StringBuffer();
    sql.append("select m.* from app.user_roles as ur");
    sql.append(", app.methoden_rechte as mr, app.methoden as m");
    sql.append(" where ur.role_name = mr.role_name");
    sql.append(" and mr.mt_id = m.mt_id");
    sql.append(" and ur.user_name = '");
    sql.append(benutzerName);
    sql.append("'");
    List rechte = select(sql.toString());
    return rechte;
  }
  
  /**
   * Pruefen ob ein Methodenaufruf erlaubt ist
   * 
   * @param klasse  vollstaendiger Name der Klasse, die die auszufuehrende Methode enthaelt
   * @param methode Name der Methode, die ausgefuehrt werden soll
   * @param benutzerName  Name des Benutzerkontos, dessen Benutzer die Methode ausfuehren moechte
   * 
   * @return true, wenn der Benutzer die Methode ausfuehren darf, false wenn nicht
   */
  @SuppressWarnings("unchecked")  
  public boolean istErlaubt(ServletRequest request, ServletResponse response, String klasse, String methode, String benutzerName) {
    boolean erlaubt = false;    
    List<Map> rechte = getRechte(benutzerName);
    if(rechte != null) {
      for(int i = 0; i < rechte.size() && !erlaubt; i++) {
        String rKlasse = (String) rechte.get(i).get("MT_KLASSE");
        String rMethode = (String) rechte.get(i).get("MT_NAME");
        /*
          Verwendung von Regular Expressions, ein Muster fuer eine Klasse 
          ist z.B. de.uhilger..*
          
          eine Methode koennte so lauten
          .*
          oder
          get.*
          
          Der Punkt steht fuer ein beliebiges Zeichen,
          das * steht fuer kein Mal oder beliebig oft das davor befindliche Zeichen
          .* bedeutet 'ab dieser Stelle keinmal oder beliebig oft ein beliebiges Zeichen'
          
          + bedeutet stattdessen ein oder mehrmals das davor befindliche Zeichen 
        */
        erlaubt = klasse.matches(rKlasse) && methode.matches(rMethode);
      }
    }
    return erlaubt;    
  }
  
}